diff --git a/common/net/src/msg/ecs_packet.rs b/common/net/src/msg/ecs_packet.rs index 599115dcd1..1ee77e127d 100644 --- a/common/net/src/msg/ecs_packet.rs +++ b/common/net/src/msg/ecs_packet.rs @@ -17,6 +17,7 @@ sum_type! { Buffs(comp::Buffs), Auras(comp::Auras), Energy(comp::Energy), + Combo(comp::Combo), Health(comp::Health), Poise(comp::Poise), LightEmitter(comp::LightEmitter), @@ -50,6 +51,7 @@ sum_type! { Buffs(PhantomData), Auras(PhantomData), Energy(PhantomData), + Combo(PhantomData), Health(PhantomData), Poise(PhantomData), LightEmitter(PhantomData), @@ -83,6 +85,7 @@ impl sync::CompPacket for EcsCompPacket { EcsCompPacket::Buffs(comp) => sync::handle_insert(comp, entity, world), EcsCompPacket::Auras(comp) => sync::handle_insert(comp, entity, world), EcsCompPacket::Energy(comp) => sync::handle_insert(comp, entity, world), + EcsCompPacket::Combo(comp) => sync::handle_insert(comp, entity, world), EcsCompPacket::Health(comp) => sync::handle_insert(comp, entity, world), EcsCompPacket::Poise(comp) => sync::handle_insert(comp, entity, world), EcsCompPacket::LightEmitter(comp) => sync::handle_insert(comp, entity, world), @@ -114,6 +117,7 @@ impl sync::CompPacket for EcsCompPacket { EcsCompPacket::Buffs(comp) => sync::handle_modify(comp, entity, world), EcsCompPacket::Auras(comp) => sync::handle_modify(comp, entity, world), EcsCompPacket::Energy(comp) => sync::handle_modify(comp, entity, world), + EcsCompPacket::Combo(comp) => sync::handle_modify(comp, entity, world), EcsCompPacket::Health(comp) => sync::handle_modify(comp, entity, world), EcsCompPacket::Poise(comp) => sync::handle_modify(comp, entity, world), EcsCompPacket::LightEmitter(comp) => sync::handle_modify(comp, entity, world), @@ -145,6 +149,7 @@ impl sync::CompPacket for EcsCompPacket { EcsCompPhantom::Buffs(_) => sync::handle_remove::(entity, world), EcsCompPhantom::Auras(_) => sync::handle_remove::(entity, world), EcsCompPhantom::Energy(_) => sync::handle_remove::(entity, world), + EcsCompPhantom::Combo(_) => sync::handle_remove::(entity, world), EcsCompPhantom::Health(_) => sync::handle_remove::(entity, world), EcsCompPhantom::Poise(_) => sync::handle_remove::(entity, world), EcsCompPhantom::LightEmitter(_) => { diff --git a/common/src/combat.rs b/common/src/combat.rs index 5580e398dc..803b009693 100644 --- a/common/src/combat.rs +++ b/common/src/combat.rs @@ -86,6 +86,13 @@ impl Attack { self } + pub fn with_combo_increment(self) -> Self { + self.with_effect( + AttackEffect::new(None, CombatEffect::Combo(1)) + .with_requirement(CombatRequirement::AnyDamage), + ) + } + pub fn effects(&self) -> impl Iterator { self.effects.iter() } #[allow(clippy::too_many_arguments)] @@ -195,6 +202,14 @@ impl Attack { }); } }, + CombatEffect::Combo(c) => { + if let Some(attacker_entity) = attacker_info.map(|a| a.entity) { + emit(ServerEvent::ComboChange { + entity: attacker_entity, + change: *c, + }); + } + }, } } } @@ -303,6 +318,14 @@ impl Attack { }); } }, + CombatEffect::Combo(c) => { + if let Some(attacker_entity) = attacker_info.map(|a| a.entity) { + emit(ServerEvent::ComboChange { + entity: attacker_entity, + change: c, + }); + } + }, } } } @@ -368,6 +391,7 @@ pub enum CombatEffect { EnergyReward(f32), Lifesteal(f32), Poise(f32), + Combo(i32), } #[cfg(not(target_arch = "wasm32"))] diff --git a/common/src/comp/ability.rs b/common/src/comp/ability.rs index 4e7f1325ff..0100612828 100644 --- a/common/src/comp/ability.rs +++ b/common/src/comp/ability.rs @@ -1216,7 +1216,6 @@ impl From<(&CharacterAbility, AbilityInfo)> for CharacterState { ability_info, }, stage: 1, - combo: 0, timer: Duration::default(), stage_section: StageSection::Buildup, next_stage: false, diff --git a/common/src/comp/combo.rs b/common/src/comp/combo.rs new file mode 100644 index 0000000000..a532d25bb3 --- /dev/null +++ b/common/src/comp/combo.rs @@ -0,0 +1,39 @@ +use serde::{Deserialize, Serialize}; +use specs::{Component, DerefFlaggedStorage}; +use specs_idvs::IdvStorage; + +#[derive(Clone, Copy, Debug, PartialEq, Serialize, Deserialize)] +pub struct Combo { + counter: u32, + last_increase: f64, +} + +impl Default for Combo { + fn default() -> Self { + Self { + counter: 0, + last_increase: 0.0, + } + } +} + +impl Combo { + pub fn counter(&self) -> u32 { self.counter } + + pub fn last_increase(&self) -> f64 { self.last_increase } + + pub fn reset(&mut self) { self.counter = 0; } + + pub fn increase_by(&mut self, amount: u32, time: f64) { + self.counter = self.counter.saturating_add(amount); + self.last_increase = time; + } + + pub fn decrease_by(&mut self, amount: u32) { + self.counter = self.counter.saturating_sub(amount); + } +} + +impl Component for Combo { + type Storage = DerefFlaggedStorage>; +} diff --git a/common/src/comp/mod.rs b/common/src/comp/mod.rs index 38d1694b52..8e342cde4e 100644 --- a/common/src/comp/mod.rs +++ b/common/src/comp/mod.rs @@ -8,6 +8,7 @@ pub mod buff; #[cfg(not(target_arch = "wasm32"))] mod character_state; #[cfg(not(target_arch = "wasm32"))] pub mod chat; +pub mod combo; #[cfg(not(target_arch = "wasm32"))] mod controller; #[cfg(not(target_arch = "wasm32"))] mod energy; @@ -58,6 +59,7 @@ pub use self::{ chat::{ ChatMode, ChatMsg, ChatType, Faction, SpeechBubble, SpeechBubbleType, UnresolvedChatMsg, }, + combo::Combo, controller::{ Climb, ControlAction, ControlEvent, Controller, ControllerInputs, GroupManip, Input, InventoryManip, LoadoutManip, MountState, Mounting, SlotManip, diff --git a/common/src/event.rs b/common/src/event.rs index 0a73cfa947..ecf2f784ea 100644 --- a/common/src/event.rs +++ b/common/src/event.rs @@ -143,6 +143,10 @@ pub enum ServerEvent { entity: EcsEntity, change: comp::EnergyChange, }, + ComboChange { + entity: EcsEntity, + change: i32, + }, } pub struct EventBus { diff --git a/common/src/states/behavior.rs b/common/src/states/behavior.rs index 73f1b054bc..f791dd13d1 100644 --- a/common/src/states/behavior.rs +++ b/common/src/states/behavior.rs @@ -1,6 +1,6 @@ use crate::{ comp::{ - item::MaterialStatManifest, Beam, Body, CharacterState, ControlAction, Controller, + item::MaterialStatManifest, Beam, Body, CharacterState, Combo, ControlAction, Controller, ControllerInputs, Energy, Health, Inventory, LoadoutManip, Melee, Ori, PhysicsState, Pos, StateUpdate, Stats, Vel, }, @@ -68,6 +68,7 @@ pub struct JoinData<'a> { pub updater: &'a LazyUpdate, pub stats: &'a Stats, pub msm: &'a MaterialStatManifest, + pub combo: &'a Combo, } type RestrictedMut<'a, C> = PairedStorage< @@ -95,6 +96,7 @@ pub struct JoinStruct<'a> { pub melee_attack: Option<&'a Melee>, pub beam: Option<&'a Beam>, pub stat: &'a Stats, + pub combo: &'a Combo, } impl<'a> JoinData<'a> { @@ -123,6 +125,7 @@ impl<'a> JoinData<'a> { updater, dt, msm, + combo: j.combo, } } } diff --git a/common/src/states/combo_melee.rs b/common/src/states/combo_melee.rs index a16bf7edac..476673e5e0 100644 --- a/common/src/states/combo_melee.rs +++ b/common/src/states/combo_melee.rs @@ -109,8 +109,6 @@ pub struct Data { pub static_data: StaticData, /// Indicates what stage the combo is in pub stage: u32, - /// Number of consecutive strikes - pub combo: u32, /// Timer for each stage pub timer: Duration, /// Checks what section a stage is in @@ -127,9 +125,6 @@ impl CharacterBehavior for Data { handle_move(data, &mut update, 0.3); if !ability_key_is_pressed(data, self.static_data.ability_info.key) { handle_interrupt(data, &mut update, self.static_data.is_interruptible); - if let CharacterState::Roll(roll) = &mut update.character { - roll.was_combo = Some((self.stage, self.combo)); - } match update.character { CharacterState::ComboMelee(_) => {}, _ => { @@ -142,7 +137,11 @@ impl CharacterBehavior for Data { let speed_modifer = 1.0 + self.static_data.max_speed_increase - * (1.0 - self.static_data.speed_increase.powi(self.combo as i32)); + * (1.0 + - self + .static_data + .speed_increase + .powi(data.combo.counter() as i32)); match self.stage_section { StageSection::Buildup => { @@ -170,7 +169,7 @@ impl CharacterBehavior for Data { + (self .static_data .scales_from_combo - .min(self.combo / self.static_data.num_stages) + .min(data.combo.counter() / self.static_data.num_stages) as f32) * self.static_data.stage_data[stage_index].damage_increase; @@ -178,7 +177,7 @@ impl CharacterBehavior for Data { + (self .static_data .scales_from_combo - .min(self.combo / self.static_data.num_stages) + .min(data.combo.counter() / self.static_data.num_stages) as f32) * self.static_data.stage_data[stage_index].poise_damage_increase; let poise = AttackEffect::new( @@ -196,7 +195,7 @@ impl CharacterBehavior for Data { .with_requirement(CombatRequirement::AnyDamage); let energy = self.static_data.max_energy_gain.min( self.static_data.initial_energy_gain - + self.combo as f32 * self.static_data.energy_increase, + + data.combo.counter() as f32 * self.static_data.energy_increase, ); let energy = AttackEffect::new(None, CombatEffect::EnergyReward(energy)) .with_requirement(CombatRequirement::AnyDamage); @@ -214,7 +213,8 @@ impl CharacterBehavior for Data { .with_crit(0.5, 1.3) .with_effect(energy) .with_effect(poise) - .with_effect(knockback); + .with_effect(knockback) + .with_combo_increment(); data.updater.insert(data.entity, Melee { attack, @@ -288,7 +288,6 @@ impl CharacterBehavior for Data { timer: Duration::default(), stage_section: StageSection::Buildup, next_stage: false, - ..*self }); } else { // Done @@ -311,7 +310,6 @@ impl CharacterBehavior for Data { update.character = CharacterState::ComboMelee(Data { static_data: self.static_data.clone(), stage: self.stage, - combo: self.combo + 1, timer: self.timer, stage_section: self.stage_section, next_stage: self.next_stage, diff --git a/common/src/states/roll.rs b/common/src/states/roll.rs index e36d80a6cb..4f46f606de 100644 --- a/common/src/states/roll.rs +++ b/common/src/states/roll.rs @@ -36,8 +36,8 @@ pub struct Data { pub was_wielded: bool, /// Was sneaking pub was_sneak: bool, - /// Was combo, .0 is stage, .1 is combo counter - pub was_combo: Option<(u32, u32)>, + /// Was in state with combo + pub was_combo: Option<(AbilityKey, u32)>, } impl CharacterBehavior for Data { @@ -114,13 +114,10 @@ impl CharacterBehavior for Data { }); } else { // Done - if self.was_wielded { + if let Some((key, stage)) = self.was_combo { + resume_combo(data, &mut update, key, stage); + } else if self.was_wielded { update.character = CharacterState::Wielding; - let combo_data = self.was_combo; - - if let Some(combo_data) = combo_data { - continue_combo(data, &mut update, combo_data); - } } else if self.was_sneak { update.character = CharacterState::Sneak; } else { @@ -130,13 +127,10 @@ impl CharacterBehavior for Data { }, _ => { // If it somehow ends up in an incorrect stage section - if self.was_wielded { + if let Some((key, stage)) = self.was_combo { + resume_combo(data, &mut update, key, stage); + } else if self.was_wielded { update.character = CharacterState::Wielding; - let combo_data = self.was_combo; - - if let Some(combo_data) = combo_data { - continue_combo(data, &mut update, combo_data); - } } else if self.was_sneak { update.character = CharacterState::Sneak; } else { diff --git a/common/src/states/utils.rs b/common/src/states/utils.rs index 51e1623403..a39f20811e 100644 --- a/common/src/states/utils.rs +++ b/common/src/states/utils.rs @@ -529,7 +529,17 @@ pub fn handle_dodge_input(data: &JoinData, update: &mut StateUpdate) { }) .filter(|ability| ability.requirements_paid(data, update)) { - if data.character.is_wield() { + if let CharacterState::ComboMelee(c) = data.character { + update.character = ( + &ability, + AbilityInfo::from_key(data, AbilityKey::Dodge, false), + ) + .into(); + if let CharacterState::Roll(roll) = &mut update.character { + roll.was_combo = Some((c.static_data.ability_info.key, c.stage)); + roll.was_wielded = true; + } + } else if data.character.is_wield() { update.character = ( &ability, AbilityInfo::from_key(data, AbilityKey::Dodge, false), @@ -576,6 +586,16 @@ pub fn handle_interrupt(data: &JoinData, update: &mut StateUpdate, attacks_inter handle_dodge_input(data, update); } +pub fn resume_combo(data: &JoinData, update: &mut StateUpdate, key: AbilityKey, stage: u32) { + if ability_key_is_pressed(data, key) { + handle_interrupt(data, update, true); + } + // If other states are introduced that progress through stages, add them here + if let CharacterState::ComboMelee(c) = &mut update.character { + c.stage = stage; + } +} + pub fn ability_key_is_pressed(data: &JoinData, ability_key: AbilityKey) -> bool { match ability_key { AbilityKey::Mouse1 => data.inputs.primary.is_pressed(), @@ -586,14 +606,6 @@ pub fn ability_key_is_pressed(data: &JoinData, ability_key: AbilityKey) -> bool } } -pub fn continue_combo(data: &JoinData, update: &mut StateUpdate, combo_data: (u32, u32)) { - handle_ability1_input(data, update); - if let CharacterState::ComboMelee(data) = &mut update.character { - data.stage = combo_data.0; - data.combo = combo_data.1; - } -} - /// Determines what portion a state is in. Used in all attacks (eventually). Is /// used to control aspects of animation code, as well as logic within the /// character states. diff --git a/common/sys/src/character_behavior.rs b/common/sys/src/character_behavior.rs index 5e67aa40d5..fc83ab21ec 100644 --- a/common/sys/src/character_behavior.rs +++ b/common/sys/src/character_behavior.rs @@ -9,8 +9,8 @@ use common::{ item::MaterialStatManifest, slot::{EquipSlot, Slot}, }, - Beam, Body, CharacterState, Controller, Energy, Health, Inventory, Melee, Mounting, Ori, - PhysicsState, Poise, PoiseState, Pos, StateUpdate, Stats, Vel, + Beam, Body, CharacterState, Combo, Controller, Energy, Health, Inventory, Melee, Mounting, + Ori, PhysicsState, Poise, PoiseState, Pos, StateUpdate, Stats, Vel, }, event::{EventBus, LocalEvent, ServerEvent}, metrics::SysMetrics, @@ -66,6 +66,7 @@ pub struct ReadData<'a> { mountings: ReadStorage<'a, Mounting>, stats: ReadStorage<'a, Stats>, msm: Read<'a, MaterialStatManifest>, + combos: ReadStorage<'a, Combo>, } /// ## Character Behavior System @@ -121,6 +122,7 @@ impl<'a> System<'a> for Sys { body, physics, stat, + combo, ) in ( &read_data.entities, &read_data.uids, @@ -135,6 +137,7 @@ impl<'a> System<'a> for Sys { &read_data.bodies, &read_data.physics_states, &read_data.stats, + &read_data.combos, ) .join() { @@ -253,6 +256,7 @@ impl<'a> System<'a> for Sys { melee_attack: read_data.melee_attacks.get(entity), beam: read_data.beams.get(entity), stat: &stat, + combo: &combo, }; for action in actions { diff --git a/common/sys/src/state.rs b/common/sys/src/state.rs index 80d00654e2..cdd3036b1e 100644 --- a/common/sys/src/state.rs +++ b/common/sys/src/state.rs @@ -127,6 +127,7 @@ impl State { ecs.register::(); ecs.register::(); ecs.register::(); + ecs.register::(); ecs.register::(); ecs.register::(); ecs.register::(); diff --git a/common/sys/src/stats.rs b/common/sys/src/stats.rs index 51f85e6184..fa9fdb8e21 100644 --- a/common/sys/src/stats.rs +++ b/common/sys/src/stats.rs @@ -1,13 +1,13 @@ use common::{ comp::{ skills::{GeneralSkill, Skill}, - Body, CharacterState, Energy, EnergyChange, EnergySource, Health, Poise, PoiseChange, - PoiseSource, Pos, Stats, + Body, CharacterState, Combo, Energy, EnergyChange, EnergySource, Health, Poise, + PoiseChange, PoiseSource, Pos, Stats, }, event::{EventBus, ServerEvent}, metrics::SysMetrics, outcome::Outcome, - resources::DeltaTime, + resources::{DeltaTime, Time}, span, uid::Uid, }; @@ -20,11 +20,13 @@ use vek::Vec3; const ENERGY_REGEN_ACCEL: f32 = 10.0; const POISE_REGEN_ACCEL: f32 = 2.0; +const COMBO_DECAY_START: f64 = 5.0; // seconds #[derive(SystemData)] pub struct ReadData<'a> { entities: Entities<'a>, dt: Read<'a, DeltaTime>, + time: Read<'a, Time>, server_bus: Read<'a, EventBus>, metrics: ReadExpect<'a, SysMetrics>, positions: ReadStorage<'a, Pos>, @@ -43,6 +45,7 @@ impl<'a> System<'a> for Sys { WriteStorage<'a, Health>, WriteStorage<'a, Poise>, WriteStorage<'a, Energy>, + WriteStorage<'a, Combo>, Write<'a, Vec>, ); @@ -54,6 +57,7 @@ impl<'a> System<'a> for Sys { mut healths, mut poises, mut energies, + mut combos, mut outcomes, ): Self::SystemData, ) { @@ -256,6 +260,13 @@ impl<'a> System<'a> for Sys { } } + // Decay combo + for (_, mut combo) in (&read_data.entities, &mut combos).join() { + if combo.counter() > 0 && read_data.time.0 - combo.last_increase() > COMBO_DECAY_START { + combo.reset(); + } + } + read_data.metrics.stats_ns.store( start_time.elapsed().as_nanos() as u64, std::sync::atomic::Ordering::Relaxed, diff --git a/server/src/events/entity_manipulation.rs b/server/src/events/entity_manipulation.rs index c3aa3a8298..cb4762e5a9 100644 --- a/server/src/events/entity_manipulation.rs +++ b/server/src/events/entity_manipulation.rs @@ -20,6 +20,7 @@ use common::{ event::{EventBus, ServerEvent}, lottery::Lottery, outcome::Outcome, + resources::Time, rtsim::RtSimEntity, terrain::{Block, TerrainGrid}, uid::{Uid, UidAllocator}, @@ -880,3 +881,15 @@ fn handle_exp_gain( exp: exp_reward as i32, }); } + +pub fn handle_combo_change(server: &Server, entity: EcsEntity, change: i32) { + let ecs = &server.state.ecs(); + if let Some(mut combo) = ecs.write_storage::().get_mut(entity) { + if change > 0 { + let time = ecs.read_resource::