veloren/common/src/sys/agent.rs

119 lines
4.8 KiB
Rust
Raw Normal View History

use crate::comp::{Agent, Controller, Pos};
use rand::{seq::SliceRandom, thread_rng};
use specs::{Entities, Join, ReadStorage, System, WriteStorage};
use vek::*;
2019-06-09 19:33:20 +00:00
/// This system will allow NPCs to modify their controller
pub struct Sys;
impl<'a> System<'a> for Sys {
type SystemData = (
Entities<'a>,
WriteStorage<'a, Agent>,
ReadStorage<'a, Pos>,
2019-06-09 14:20:20 +00:00
WriteStorage<'a, Controller>,
);
fn run(&mut self, (entities, mut agents, positions, mut controllers): Self::SystemData) {
2019-08-02 18:22:51 +00:00
for (entity, agent, pos, controller) in
2019-06-09 14:20:20 +00:00
(&entities, &mut agents, &positions, &mut controllers).join()
{
match agent {
Agent::Wanderer(bearing) => {
*bearing += Vec2::new(rand::random::<f32>() - 0.5, rand::random::<f32>() - 0.5)
* 0.1
- *bearing * 0.01
- pos.0 * 0.0002;
if bearing.magnitude_squared() != 0.0 {
2019-06-09 14:20:20 +00:00
controller.move_dir = bearing.normalized();
}
}
Agent::Pet { target, offset } => {
// Run towards target.
match positions.get(*target) {
Some(tgt_pos) => {
let tgt_pos = tgt_pos.0 + *offset;
if tgt_pos.z > pos.0.z + 1.0 {
2019-06-29 20:51:22 +00:00
controller.jump = true;
}
// Move towards the target.
let dist: f32 = Vec2::from(tgt_pos - pos.0).magnitude();
2019-06-09 14:20:20 +00:00
controller.move_dir = if dist > 5.0 {
Vec2::from(tgt_pos - pos.0).normalized()
} else if dist < 1.5 && dist > 0.0 {
Vec2::from(pos.0 - tgt_pos).normalized()
} else {
Vec2::zero()
};
}
2019-06-09 14:20:20 +00:00
_ => controller.move_dir = Vec2::zero(),
}
// Change offset occasionally.
if rand::random::<f32>() < 0.003 {
*offset =
Vec2::new(rand::random::<f32>() - 0.5, rand::random::<f32>() - 0.5)
* 10.0;
}
}
Agent::Enemy { bearing, target } => {
2019-08-02 20:35:03 +00:00
const SIGHT_DIST: f32 = 30.0;
let choose_new = match target.map(|tgt| positions.get(tgt)).flatten() {
Some(tgt_pos) => {
let dist = Vec2::<f32>::from(tgt_pos.0 - pos.0).magnitude();
if dist < 2.0 {
2019-08-02 19:52:46 +00:00
controller.move_dir = Vec2::zero();
2019-08-02 19:41:38 +00:00
if rand::random::<f32>() < 0.05 {
2019-06-29 20:51:22 +00:00
controller.attack = true;
2019-08-02 19:41:38 +00:00
} else {
controller.attack = false;
}
false
2019-08-02 20:35:03 +00:00
} else if dist < SIGHT_DIST {
2019-06-09 14:20:20 +00:00
controller.move_dir =
2019-08-02 19:41:38 +00:00
Vec2::<f32>::from(tgt_pos.0 - pos.0).normalized();
false
} else {
true
}
}
None => {
*bearing +=
Vec2::new(rand::random::<f32>() - 0.5, rand::random::<f32>() - 0.5)
* 0.1
2019-08-02 19:41:38 +00:00
- *bearing * 0.005;
2019-08-02 19:41:38 +00:00
controller.move_dir = if bearing.magnitude_squared() > 0.1 {
bearing.normalized()
} else {
Vec2::zero()
};
true
}
};
2019-08-02 19:41:38 +00:00
if choose_new && rand::random::<f32>() < 0.1 {
let entities = (&entities, &positions)
.join()
2019-08-02 18:22:51 +00:00
.filter(|(e, e_pos)| {
2019-08-02 20:35:03 +00:00
Vec2::<f32>::from(e_pos.0 - pos.0).magnitude() < SIGHT_DIST
2019-08-02 18:22:51 +00:00
&& *e != entity
})
.map(|(e, _)| e)
.collect::<Vec<_>>();
let mut rng = thread_rng();
*target = (&entities).choose(&mut rng).cloned();
}
}
}
}
}
}