mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
399 lines
18 KiB
Rust
399 lines
18 KiB
Rust
use specs::{
|
|
shred::ResourceId, Entities, Join, LazyUpdate, Read, ReadExpect, ReadStorage, SystemData,
|
|
World, Write, WriteStorage,
|
|
};
|
|
|
|
use common::{
|
|
comp::{
|
|
self, inventory::item::MaterialStatManifest, Beam, Body, CharacterState, Combo, Controller,
|
|
Density, Energy, Health, Inventory, Mass, Melee, Mounting, Ori, PhysicsState, Poise,
|
|
PoiseState, Pos, SkillSet, StateUpdate, Stats, Vel,
|
|
},
|
|
event::{EventBus, LocalEvent, ServerEvent},
|
|
outcome::Outcome,
|
|
resources::DeltaTime,
|
|
states::{
|
|
self,
|
|
behavior::{CharacterBehavior, JoinData, JoinStruct},
|
|
},
|
|
terrain::TerrainGrid,
|
|
uid::Uid,
|
|
};
|
|
use common_ecs::{Job, Origin, Phase, System};
|
|
use std::time::Duration;
|
|
|
|
fn incorporate_update(join: &mut JoinStruct, mut state_update: StateUpdate) {
|
|
// TODO: if checking equality is expensive use optional field in StateUpdate
|
|
if join.char_state.get_unchecked() != &state_update.character {
|
|
*join.char_state.get_mut_unchecked() = state_update.character
|
|
};
|
|
*join.pos = state_update.pos;
|
|
*join.vel = state_update.vel;
|
|
*join.ori = state_update.ori;
|
|
*join.density = state_update.density;
|
|
// Note: might be changed every tick by timer anyway
|
|
if join.energy.get_unchecked() != &state_update.energy {
|
|
*join.energy.get_mut_unchecked() = state_update.energy
|
|
};
|
|
join.controller
|
|
.queued_inputs
|
|
.append(&mut state_update.queued_inputs);
|
|
for input in state_update.removed_inputs {
|
|
join.controller.queued_inputs.remove(&input);
|
|
}
|
|
if state_update.swap_equipped_weapons {
|
|
let mut inventory = join.inventory.get_mut_unchecked();
|
|
let inventory = &mut *inventory;
|
|
inventory.swap_equipped_weapons();
|
|
}
|
|
}
|
|
|
|
#[derive(SystemData)]
|
|
pub struct ReadData<'a> {
|
|
entities: Entities<'a>,
|
|
server_bus: Read<'a, EventBus<ServerEvent>>,
|
|
local_bus: Read<'a, EventBus<LocalEvent>>,
|
|
dt: Read<'a, DeltaTime>,
|
|
lazy_update: Read<'a, LazyUpdate>,
|
|
healths: ReadStorage<'a, Health>,
|
|
bodies: ReadStorage<'a, Body>,
|
|
masses: ReadStorage<'a, Mass>,
|
|
physics_states: ReadStorage<'a, PhysicsState>,
|
|
melee_attacks: ReadStorage<'a, Melee>,
|
|
beams: ReadStorage<'a, Beam>,
|
|
uids: ReadStorage<'a, Uid>,
|
|
mountings: ReadStorage<'a, Mounting>,
|
|
stats: ReadStorage<'a, Stats>,
|
|
skill_sets: ReadStorage<'a, SkillSet>,
|
|
msm: Read<'a, MaterialStatManifest>,
|
|
combos: ReadStorage<'a, Combo>,
|
|
alignments: ReadStorage<'a, comp::Alignment>,
|
|
terrain: ReadExpect<'a, TerrainGrid>,
|
|
}
|
|
|
|
/// ## Character Behavior System
|
|
/// Passes `JoinData` to `CharacterState`'s `behavior` handler fn's. Receives a
|
|
/// `StateUpdate` in return and performs updates to ECS Components from that.
|
|
#[derive(Default)]
|
|
pub struct Sys;
|
|
|
|
impl<'a> System<'a> for Sys {
|
|
#[allow(clippy::type_complexity)]
|
|
type SystemData = (
|
|
ReadData<'a>,
|
|
WriteStorage<'a, CharacterState>,
|
|
WriteStorage<'a, Pos>,
|
|
WriteStorage<'a, Vel>,
|
|
WriteStorage<'a, Ori>,
|
|
WriteStorage<'a, Density>,
|
|
WriteStorage<'a, Energy>,
|
|
WriteStorage<'a, Inventory>,
|
|
WriteStorage<'a, Controller>,
|
|
WriteStorage<'a, Poise>,
|
|
Write<'a, Vec<Outcome>>,
|
|
);
|
|
|
|
const NAME: &'static str = "character_behavior";
|
|
const ORIGIN: Origin = Origin::Common;
|
|
const PHASE: Phase = Phase::Create;
|
|
|
|
#[allow(clippy::while_let_on_iterator)] // TODO: Pending review in #587
|
|
fn run(
|
|
_job: &mut Job<Self>,
|
|
(
|
|
read_data,
|
|
mut character_states,
|
|
mut positions,
|
|
mut velocities,
|
|
mut orientations,
|
|
mut densities,
|
|
mut energies,
|
|
mut inventories,
|
|
mut controllers,
|
|
mut poises,
|
|
mut outcomes,
|
|
): Self::SystemData,
|
|
) {
|
|
let mut server_emitter = read_data.server_bus.emitter();
|
|
let mut local_emitter = read_data.local_bus.emitter();
|
|
|
|
for (
|
|
entity,
|
|
uid,
|
|
mut char_state,
|
|
mut pos,
|
|
mut vel,
|
|
mut ori,
|
|
mass,
|
|
mut density,
|
|
energy,
|
|
inventory,
|
|
mut controller,
|
|
health,
|
|
body,
|
|
physics,
|
|
(stat, skill_set),
|
|
combo,
|
|
) in (
|
|
&read_data.entities,
|
|
&read_data.uids,
|
|
&mut character_states.restrict_mut(),
|
|
&mut positions,
|
|
&mut velocities,
|
|
&mut orientations,
|
|
&read_data.masses,
|
|
&mut densities,
|
|
&mut energies.restrict_mut(),
|
|
&mut inventories.restrict_mut(),
|
|
&mut controllers,
|
|
read_data.healths.maybe(),
|
|
&read_data.bodies,
|
|
&read_data.physics_states,
|
|
(&read_data.stats, &read_data.skill_sets),
|
|
&read_data.combos,
|
|
)
|
|
.join()
|
|
{
|
|
// Being dead overrides all other states
|
|
if health.map_or(false, |h| h.is_dead) {
|
|
// Do nothing
|
|
continue;
|
|
}
|
|
|
|
// Enter stunned state if poise damage is enough
|
|
if let Some(mut poise) = poises.get_mut(entity) {
|
|
let was_wielded = char_state.get_unchecked().is_wield();
|
|
let poise_state = poise.poise_state();
|
|
let pos = pos.0;
|
|
match poise_state {
|
|
PoiseState::Normal => {},
|
|
PoiseState::Interrupted => {
|
|
poise.reset();
|
|
*char_state.get_mut_unchecked() =
|
|
CharacterState::Stunned(common::states::stunned::Data {
|
|
static_data: common::states::stunned::StaticData {
|
|
buildup_duration: Duration::from_millis(125),
|
|
recover_duration: Duration::from_millis(125),
|
|
movement_speed: 0.80,
|
|
poise_state,
|
|
},
|
|
timer: Duration::default(),
|
|
stage_section: common::states::utils::StageSection::Buildup,
|
|
was_wielded,
|
|
});
|
|
outcomes.push(Outcome::PoiseChange {
|
|
pos,
|
|
state: PoiseState::Interrupted,
|
|
});
|
|
},
|
|
PoiseState::Stunned => {
|
|
poise.reset();
|
|
*char_state.get_mut_unchecked() =
|
|
CharacterState::Stunned(common::states::stunned::Data {
|
|
static_data: common::states::stunned::StaticData {
|
|
buildup_duration: Duration::from_millis(300),
|
|
recover_duration: Duration::from_millis(300),
|
|
movement_speed: 0.65,
|
|
poise_state,
|
|
},
|
|
timer: Duration::default(),
|
|
stage_section: common::states::utils::StageSection::Buildup,
|
|
was_wielded,
|
|
});
|
|
outcomes.push(Outcome::PoiseChange {
|
|
pos,
|
|
state: PoiseState::Stunned,
|
|
});
|
|
server_emitter.emit(ServerEvent::Knockback {
|
|
entity,
|
|
impulse: 5.0 * poise.knockback(),
|
|
});
|
|
},
|
|
PoiseState::Dazed => {
|
|
poise.reset();
|
|
*char_state.get_mut_unchecked() =
|
|
CharacterState::Stunned(common::states::stunned::Data {
|
|
static_data: common::states::stunned::StaticData {
|
|
buildup_duration: Duration::from_millis(600),
|
|
recover_duration: Duration::from_millis(250),
|
|
movement_speed: 0.45,
|
|
poise_state,
|
|
},
|
|
timer: Duration::default(),
|
|
stage_section: common::states::utils::StageSection::Buildup,
|
|
was_wielded,
|
|
});
|
|
outcomes.push(Outcome::PoiseChange {
|
|
pos,
|
|
state: PoiseState::Dazed,
|
|
});
|
|
server_emitter.emit(ServerEvent::Knockback {
|
|
entity,
|
|
impulse: 10.0 * poise.knockback(),
|
|
});
|
|
},
|
|
PoiseState::KnockedDown => {
|
|
poise.reset();
|
|
*char_state.get_mut_unchecked() =
|
|
CharacterState::Stunned(common::states::stunned::Data {
|
|
static_data: common::states::stunned::StaticData {
|
|
buildup_duration: Duration::from_millis(750),
|
|
recover_duration: Duration::from_millis(500),
|
|
movement_speed: 0.4,
|
|
poise_state,
|
|
},
|
|
timer: Duration::default(),
|
|
stage_section: common::states::utils::StageSection::Buildup,
|
|
was_wielded,
|
|
});
|
|
outcomes.push(Outcome::PoiseChange {
|
|
pos,
|
|
state: PoiseState::KnockedDown,
|
|
});
|
|
server_emitter.emit(ServerEvent::Knockback {
|
|
entity,
|
|
impulse: 10.0 * poise.knockback(),
|
|
});
|
|
},
|
|
}
|
|
}
|
|
|
|
// Controller actions
|
|
let actions = std::mem::take(&mut controller.actions);
|
|
|
|
let mut join_struct = JoinStruct {
|
|
entity,
|
|
uid: &uid,
|
|
char_state,
|
|
pos: &mut pos,
|
|
vel: &mut vel,
|
|
ori: &mut ori,
|
|
mass: &mass,
|
|
density: &mut density,
|
|
energy,
|
|
inventory,
|
|
controller: &mut controller,
|
|
health,
|
|
body: &body,
|
|
physics: &physics,
|
|
melee_attack: read_data.melee_attacks.get(entity),
|
|
beam: read_data.beams.get(entity),
|
|
stat: &stat,
|
|
skill_set: &skill_set,
|
|
combo: &combo,
|
|
alignment: read_data.alignments.get(entity),
|
|
terrain: &read_data.terrain,
|
|
};
|
|
|
|
for action in actions {
|
|
let j = JoinData::new(
|
|
&join_struct,
|
|
&read_data.lazy_update,
|
|
&read_data.dt,
|
|
&read_data.msm,
|
|
);
|
|
let mut state_update = match j.character {
|
|
CharacterState::Idle => states::idle::Data.handle_event(&j, action),
|
|
CharacterState::Talk => states::talk::Data.handle_event(&j, action),
|
|
CharacterState::Climb(data) => data.handle_event(&j, action),
|
|
CharacterState::Glide(data) => data.handle_event(&j, action),
|
|
CharacterState::GlideWield => {
|
|
states::glide_wield::Data.handle_event(&j, action)
|
|
},
|
|
CharacterState::Stunned(data) => data.handle_event(&j, action),
|
|
CharacterState::Sit => {
|
|
states::sit::Data::handle_event(&states::sit::Data, &j, action)
|
|
},
|
|
CharacterState::Dance => {
|
|
states::dance::Data::handle_event(&states::dance::Data, &j, action)
|
|
},
|
|
CharacterState::Sneak => {
|
|
states::sneak::Data::handle_event(&states::sneak::Data, &j, action)
|
|
},
|
|
CharacterState::BasicBlock(data) => 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),
|
|
CharacterState::ComboMelee(data) => data.handle_event(&j, action),
|
|
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),
|
|
CharacterState::LeapMelee(data) => data.handle_event(&j, action),
|
|
CharacterState::SpinMelee(data) => data.handle_event(&j, action),
|
|
CharacterState::ChargedMelee(data) => data.handle_event(&j, action),
|
|
CharacterState::ChargedRanged(data) => data.handle_event(&j, action),
|
|
CharacterState::RepeaterRanged(data) => data.handle_event(&j, action),
|
|
CharacterState::Shockwave(data) => data.handle_event(&j, action),
|
|
CharacterState::BasicBeam(data) => data.handle_event(&j, action),
|
|
CharacterState::BasicAura(data) => data.handle_event(&j, action),
|
|
CharacterState::HealingBeam(data) => data.handle_event(&j, action),
|
|
CharacterState::Blink(data) => data.handle_event(&j, action),
|
|
CharacterState::BasicSummon(data) => data.handle_event(&j, action),
|
|
CharacterState::SelfBuff(data) => data.handle_event(&j, action),
|
|
CharacterState::SpriteSummon(data) => data.handle_event(&j, action),
|
|
};
|
|
local_emitter.append(&mut state_update.local_events);
|
|
server_emitter.append(&mut state_update.server_events);
|
|
incorporate_update(&mut join_struct, state_update);
|
|
}
|
|
|
|
// Mounted occurs after control actions have been handled
|
|
// If mounted, character state is controlled by mount
|
|
// TODO: Make mounting a state
|
|
if let Some(Mounting(_)) = read_data.mountings.get(entity) {
|
|
let sit_state = CharacterState::Sit {};
|
|
if join_struct.char_state.get_unchecked() != &sit_state {
|
|
*join_struct.char_state.get_mut_unchecked() = sit_state;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
let j = JoinData::new(
|
|
&join_struct,
|
|
&read_data.lazy_update,
|
|
&read_data.dt,
|
|
&read_data.msm,
|
|
);
|
|
|
|
let mut state_update = match j.character {
|
|
CharacterState::Idle => states::idle::Data.behavior(&j),
|
|
CharacterState::Talk => states::talk::Data.behavior(&j),
|
|
CharacterState::Climb(data) => data.behavior(&j),
|
|
CharacterState::Glide(data) => data.behavior(&j),
|
|
CharacterState::GlideWield => states::glide_wield::Data.behavior(&j),
|
|
CharacterState::Stunned(data) => data.behavior(&j),
|
|
CharacterState::Sit => states::sit::Data::behavior(&states::sit::Data, &j),
|
|
CharacterState::Dance => states::dance::Data::behavior(&states::dance::Data, &j),
|
|
CharacterState::Sneak => states::sneak::Data::behavior(&states::sneak::Data, &j),
|
|
CharacterState::BasicBlock(data) => data.behavior(&j),
|
|
CharacterState::Roll(data) => data.behavior(&j),
|
|
CharacterState::Wielding => states::wielding::Data.behavior(&j),
|
|
CharacterState::Equipping(data) => data.behavior(&j),
|
|
CharacterState::ComboMelee(data) => data.behavior(&j),
|
|
CharacterState::BasicMelee(data) => data.behavior(&j),
|
|
CharacterState::BasicRanged(data) => data.behavior(&j),
|
|
CharacterState::Boost(data) => data.behavior(&j),
|
|
CharacterState::DashMelee(data) => data.behavior(&j),
|
|
CharacterState::LeapMelee(data) => data.behavior(&j),
|
|
CharacterState::SpinMelee(data) => data.behavior(&j),
|
|
CharacterState::ChargedMelee(data) => data.behavior(&j),
|
|
CharacterState::ChargedRanged(data) => data.behavior(&j),
|
|
CharacterState::RepeaterRanged(data) => data.behavior(&j),
|
|
CharacterState::Shockwave(data) => data.behavior(&j),
|
|
CharacterState::BasicBeam(data) => data.behavior(&j),
|
|
CharacterState::BasicAura(data) => data.behavior(&j),
|
|
CharacterState::HealingBeam(data) => data.behavior(&j),
|
|
CharacterState::Blink(data) => data.behavior(&j),
|
|
CharacterState::BasicSummon(data) => data.behavior(&j),
|
|
CharacterState::SelfBuff(data) => data.behavior(&j),
|
|
CharacterState::SpriteSummon(data) => data.behavior(&j),
|
|
};
|
|
|
|
local_emitter.append(&mut state_update.local_events);
|
|
server_emitter.append(&mut state_update.server_events);
|
|
incorporate_update(&mut join_struct, state_update);
|
|
}
|
|
}
|
|
}
|