2020-12-01 00:28:00 +00:00
|
|
|
use common::{
|
2019-12-20 13:30:37 +00:00
|
|
|
comp::{
|
2020-12-01 00:28:00 +00:00
|
|
|
Attacking, Beam, Body, CharacterState, Controller, Energy, Health, Loadout, Mounting, Ori,
|
|
|
|
PhysicsState, Pos, StateUpdate, Vel,
|
2019-12-20 13:30:37 +00:00
|
|
|
},
|
2019-12-26 14:43:59 +00:00
|
|
|
event::{EventBus, LocalEvent, ServerEvent},
|
2020-09-14 12:56:05 +00:00
|
|
|
metrics::SysMetrics,
|
2020-12-01 00:28:00 +00:00
|
|
|
resources::DeltaTime,
|
2020-08-26 08:11:31 +00:00
|
|
|
span,
|
2020-12-01 00:28:00 +00:00
|
|
|
states::{
|
|
|
|
self,
|
|
|
|
behavior::{CharacterBehavior, JoinData, JoinTuple},
|
|
|
|
},
|
2020-12-13 17:11:55 +00:00
|
|
|
uid::{Uid, UidAllocator},
|
2019-12-20 13:30:37 +00:00
|
|
|
};
|
|
|
|
|
2020-12-01 00:28:00 +00:00
|
|
|
use specs::{Entities, Join, LazyUpdate, Read, ReadExpect, ReadStorage, System, WriteStorage};
|
2020-03-07 18:15:02 +00:00
|
|
|
|
2020-03-24 07:38:16 +00:00
|
|
|
fn incorporate_update(tuple: &mut JoinTuple, state_update: StateUpdate) {
|
2020-07-06 05:56:02 +00:00
|
|
|
// TODO: if checking equality is expensive use optional field in StateUpdate
|
|
|
|
if tuple.2.get_unchecked() != &state_update.character {
|
|
|
|
*tuple.2.get_mut_unchecked() = state_update.character
|
|
|
|
};
|
2020-03-24 07:38:16 +00:00
|
|
|
*tuple.3 = state_update.pos;
|
|
|
|
*tuple.4 = state_update.vel;
|
|
|
|
*tuple.5 = state_update.ori;
|
2020-07-06 05:56:02 +00:00
|
|
|
// Note: might be changed every tick by timer anyway
|
|
|
|
if tuple.6.get_unchecked() != &state_update.energy {
|
|
|
|
*tuple.6.get_mut_unchecked() = state_update.energy
|
|
|
|
};
|
|
|
|
if state_update.swap_loadout {
|
2021-01-07 20:25:12 +00:00
|
|
|
let mut loadout = tuple.7.get_mut_unchecked();
|
|
|
|
let loadout = &mut *loadout;
|
2020-07-06 05:56:02 +00:00
|
|
|
std::mem::swap(&mut loadout.active_item, &mut loadout.second_item);
|
|
|
|
}
|
2020-03-24 07:38:16 +00:00
|
|
|
}
|
|
|
|
|
2020-03-21 22:55:20 +00:00
|
|
|
/// ## Character Behavior System
|
2020-08-25 12:21:25 +00:00
|
|
|
/// Passes `JoinData` to `CharacterState`'s `behavior` handler fn's. Receives a
|
2020-03-21 22:55:20 +00:00
|
|
|
/// `StateUpdate` in return and performs updates to ECS Components from that.
|
2019-12-20 13:30:37 +00:00
|
|
|
pub struct Sys;
|
|
|
|
|
|
|
|
impl<'a> System<'a> for Sys {
|
2020-06-10 19:47:36 +00:00
|
|
|
#[allow(clippy::type_complexity)]
|
2019-12-20 13:30:37 +00:00
|
|
|
type SystemData = (
|
|
|
|
Entities<'a>,
|
|
|
|
Read<'a, UidAllocator>,
|
|
|
|
Read<'a, EventBus<ServerEvent>>,
|
|
|
|
Read<'a, EventBus<LocalEvent>>,
|
|
|
|
Read<'a, DeltaTime>,
|
2019-12-26 14:43:59 +00:00
|
|
|
Read<'a, LazyUpdate>,
|
2020-09-14 12:56:05 +00:00
|
|
|
ReadExpect<'a, SysMetrics>,
|
2019-12-20 13:30:37 +00:00
|
|
|
WriteStorage<'a, CharacterState>,
|
2019-12-21 15:57:15 +00:00
|
|
|
WriteStorage<'a, Pos>,
|
|
|
|
WriteStorage<'a, Vel>,
|
|
|
|
WriteStorage<'a, Ori>,
|
2020-02-24 18:17:16 +00:00
|
|
|
WriteStorage<'a, Energy>,
|
2020-03-21 21:16:26 +00:00
|
|
|
WriteStorage<'a, Loadout>,
|
2020-03-24 07:38:16 +00:00
|
|
|
WriteStorage<'a, Controller>,
|
2020-10-31 22:34:08 +00:00
|
|
|
ReadStorage<'a, Health>,
|
2019-12-20 13:30:37 +00:00
|
|
|
ReadStorage<'a, Body>,
|
|
|
|
ReadStorage<'a, PhysicsState>,
|
2020-03-10 17:54:59 +00:00
|
|
|
ReadStorage<'a, Attacking>,
|
2020-09-24 02:02:30 +00:00
|
|
|
ReadStorage<'a, Beam>,
|
2019-12-20 13:30:37 +00:00
|
|
|
ReadStorage<'a, Uid>,
|
|
|
|
ReadStorage<'a, Mounting>,
|
|
|
|
);
|
2020-02-24 18:17:16 +00:00
|
|
|
|
2020-06-10 19:47:36 +00:00
|
|
|
#[allow(clippy::while_let_on_iterator)] // TODO: Pending review in #587
|
2019-12-20 13:30:37 +00:00
|
|
|
fn run(
|
|
|
|
&mut self,
|
|
|
|
(
|
|
|
|
entities,
|
2019-12-26 18:01:19 +00:00
|
|
|
_uid_allocator,
|
2019-12-20 13:30:37 +00:00
|
|
|
server_bus,
|
|
|
|
local_bus,
|
|
|
|
dt,
|
2019-12-26 14:43:59 +00:00
|
|
|
updater,
|
2020-09-14 12:56:05 +00:00
|
|
|
sys_metrics,
|
2019-12-20 13:30:37 +00:00
|
|
|
mut character_states,
|
2019-12-21 15:57:15 +00:00
|
|
|
mut positions,
|
|
|
|
mut velocities,
|
|
|
|
mut orientations,
|
2020-02-24 18:17:16 +00:00
|
|
|
mut energies,
|
2020-03-21 21:16:26 +00:00
|
|
|
mut loadouts,
|
2020-03-24 07:38:16 +00:00
|
|
|
mut controllers,
|
2020-10-31 22:34:08 +00:00
|
|
|
healths,
|
2019-12-20 13:30:37 +00:00
|
|
|
bodies,
|
|
|
|
physics_states,
|
2020-03-10 17:54:59 +00:00
|
|
|
attacking_storage,
|
2020-09-24 02:02:30 +00:00
|
|
|
beam_storage,
|
2019-12-20 13:30:37 +00:00
|
|
|
uids,
|
|
|
|
mountings,
|
|
|
|
): Self::SystemData,
|
|
|
|
) {
|
2020-09-14 12:56:05 +00:00
|
|
|
let start_time = std::time::Instant::now();
|
2020-09-07 04:59:16 +00:00
|
|
|
span!(_guard, "run", "character_behavior::Sys::run");
|
2020-03-22 04:49:32 +00:00
|
|
|
let mut server_emitter = server_bus.emitter();
|
|
|
|
let mut local_emitter = local_bus.emitter();
|
|
|
|
|
2020-07-06 05:56:02 +00:00
|
|
|
for mut tuple in (
|
2019-12-20 13:30:37 +00:00
|
|
|
&entities,
|
|
|
|
&uids,
|
2020-07-06 05:56:02 +00:00
|
|
|
&mut character_states.restrict_mut(),
|
2019-12-21 15:57:15 +00:00
|
|
|
&mut positions,
|
|
|
|
&mut velocities,
|
|
|
|
&mut orientations,
|
2020-07-06 05:56:02 +00:00
|
|
|
&mut energies.restrict_mut(),
|
|
|
|
&mut loadouts.restrict_mut(),
|
2020-03-24 07:38:16 +00:00
|
|
|
&mut controllers,
|
2020-10-31 22:34:08 +00:00
|
|
|
&healths,
|
2019-12-20 13:30:37 +00:00
|
|
|
&bodies,
|
|
|
|
&physics_states,
|
2020-03-10 17:54:59 +00:00
|
|
|
attacking_storage.maybe(),
|
2020-09-24 02:02:30 +00:00
|
|
|
beam_storage.maybe(),
|
2019-12-20 13:30:37 +00:00
|
|
|
)
|
2020-07-06 05:56:02 +00:00
|
|
|
.join()
|
|
|
|
{
|
2019-12-29 23:47:42 +00:00
|
|
|
// Being dead overrides all other states
|
2020-03-24 07:38:16 +00:00
|
|
|
if tuple.9.is_dead {
|
|
|
|
// Do nothing
|
|
|
|
continue;
|
2019-12-29 23:47:42 +00:00
|
|
|
}
|
|
|
|
// If mounted, character state is controlled by mount
|
|
|
|
// TODO: Make mounting a state
|
2020-03-24 07:38:16 +00:00
|
|
|
if let Some(Mounting(_)) = mountings.get(tuple.0) {
|
2020-07-06 05:56:02 +00:00
|
|
|
let sit_state = CharacterState::Sit {};
|
|
|
|
if tuple.2.get_unchecked() != &sit_state {
|
|
|
|
*tuple.2.get_mut_unchecked() = sit_state;
|
|
|
|
}
|
2020-03-24 07:38:16 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
let actions = std::mem::replace(&mut tuple.8.actions, Vec::new());
|
|
|
|
for action in actions {
|
|
|
|
let j = JoinData::new(&tuple, &updater, &dt);
|
|
|
|
let mut state_update = match j.character {
|
|
|
|
CharacterState::Idle => states::idle::Data.handle_event(&j, action),
|
|
|
|
CharacterState::Climb => states::climb::Data.handle_event(&j, action),
|
|
|
|
CharacterState::Glide => states::glide::Data.handle_event(&j, action),
|
2020-06-16 21:32:39 +00:00
|
|
|
CharacterState::GlideWield => {
|
|
|
|
states::glide_wield::Data.handle_event(&j, action)
|
|
|
|
},
|
2020-03-24 07:38:16 +00:00
|
|
|
CharacterState::Sit => {
|
|
|
|
states::sit::Data::handle_event(&states::sit::Data, &j, action)
|
|
|
|
},
|
2020-05-27 06:41:55 +00:00
|
|
|
CharacterState::Dance => {
|
|
|
|
states::dance::Data::handle_event(&states::dance::Data, &j, action)
|
|
|
|
},
|
2020-08-02 05:09:11 +00:00
|
|
|
CharacterState::Sneak => {
|
|
|
|
states::sneak::Data::handle_event(&states::sneak::Data, &j, action)
|
|
|
|
},
|
2020-03-24 07:38:16 +00:00
|
|
|
CharacterState::BasicBlock => {
|
|
|
|
states::basic_block::Data.handle_event(&j, action)
|
|
|
|
},
|
|
|
|
CharacterState::Roll(data) => data.handle_event(&j, action),
|
|
|
|
CharacterState::Wielding => states::wielding::Data.handle_event(&j, action),
|
|
|
|
CharacterState::Equipping(data) => data.handle_event(&j, action),
|
2020-09-04 01:54:59 +00:00
|
|
|
CharacterState::ComboMelee(data) => data.handle_event(&j, action),
|
2020-03-24 07:38:16 +00:00
|
|
|
CharacterState::BasicMelee(data) => data.handle_event(&j, action),
|
|
|
|
CharacterState::BasicRanged(data) => data.handle_event(&j, action),
|
|
|
|
CharacterState::Boost(data) => data.handle_event(&j, action),
|
|
|
|
CharacterState::DashMelee(data) => data.handle_event(&j, action),
|
2020-07-03 15:40:12 +00:00
|
|
|
CharacterState::LeapMelee(data) => data.handle_event(&j, action),
|
2020-07-08 19:58:41 +00:00
|
|
|
CharacterState::SpinMelee(data) => data.handle_event(&j, action),
|
2020-09-21 04:16:52 +00:00
|
|
|
CharacterState::ChargedMelee(data) => data.handle_event(&j, action),
|
2020-07-26 03:06:53 +00:00
|
|
|
CharacterState::ChargedRanged(data) => data.handle_event(&j, action),
|
2020-09-21 04:16:52 +00:00
|
|
|
CharacterState::RepeaterRanged(data) => data.handle_event(&j, action),
|
2020-10-04 01:24:15 +00:00
|
|
|
CharacterState::Shockwave(data) => data.handle_event(&j, action),
|
2020-08-27 00:08:29 +00:00
|
|
|
CharacterState::BasicBeam(data) => data.handle_event(&j, action),
|
2020-03-24 07:38:16 +00:00
|
|
|
};
|
|
|
|
local_emitter.append(&mut state_update.local_events);
|
|
|
|
server_emitter.append(&mut state_update.server_events);
|
|
|
|
incorporate_update(&mut tuple, state_update);
|
2019-12-29 23:47:42 +00:00
|
|
|
}
|
|
|
|
|
2020-03-24 07:38:16 +00:00
|
|
|
let j = JoinData::new(&tuple, &updater, &dt);
|
|
|
|
|
2020-03-07 18:15:02 +00:00
|
|
|
let mut state_update = match j.character {
|
2020-03-15 13:34:17 +00:00
|
|
|
CharacterState::Idle => states::idle::Data.behavior(&j),
|
|
|
|
CharacterState::Climb => states::climb::Data.behavior(&j),
|
|
|
|
CharacterState::Glide => states::glide::Data.behavior(&j),
|
2020-06-16 21:32:39 +00:00
|
|
|
CharacterState::GlideWield => states::glide_wield::Data.behavior(&j),
|
2020-03-14 21:17:27 +00:00
|
|
|
CharacterState::Sit => states::sit::Data::behavior(&states::sit::Data, &j),
|
2020-05-27 06:41:55 +00:00
|
|
|
CharacterState::Dance => states::dance::Data::behavior(&states::dance::Data, &j),
|
2020-08-02 05:09:11 +00:00
|
|
|
CharacterState::Sneak => states::sneak::Data::behavior(&states::sneak::Data, &j),
|
2020-03-21 22:55:20 +00:00
|
|
|
CharacterState::BasicBlock => states::basic_block::Data.behavior(&j),
|
2020-03-15 13:34:17 +00:00
|
|
|
CharacterState::Roll(data) => data.behavior(&j),
|
|
|
|
CharacterState::Wielding => states::wielding::Data.behavior(&j),
|
|
|
|
CharacterState::Equipping(data) => data.behavior(&j),
|
2020-09-04 01:54:59 +00:00
|
|
|
CharacterState::ComboMelee(data) => data.behavior(&j),
|
2020-03-16 11:32:57 +00:00
|
|
|
CharacterState::BasicMelee(data) => data.behavior(&j),
|
|
|
|
CharacterState::BasicRanged(data) => data.behavior(&j),
|
|
|
|
CharacterState::Boost(data) => data.behavior(&j),
|
2020-03-16 15:34:53 +00:00
|
|
|
CharacterState::DashMelee(data) => data.behavior(&j),
|
2020-07-03 15:40:12 +00:00
|
|
|
CharacterState::LeapMelee(data) => data.behavior(&j),
|
2020-07-08 19:58:41 +00:00
|
|
|
CharacterState::SpinMelee(data) => data.behavior(&j),
|
2020-09-21 04:16:52 +00:00
|
|
|
CharacterState::ChargedMelee(data) => data.behavior(&j),
|
2020-07-26 03:06:53 +00:00
|
|
|
CharacterState::ChargedRanged(data) => data.behavior(&j),
|
2020-09-21 04:16:52 +00:00
|
|
|
CharacterState::RepeaterRanged(data) => data.behavior(&j),
|
2020-10-04 01:24:15 +00:00
|
|
|
CharacterState::Shockwave(data) => data.behavior(&j),
|
2020-08-27 00:08:29 +00:00
|
|
|
CharacterState::BasicBeam(data) => data.behavior(&j),
|
2020-03-07 18:15:02 +00:00
|
|
|
};
|
2019-12-26 14:43:59 +00:00
|
|
|
|
2020-03-22 04:49:32 +00:00
|
|
|
local_emitter.append(&mut state_update.local_events);
|
|
|
|
server_emitter.append(&mut state_update.server_events);
|
2020-03-24 07:38:16 +00:00
|
|
|
incorporate_update(&mut tuple, state_update);
|
2019-12-29 23:47:42 +00:00
|
|
|
}
|
2020-09-14 12:56:05 +00:00
|
|
|
sys_metrics.character_behavior_ns.store(
|
2020-12-15 23:51:07 +00:00
|
|
|
start_time.elapsed().as_nanos() as u64,
|
2020-09-14 12:56:05 +00:00
|
|
|
std::sync::atomic::Ordering::Relaxed,
|
|
|
|
);
|
2019-12-22 16:08:48 +00:00
|
|
|
}
|
|
|
|
}
|