2020-11-23 15:39:03 +00:00
|
|
|
#![allow(dead_code)] // TODO: Remove this when rtsim is fleshed out
|
|
|
|
|
2020-11-11 23:59:09 +00:00
|
|
|
use super::*;
|
|
|
|
use common::{
|
2020-11-15 01:40:23 +00:00
|
|
|
comp,
|
2021-01-08 19:12:09 +00:00
|
|
|
comp::inventory::loadout_builder::LoadoutBuilder,
|
2020-11-11 23:59:09 +00:00
|
|
|
event::{EventBus, ServerEvent},
|
2020-12-01 00:28:00 +00:00
|
|
|
resources::DeltaTime,
|
2020-11-15 01:40:23 +00:00
|
|
|
terrain::TerrainGrid,
|
2020-11-11 23:59:09 +00:00
|
|
|
};
|
2020-11-15 01:40:23 +00:00
|
|
|
use specs::{Join, Read, ReadExpect, ReadStorage, System, WriteExpect, WriteStorage};
|
2020-11-11 23:59:09 +00:00
|
|
|
use std::sync::Arc;
|
|
|
|
|
2020-11-12 21:31:28 +00:00
|
|
|
const ENTITY_TICK_PERIOD: u64 = 30;
|
|
|
|
|
2020-11-11 23:59:09 +00:00
|
|
|
pub struct Sys;
|
|
|
|
impl<'a> System<'a> for Sys {
|
2020-11-23 15:39:03 +00:00
|
|
|
#[allow(clippy::type_complexity)]
|
2020-11-11 23:59:09 +00:00
|
|
|
type SystemData = (
|
2020-11-12 21:31:28 +00:00
|
|
|
Read<'a, DeltaTime>,
|
2020-11-11 23:59:09 +00:00
|
|
|
Read<'a, EventBus<ServerEvent>>,
|
|
|
|
WriteExpect<'a, RtSim>,
|
|
|
|
ReadExpect<'a, TerrainGrid>,
|
|
|
|
ReadExpect<'a, Arc<world::World>>,
|
2020-11-12 21:31:28 +00:00
|
|
|
ReadExpect<'a, world::IndexOwned>,
|
2020-11-12 00:08:28 +00:00
|
|
|
ReadStorage<'a, comp::Pos>,
|
|
|
|
ReadStorage<'a, RtSimEntity>,
|
2020-11-12 21:31:28 +00:00
|
|
|
WriteStorage<'a, comp::Agent>,
|
2020-11-11 23:59:09 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&mut self,
|
|
|
|
(
|
2020-11-12 21:31:28 +00:00
|
|
|
dt,
|
2020-11-11 23:59:09 +00:00
|
|
|
server_event_bus,
|
|
|
|
mut rtsim,
|
|
|
|
terrain,
|
|
|
|
world,
|
2020-11-23 15:39:03 +00:00
|
|
|
_index,
|
2020-11-12 00:08:28 +00:00
|
|
|
positions,
|
|
|
|
rtsim_entities,
|
2020-11-12 21:31:28 +00:00
|
|
|
mut agents,
|
2020-11-11 23:59:09 +00:00
|
|
|
): Self::SystemData,
|
|
|
|
) {
|
|
|
|
let rtsim = &mut *rtsim;
|
2020-11-12 21:31:28 +00:00
|
|
|
rtsim.tick += 1;
|
2020-11-11 23:59:09 +00:00
|
|
|
|
2020-11-12 00:08:28 +00:00
|
|
|
// Update rtsim entities
|
2020-11-11 23:59:09 +00:00
|
|
|
// TODO: don't update all of them each tick
|
|
|
|
let mut to_reify = Vec::new();
|
|
|
|
for (id, entity) in rtsim.entities.iter_mut() {
|
|
|
|
if entity.is_loaded {
|
2020-11-12 21:31:28 +00:00
|
|
|
// No load-specific behaviour yet
|
2020-11-15 01:40:23 +00:00
|
|
|
} else if rtsim
|
|
|
|
.chunks
|
|
|
|
.chunk_at(entity.pos.xy())
|
|
|
|
.map(|c| c.is_loaded)
|
|
|
|
.unwrap_or(false)
|
|
|
|
{
|
2020-11-11 23:59:09 +00:00
|
|
|
to_reify.push(id);
|
2020-11-12 21:31:28 +00:00
|
|
|
} else {
|
|
|
|
// Simulate behaviour
|
|
|
|
if let Some(travel_to) = entity.controller.travel_to {
|
|
|
|
// Move towards target at approximate character speed
|
2020-11-15 01:40:23 +00:00
|
|
|
entity.pos += Vec3::from(
|
|
|
|
(travel_to.xy() - entity.pos.xy())
|
|
|
|
.try_normalized()
|
|
|
|
.unwrap_or_else(Vec2::zero)
|
|
|
|
* entity.get_body().max_speed_approx()
|
|
|
|
* entity.controller.speed_factor,
|
|
|
|
) * dt.0;
|
2020-11-12 21:31:28 +00:00
|
|
|
}
|
|
|
|
|
2020-11-15 01:40:23 +00:00
|
|
|
if let Some(alt) = world
|
|
|
|
.sim()
|
|
|
|
.get_alt_approx(entity.pos.xy().map(|e| e.floor() as i32))
|
|
|
|
{
|
2020-11-12 21:31:28 +00:00
|
|
|
entity.pos.z = alt;
|
|
|
|
}
|
2020-11-11 23:59:09 +00:00
|
|
|
}
|
|
|
|
|
2020-11-12 21:31:28 +00:00
|
|
|
// Tick entity AI
|
|
|
|
if entity.last_tick + ENTITY_TICK_PERIOD <= rtsim.tick {
|
|
|
|
entity.tick(&terrain, &world);
|
|
|
|
entity.last_tick = rtsim.tick;
|
2020-11-11 23:59:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut server_emitter = server_event_bus.emitter();
|
|
|
|
for id in to_reify {
|
|
|
|
rtsim.reify_entity(id);
|
|
|
|
let entity = &rtsim.entities[id];
|
2020-11-15 01:40:23 +00:00
|
|
|
let spawn_pos = terrain
|
|
|
|
.find_space(entity.pos.map(|e| e.floor() as i32))
|
|
|
|
.map(|e| e as f32)
|
|
|
|
+ Vec3::new(0.5, 0.5, 0.0);
|
2020-11-12 21:31:28 +00:00
|
|
|
let body = entity.get_body();
|
2020-11-11 23:59:09 +00:00
|
|
|
server_emitter.emit(ServerEvent::CreateNpc {
|
2020-11-12 21:31:28 +00:00
|
|
|
pos: comp::Pos(spawn_pos),
|
2021-01-05 01:04:01 +00:00
|
|
|
stats: comp::Stats::new(entity.get_name(), body).with_level(entity.get_level()),
|
2020-11-11 23:59:09 +00:00
|
|
|
health: comp::Health::new(body, 10),
|
2021-01-05 01:04:01 +00:00
|
|
|
loadout: match body {
|
2021-01-08 19:12:09 +00:00
|
|
|
comp::Body::Humanoid(_) => entity.get_loadout(),
|
|
|
|
_ => LoadoutBuilder::new().build(),
|
2021-01-05 01:04:01 +00:00
|
|
|
},
|
2020-11-11 23:59:09 +00:00
|
|
|
body,
|
2021-01-05 01:04:01 +00:00
|
|
|
agent: Some(comp::Agent::new(
|
|
|
|
None,
|
|
|
|
matches!(body, comp::Body::Humanoid(_)),
|
|
|
|
&body,
|
|
|
|
false,
|
|
|
|
)),
|
|
|
|
alignment: match body {
|
|
|
|
comp::Body::Humanoid(_) => comp::Alignment::Npc,
|
|
|
|
_ => comp::Alignment::Wild,
|
|
|
|
},
|
2020-11-11 23:59:09 +00:00
|
|
|
scale: comp::Scale(1.0),
|
|
|
|
drop_item: None,
|
|
|
|
home_chunk: None,
|
|
|
|
rtsim_entity: Some(RtSimEntity(id)),
|
|
|
|
});
|
|
|
|
}
|
2020-11-12 00:08:28 +00:00
|
|
|
|
|
|
|
// Update rtsim with real entity data
|
2020-11-12 21:31:28 +00:00
|
|
|
for (pos, rtsim_entity, agent) in (&positions, &rtsim_entities, &mut agents).join() {
|
|
|
|
rtsim.entities.get_mut(rtsim_entity.0).map(|entity| {
|
|
|
|
entity.pos = pos.0;
|
|
|
|
agent.rtsim_controller = entity.controller.clone();
|
|
|
|
});
|
2020-11-12 00:08:28 +00:00
|
|
|
}
|
2020-11-11 23:59:09 +00:00
|
|
|
}
|
|
|
|
}
|