veloren/common/sys/src/controller.rs

117 lines
4.4 KiB
Rust
Raw Normal View History

use common::{
comp::{BuffChange, ControlEvent, Controller},
2020-03-07 21:03:10 +00:00
event::{EventBus, LocalEvent, ServerEvent},
metrics::SysMetrics,
resources::DeltaTime,
span,
uid::{Uid, UidAllocator},
2019-06-09 14:20:20 +00:00
};
use specs::{
saveload::{Marker, MarkerAllocator},
Entities, Join, Read, ReadExpect, ReadStorage, System, WriteStorage,
};
2020-11-03 22:46:07 +00:00
use vek::*;
2019-06-09 14:20:20 +00:00
2020-03-07 21:03:10 +00:00
// const CHARGE_COST: i32 = 200;
// const ROLL_COST: i32 = 30;
2019-06-09 14:20:20 +00:00
pub struct Sys;
2019-06-09 14:20:20 +00:00
impl<'a> System<'a> for Sys {
#[allow(clippy::type_complexity)]
2019-06-09 14:20:20 +00:00
type SystemData = (
Entities<'a>,
Read<'a, UidAllocator>,
Read<'a, EventBus<ServerEvent>>,
Read<'a, EventBus<LocalEvent>>,
Read<'a, DeltaTime>,
ReadExpect<'a, SysMetrics>,
2019-07-21 16:50:13 +00:00
WriteStorage<'a, Controller>,
2019-10-11 04:30:34 +00:00
ReadStorage<'a, Uid>,
2019-06-09 14:20:20 +00:00
);
2019-06-09 14:20:20 +00:00
fn run(
&mut self,
2020-03-20 16:15:09 +00:00
(
entities,
uid_allocator,
server_bus,
_local_bus,
2020-03-24 07:38:16 +00:00
_dt,
sys_metrics,
2020-03-20 16:15:09 +00:00
mut controllers,
uids,
): Self::SystemData,
2019-06-09 14:20:20 +00:00
) {
let start_time = std::time::Instant::now();
span!(_guard, "run", "controller::Sys::run");
let mut server_emitter = server_bus.emitter();
2020-03-23 17:13:44 +00:00
for (entity, _uid, controller) in (&entities, &uids, &mut controllers).join() {
2020-03-24 07:38:16 +00:00
let mut inputs = &mut controller.inputs;
// Note(imbris): I avoided incrementing the duration with inputs.tick() because
// this is being done manually in voxygen right now so it would double up on
// speed of time.
// Perhaphs the duration aspects of inputs could be
// calculated exclusively on the server (since the client can't be
// trusted anyway). It needs to be considered if these calculations
// being on the client are critical for responsiveness/client-side prediction.
inputs.tick_freshness();
2019-12-20 13:30:37 +00:00
// Update `inputs.move_dir`.
inputs.move_dir = if inputs.move_dir.magnitude_squared() > 1.0 {
// Cap move_dir to 1
inputs.move_dir.normalized()
} else {
inputs.move_dir
};
2020-11-03 22:46:07 +00:00
inputs.move_z = inputs.move_z.clamped(-1.0, 1.0);
2019-06-09 14:20:20 +00:00
// Process other controller events
for event in controller.events.drain(..) {
match event {
ControlEvent::Mount(mountee_uid) => {
if let Some(mountee_entity) =
uid_allocator.retrieve_entity_internal(mountee_uid.id())
{
server_emitter.emit(ServerEvent::Mount(entity, mountee_entity));
}
},
ControlEvent::RemoveBuff(buff_id) => {
server_emitter.emit(ServerEvent::Buff {
entity,
buff_change: BuffChange::RemoveFromController(buff_id),
});
},
ControlEvent::Unmount => server_emitter.emit(ServerEvent::Unmount(entity)),
2020-10-07 02:23:20 +00:00
ControlEvent::EnableLantern => {
server_emitter.emit(ServerEvent::EnableLantern(entity))
},
ControlEvent::DisableLantern => {
server_emitter.emit(ServerEvent::DisableLantern(entity))
},
ControlEvent::Interact(npc_uid) => {
if let Some(npc_entity) =
uid_allocator.retrieve_entity_internal(npc_uid.id())
{
server_emitter.emit(ServerEvent::NpcInteract(entity, npc_entity));
}
},
ControlEvent::InventoryManip(manip) => {
server_emitter.emit(ServerEvent::InventoryManip(entity, manip.into()));
2020-03-24 07:38:16 +00:00
},
ControlEvent::GroupManip(manip) => {
server_emitter.emit(ServerEvent::GroupManip(entity, manip))
},
2020-03-24 07:38:16 +00:00
ControlEvent::Respawn => server_emitter.emit(ServerEvent::Respawn(entity)),
}
}
2019-06-09 14:20:20 +00:00
}
sys_metrics.controller_ns.store(
start_time.elapsed().as_nanos() as u64,
std::sync::atomic::Ordering::Relaxed,
);
2019-06-09 14:20:20 +00:00
}
}