From d5bbe761ae7a1fdbd11ccd4f4af7006d078c0740 Mon Sep 17 00:00:00 2001 From: Sam Date: Thu, 5 Nov 2020 12:28:18 -0600 Subject: [PATCH 1/6] Created new fields in roll. Centralized generation of abilities. Added builder function for rolls since they are not yet tied to equipment. --- common/src/comp/ability.rs | 54 ++++++++++++++++++++++++++------ common/src/loadout_builder.rs | 14 ++------- common/src/states/roll.rs | 8 +++++ server/src/events/interaction.rs | 13 ++------ 4 files changed, 56 insertions(+), 33 deletions(-) diff --git a/common/src/comp/ability.rs b/common/src/comp/ability.rs index af635ab264..57c0701ba5 100644 --- a/common/src/comp/ability.rs +++ b/common/src/comp/ability.rs @@ -113,7 +113,16 @@ pub enum CharacterAbility { is_interruptible: bool, }, BasicBlock, - Roll, + Roll { + energy_cost: u32, + buildup_duration: Duration, + movement_duration: Duration, + recover_duration: Duration, + roll_strength: f32, + buildup_iframes: bool, + movement_iframes: bool, + recover_iframes: bool, + }, ComboMelee { stage_data: Vec, initial_energy_gain: u32, @@ -214,13 +223,12 @@ impl CharacterAbility { /// applicable. pub fn requirements_paid(&self, data: &JoinData, update: &mut StateUpdate) -> bool { match self { - CharacterAbility::Roll => { + CharacterAbility::Roll {energy_cost, .. } => { data.physics.on_ground - && data.body.is_humanoid() && data.vel.0.xy().magnitude_squared() > 0.5 && update .energy - .try_change_by(-220, EnergySource::Ability) + .try_change_by(-(*energy_cost as i32), EnergySource::Ability) .is_ok() }, CharacterAbility::DashMelee { energy_cost, .. } => update @@ -293,8 +301,8 @@ impl From for ItemConfig { ability1: ability_drain.next(), ability2: ability_drain.next(), ability3: ability_drain.next(), - block_ability: Some(CharacterAbility::BasicBlock), - dodge_ability: Some(CharacterAbility::Roll), + block_ability: None, + dodge_ability: Some(get_roll()), }; } @@ -302,6 +310,19 @@ impl From for ItemConfig { } } +fn get_roll() -> CharacterAbility { + CharacterAbility::Roll { + energy_cost: 100, + buildup_duration: Duration::from_millis(100), + movement_duration: Duration::from_millis(250), + recover_duration: Duration::from_millis(150), + roll_strength: 1.0, + buildup_iframes: true, + movement_iframes: true, + recover_iframes: false, + } +} + #[derive(Arraygen, Clone, PartialEq, Default, Debug, Serialize, Deserialize)] #[gen_array(pub fn get_armor: &Option)] pub struct Loadout { @@ -461,11 +482,24 @@ impl From<(&CharacterAbility, AbilityKey)> for CharacterState { exhausted: false, }), CharacterAbility::BasicBlock => CharacterState::BasicBlock, - CharacterAbility::Roll => CharacterState::Roll(roll::Data { + CharacterAbility::Roll { + energy_cost: _, + buildup_duration, + movement_duration, + recover_duration, + roll_strength, + buildup_iframes, + movement_iframes, + recover_iframes, + } => CharacterState::Roll(roll::Data { static_data: roll::StaticData { - buildup_duration: Duration::from_millis(100), - movement_duration: Duration::from_millis(300), - recover_duration: Duration::from_millis(100), + buildup_duration: *buildup_duration, + movement_duration: *movement_duration, + recover_duration: *recover_duration, + roll_strength: *roll_strength, + buildup_iframes: *buildup_iframes, + movement_iframes: *movement_iframes, + recover_iframes: *recover_iframes, }, timer: Duration::default(), stage_section: StageSection::Buildup, diff --git a/common/src/loadout_builder.rs b/common/src/loadout_builder.rs index 5511a7826d..09f104c1a4 100644 --- a/common/src/loadout_builder.rs +++ b/common/src/loadout_builder.rs @@ -144,18 +144,8 @@ impl LoadoutBuilder { _ => {}, }; - let active_item = if let Some(ItemKind::Tool(tool)) = main_tool.as_ref().map(|i| i.kind()) { - let mut abilities = tool.get_abilities(); - let mut ability_drain = abilities.drain(..); - - main_tool.map(|item| ItemConfig { - item, - ability1: ability_drain.next(), - ability2: ability_drain.next(), - ability3: ability_drain.next(), - block_ability: None, - dodge_ability: Some(CharacterAbility::Roll), - }) + let active_item = if let Some(ItemKind::Tool(_)) = main_tool.as_ref().map(|i| i.kind()) { + main_tool.map(|item| ItemConfig::from(item)) } else { Some(ItemConfig { // We need the empty item so npcs can attack diff --git a/common/src/states/roll.rs b/common/src/states/roll.rs index 172d79254d..e6144b41b9 100644 --- a/common/src/states/roll.rs +++ b/common/src/states/roll.rs @@ -19,6 +19,14 @@ pub struct StaticData { pub movement_duration: Duration, /// How long it takes to recover from roll pub recover_duration: Duration, + /// How strong the roll is + pub roll_strength: f32, + /// Whether you are immune to damage in buildup + pub buildup_iframes: bool, + /// Whether you are immune to damage in movement + pub movement_iframes: bool, + /// Whether you are immune to damage in recover + pub recover_iframes: bool, } #[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize)] diff --git a/server/src/events/interaction.rs b/server/src/events/interaction.rs index 8d034bd177..1789f94b22 100644 --- a/server/src/events/interaction.rs +++ b/server/src/events/interaction.rs @@ -176,17 +176,8 @@ pub fn handle_possess(server: &Server, possessor_uid: Uid, possesse_uid: Uid) { .or_insert(comp::Loadout::default()); let item = comp::Item::new_from_asset_expect("common.items.debug.possess"); - if let item::ItemKind::Tool(tool) = item.kind() { - let mut abilities = tool.get_abilities(); - let mut ability_drain = abilities.drain(..); - let debug_item = comp::ItemConfig { - item, - ability1: ability_drain.next(), - ability2: ability_drain.next(), - ability3: ability_drain.next(), - block_ability: None, - dodge_ability: None, - }; + if let item::ItemKind::Tool(_) = item.kind() { + let debug_item = comp::ItemConfig::from(item); std::mem::swap(&mut loadout.active_item, &mut loadout.second_item); loadout.active_item = Some(debug_item); } From 47c442423ef7e9b40ae9d24f25e2591abf9fbe55 Mon Sep 17 00:00:00 2001 From: Sam Date: Thu, 5 Nov 2020 14:22:30 -0600 Subject: [PATCH 2/6] Changed roll movement. --- common/src/comp/ability.rs | 4 ++-- common/src/loadout_builder.rs | 2 +- common/src/states/roll.rs | 27 ++++++++++++++------------- 3 files changed, 17 insertions(+), 16 deletions(-) diff --git a/common/src/comp/ability.rs b/common/src/comp/ability.rs index 57c0701ba5..623fce073d 100644 --- a/common/src/comp/ability.rs +++ b/common/src/comp/ability.rs @@ -223,7 +223,7 @@ impl CharacterAbility { /// applicable. pub fn requirements_paid(&self, data: &JoinData, update: &mut StateUpdate) -> bool { match self { - CharacterAbility::Roll {energy_cost, .. } => { + CharacterAbility::Roll { energy_cost, .. } => { data.physics.on_ground && data.vel.0.xy().magnitude_squared() > 0.5 && update @@ -316,7 +316,7 @@ fn get_roll() -> CharacterAbility { buildup_duration: Duration::from_millis(100), movement_duration: Duration::from_millis(250), recover_duration: Duration::from_millis(150), - roll_strength: 1.0, + roll_strength: 2.5, buildup_iframes: true, movement_iframes: true, recover_iframes: false, diff --git a/common/src/loadout_builder.rs b/common/src/loadout_builder.rs index 09f104c1a4..568315877e 100644 --- a/common/src/loadout_builder.rs +++ b/common/src/loadout_builder.rs @@ -145,7 +145,7 @@ impl LoadoutBuilder { }; let active_item = if let Some(ItemKind::Tool(_)) = main_tool.as_ref().map(|i| i.kind()) { - main_tool.map(|item| ItemConfig::from(item)) + main_tool.map(ItemConfig::from) } else { Some(ItemConfig { // We need the empty item so npcs can attack diff --git a/common/src/states/roll.rs b/common/src/states/roll.rs index e6144b41b9..587d314147 100644 --- a/common/src/states/roll.rs +++ b/common/src/states/roll.rs @@ -2,13 +2,9 @@ use crate::{ comp::{CharacterState, StateUpdate}, states::utils::*, sys::character_behavior::{CharacterBehavior, JoinData}, - util::Dir, }; use serde::{Deserialize, Serialize}; use std::time::Duration; -use vek::Vec3; - -const ROLL_SPEED: f32 = 25.0; /// Separated out to condense update portions of character state #[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize)] @@ -48,19 +44,12 @@ impl CharacterBehavior for Data { fn behavior(&self, data: &JoinData) -> StateUpdate { let mut update = StateUpdate::from(data); - // Update velocity - update.vel.0 = Vec3::new(0.0, 0.0, update.vel.0.z) - + (update.vel.0 * Vec3::new(1.0, 1.0, 0.0) - + 0.25 * data.inputs.move_dir.try_normalized().unwrap_or_default()) - .try_normalized() - .unwrap_or_default() - * ROLL_SPEED; - // Smooth orientation - update.ori.0 = Dir::slerp_to_vec3(update.ori.0, update.vel.0.xy().into(), 9.0 * data.dt.0); + handle_orientation(data, &mut update, 1.0); match self.stage_section { StageSection::Buildup => { + handle_move(data, &mut update, 1.0); if self.timer < self.static_data.buildup_duration { // Build up update.character = CharacterState::Roll(Data { @@ -80,6 +69,16 @@ impl CharacterBehavior for Data { } }, StageSection::Movement => { + // Update velocity + handle_forced_movement( + data, + &mut update, + ForcedMovement::Forward { + strength: self.static_data.roll_strength, + }, + 0.0, + ); + if self.timer < self.static_data.movement_duration { // Movement update.character = CharacterState::Roll(Data { @@ -123,6 +122,8 @@ impl CharacterBehavior for Data { // If it somehow ends up in an incorrect stage section if self.was_wielded { update.character = CharacterState::Wielding; + } else if self.was_sneak { + update.character = CharacterState::Sneak; } else { update.character = CharacterState::Idle; } From 5ede5e0116d6441c7ce317961575c460703f7b86 Mon Sep 17 00:00:00 2001 From: jshipsey Date: Thu, 5 Nov 2020 16:40:30 -0500 Subject: [PATCH 3/6] roll anim speedup --- voxygen/src/anim/src/character/roll.rs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/voxygen/src/anim/src/character/roll.rs b/voxygen/src/anim/src/character/roll.rs index 8ff76d0a96..3621311641 100644 --- a/voxygen/src/anim/src/character/roll.rs +++ b/voxygen/src/anim/src/character/roll.rs @@ -31,7 +31,8 @@ impl Animation for RollAnimation { *rate = 1.0; let mut next = (*skeleton).clone(); - let spin = anim_time as f32; + let spin = anim_time as f32 * 1.1; + let ori: Vec2 = Vec2::from(orientation); let last_ori = Vec2::from(last_ori); let tilt = if ::vek::Vec2::new(ori, last_ori) From 97b988a74794d2c2637cf150fdb062284de5f29f Mon Sep 17 00:00:00 2001 From: Sam Date: Thu, 5 Nov 2020 16:48:04 -0600 Subject: [PATCH 4/6] Roll now gives i-frames. --- common/src/comp/character_state.rs | 14 +++++++++++++ common/src/sys/beam.rs | 25 ++++++++++++++++++++---- common/src/sys/melee.rs | 22 +++++++++++++++++---- common/src/sys/projectile.rs | 18 +++++++++++++++-- common/src/sys/shockwave.rs | 15 +++++++++++--- server/src/events/entity_manipulation.rs | 9 ++++++++- 6 files changed, 89 insertions(+), 14 deletions(-) diff --git a/common/src/comp/character_state.rs b/common/src/comp/character_state.rs index f1b7e91502..e90ad1d404 100644 --- a/common/src/comp/character_state.rs +++ b/common/src/comp/character_state.rs @@ -145,6 +145,20 @@ impl CharacterState { // Check if state is the same without looking at the inner data std::mem::discriminant(self) == std::mem::discriminant(other) } + + pub fn is_invincible(&self) -> bool { + if let CharacterState::Roll(data) = self { + use utils::StageSection; + match data.stage_section { + StageSection::Buildup => data.static_data.buildup_iframes, + StageSection::Movement => data.static_data.movement_iframes, + StageSection::Recover => data.static_data.recover_iframes, + _ => false, + } + } else { + false + } + } } impl Default for CharacterState { diff --git a/common/src/sys/beam.rs b/common/src/sys/beam.rs index 0642e0e9f7..d5cd98eb98 100644 --- a/common/src/sys/beam.rs +++ b/common/src/sys/beam.rs @@ -1,7 +1,7 @@ use crate::{ comp::{ - group, Beam, BeamSegment, Body, Energy, EnergyChange, EnergySource, Health, HealthChange, - HealthSource, Last, Loadout, Ori, Pos, Scale, + group, Beam, BeamSegment, Body, CharacterState, Energy, EnergyChange, EnergySource, Health, + HealthChange, HealthSource, Last, Loadout, Ori, Pos, Scale, }, event::{EventBus, ServerEvent}, state::{DeltaTime, Time}, @@ -34,6 +34,7 @@ impl<'a> System<'a> for Sys { ReadStorage<'a, Energy>, WriteStorage<'a, BeamSegment>, WriteStorage<'a, Beam>, + ReadStorage<'a, CharacterState>, ); fn run( @@ -56,6 +57,7 @@ impl<'a> System<'a> for Sys { energies, mut beam_segments, mut beams, + char_states, ): Self::SystemData, ) { let mut server_emitter = server_bus.emitter(); @@ -112,7 +114,16 @@ impl<'a> System<'a> for Sys { }; // Go through all other effectable entities - for (b, uid_b, pos_b, last_pos_b_maybe, scale_b_maybe, health_b, body_b) in ( + for ( + b, + uid_b, + pos_b, + last_pos_b_maybe, + scale_b_maybe, + health_b, + body_b, + char_state_b_maybe, + ) in ( &entities, &uids, &positions, @@ -121,6 +132,7 @@ impl<'a> System<'a> for Sys { scales.maybe(), &healths, &bodies, + char_states.maybe(), ) .join() { @@ -134,6 +146,9 @@ impl<'a> System<'a> for Sys { let rad_b = body_b.radius() * scale_b; let height_b = body_b.height() * scale_b; + // Check if entity is immune to damage + let is_invincible = char_state_b_maybe.map_or(false, |c_s| c_s.is_invincible()); + // Check if it is a hit let hit = entity != b && !health_b.is_dead @@ -160,7 +175,9 @@ impl<'a> System<'a> for Sys { for (target, damage) in beam_segment.damages.iter() { if let Some(target) = target { - if *target != target_group { + if *target != target_group + || (!matches!(target, GroupTarget::InGroup) && is_invincible) + { continue; } } diff --git a/common/src/sys/melee.rs b/common/src/sys/melee.rs index c22371a9d4..0377765c63 100644 --- a/common/src/sys/melee.rs +++ b/common/src/sys/melee.rs @@ -1,5 +1,5 @@ use crate::{ - comp::{buff, group, Attacking, Body, Health, Loadout, Ori, Pos, Scale}, + comp::{buff, group, Attacking, Body, CharacterState, Health, Loadout, Ori, Pos, Scale}, event::{EventBus, LocalEvent, ServerEvent}, metrics::SysMetrics, span, @@ -31,6 +31,7 @@ impl<'a> System<'a> for Sys { ReadStorage<'a, Loadout>, ReadStorage<'a, group::Group>, WriteStorage<'a, Attacking>, + ReadStorage<'a, CharacterState>, ); fn run( @@ -49,6 +50,7 @@ impl<'a> System<'a> for Sys { loadouts, groups, mut attacking_storage, + char_states, ): Self::SystemData, ) { let start_time = std::time::Instant::now(); @@ -72,8 +74,15 @@ impl<'a> System<'a> for Sys { attack.applied = true; // Go through all other entities - for (b, pos_b, scale_b_maybe, health_b, body_b) in - (&entities, &positions, scales.maybe(), &healths, &bodies).join() + for (b, pos_b, scale_b_maybe, health_b, body_b, char_state_b_maybe) in ( + &entities, + &positions, + scales.maybe(), + &healths, + &bodies, + char_states.maybe(), + ) + .join() { // 2D versions let pos2 = Vec2::from(pos.0); @@ -85,6 +94,9 @@ impl<'a> System<'a> for Sys { let scale_b = scale_b_maybe.map_or(1.0, |s| s.0); let rad_b = body_b.radius() * scale_b; + // Check if entity is invincible + let is_invincible = char_state_b_maybe.map_or(false, |c_s| c_s.is_invincible()); + // Check if it is a hit if entity != b && !health_b.is_dead @@ -106,7 +118,9 @@ impl<'a> System<'a> for Sys { for (target, damage) in attack.damages.iter() { if let Some(target) = target { - if *target != target_group { + if *target != target_group + || (!matches!(target, GroupTarget::InGroup) && is_invincible) + { continue; } } diff --git a/common/src/sys/projectile.rs b/common/src/sys/projectile.rs index 0e4a8b2a07..023a7ce998 100644 --- a/common/src/sys/projectile.rs +++ b/common/src/sys/projectile.rs @@ -1,8 +1,8 @@ use crate::{ comp::{ buff::{BuffChange, BuffSource}, - projectile, EnergyChange, EnergySource, Group, HealthSource, Loadout, Ori, PhysicsState, - Pos, Projectile, Vel, + projectile, CharacterState, EnergyChange, EnergySource, Group, HealthSource, Loadout, Ori, + PhysicsState, Pos, Projectile, Vel, }, event::{EventBus, LocalEvent, ServerEvent}, metrics::SysMetrics, @@ -35,6 +35,7 @@ impl<'a> System<'a> for Sys { WriteStorage<'a, Projectile>, ReadStorage<'a, Loadout>, ReadStorage<'a, Group>, + ReadStorage<'a, CharacterState>, ); fn run( @@ -53,6 +54,7 @@ impl<'a> System<'a> for Sys { mut projectiles, loadouts, groups, + char_states, ): Self::SystemData, ) { let start_time = std::time::Instant::now(); @@ -102,6 +104,18 @@ impl<'a> System<'a> for Sys { continue; } + // Checks if entity is immune to damage + // TODO: When projectiles are reduced down to a collection of (target, effect)s, + // move this check there so that future projectiles intended for allies cannot + // be dodged by those allies + let entity_invincible = uid_allocator + .retrieve_entity_internal(other.into()) + .and_then(|e| char_states.get(e)) + .map_or(false, |c_s| c_s.is_invincible()); + if entity_invincible { + continue; + } + for effect in projectile.hit_entity.drain(..) { match effect { projectile::Effect::Damage(target, damage) => { diff --git a/common/src/sys/shockwave.rs b/common/src/sys/shockwave.rs index de13dfc17a..5d02caa340 100644 --- a/common/src/sys/shockwave.rs +++ b/common/src/sys/shockwave.rs @@ -1,7 +1,7 @@ use crate::{ comp::{ - group, Body, Health, HealthSource, Last, Loadout, Ori, PhysicsState, Pos, Scale, Shockwave, - ShockwaveHitEntities, + group, Body, CharacterState, Health, HealthSource, Last, Loadout, Ori, PhysicsState, Pos, + Scale, Shockwave, ShockwaveHitEntities, }, event::{EventBus, LocalEvent, ServerEvent}, state::{DeltaTime, Time}, @@ -36,6 +36,7 @@ impl<'a> System<'a> for Sys { ReadStorage<'a, PhysicsState>, WriteStorage<'a, Shockwave>, WriteStorage<'a, ShockwaveHitEntities>, + ReadStorage<'a, CharacterState>, ); fn run( @@ -59,6 +60,7 @@ impl<'a> System<'a> for Sys { physics_states, mut shockwaves, mut shockwave_hit_lists, + char_states, ): Self::SystemData, ) { let mut server_emitter = server_bus.emitter(); @@ -131,6 +133,7 @@ impl<'a> System<'a> for Sys { health_b, body_b, physics_state_b, + char_state_b_maybe, ) in ( &entities, &uids, @@ -141,6 +144,7 @@ impl<'a> System<'a> for Sys { &healths, &bodies, &physics_states, + char_states.maybe(), ) .join() { @@ -176,6 +180,9 @@ impl<'a> System<'a> for Sys { GroupTarget::OutOfGroup }; + // Check if entity is immune to damage + let is_invincible = char_state_b_maybe.map_or(false, |c_s| c_s.is_invincible()); + // Check if it is a hit let hit = entity != b && !health_b.is_dead @@ -193,7 +200,9 @@ impl<'a> System<'a> for Sys { if hit { for (target, damage) in shockwave.damages.iter() { if let Some(target) = target { - if *target != target_group { + if *target != target_group + || (!matches!(target, GroupTarget::InGroup) && is_invincible) + { continue; } } diff --git a/server/src/events/entity_manipulation.rs b/server/src/events/entity_manipulation.rs index 91717fd532..05c42172fd 100644 --- a/server/src/events/entity_manipulation.rs +++ b/server/src/events/entity_manipulation.rs @@ -647,7 +647,14 @@ pub fn handle_explosion( .get(entity_b) .map_or(false, |h| !h.is_dead); - if is_alive { + let is_invincible = ecs + .read_storage::() + .get(entity_b) + .map_or(false, |c_s| c_s.is_invincible()); + + if is_alive + && (matches!(target, Some(GroupTarget::InGroup)) || !is_invincible) + { effect.modify_strength(strength); server.state().apply_effect(entity_b, effect, owner); // Apply energy change From 34156f2cfaf62c9dc8c1870f6bab0a126535e303 Mon Sep 17 00:00:00 2001 From: Sam Date: Thu, 5 Nov 2020 18:01:46 -0600 Subject: [PATCH 5/6] Fixed audio tests. Added changelog line. --- CHANGELOG.md | 1 + voxygen/src/audio/sfx/event_mapper/movement/tests.rs | 4 ++++ 2 files changed, 5 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index ac752d39f2..13faa58783 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -28,6 +28,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Buff system - Sneaking lets you be closer to enemies without being detected - Flight +- Roll has i-frames ### Changed diff --git a/voxygen/src/audio/sfx/event_mapper/movement/tests.rs b/voxygen/src/audio/sfx/event_mapper/movement/tests.rs index 8b557a74b1..c054cd9e2a 100644 --- a/voxygen/src/audio/sfx/event_mapper/movement/tests.rs +++ b/voxygen/src/audio/sfx/event_mapper/movement/tests.rs @@ -157,6 +157,10 @@ fn maps_roll() { buildup_duration: Duration::default(), movement_duration: Duration::default(), recover_duration: Duration::default(), + roll_strength: 0.0, + buildup_iframes: false, + movement_iframes: false, + recover_iframes: false, }, timer: Duration::default(), stage_section: states::utils::StageSection::Buildup, From bd6acca55c98991c1d9742c5d595638990f486f2 Mon Sep 17 00:00:00 2001 From: Sam Date: Thu, 5 Nov 2020 19:22:26 -0600 Subject: [PATCH 6/6] Addressed comments. Roll now only provides i-frames to melee attacks. It also reduces height of hitbox. --- CHANGELOG.md | 2 +- common/src/comp/ability.rs | 34 +++------- common/src/comp/character_state.rs | 14 ---- common/src/comp/phys.rs | 4 +- common/src/states/roll.rs | 8 +-- common/src/sys/beam.rs | 25 ++----- common/src/sys/melee.rs | 6 +- common/src/sys/phys.rs | 67 ++++++++++++++----- common/src/sys/projectile.rs | 18 +---- common/src/sys/shockwave.rs | 15 +---- server/src/events/entity_manipulation.rs | 9 +-- .../audio/sfx/event_mapper/movement/tests.rs | 3 - 12 files changed, 79 insertions(+), 126 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 13faa58783..d58d790b2c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -28,7 +28,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Buff system - Sneaking lets you be closer to enemies without being detected - Flight -- Roll has i-frames +- Roll dodges melee attacks, and reduces the height of your hitbox ### Changed diff --git a/common/src/comp/ability.rs b/common/src/comp/ability.rs index 623fce073d..47e6213182 100644 --- a/common/src/comp/ability.rs +++ b/common/src/comp/ability.rs @@ -119,9 +119,6 @@ pub enum CharacterAbility { movement_duration: Duration, recover_duration: Duration, roll_strength: f32, - buildup_iframes: bool, - movement_iframes: bool, - recover_iframes: bool, }, ComboMelee { stage_data: Vec, @@ -278,6 +275,16 @@ impl CharacterAbility { _ => true, } } + + fn default_roll() -> CharacterAbility { + CharacterAbility::Roll { + energy_cost: 100, + buildup_duration: Duration::from_millis(100), + movement_duration: Duration::from_millis(250), + recover_duration: Duration::from_millis(150), + roll_strength: 2.5, + } + } } #[derive(Clone, PartialEq, Debug, Serialize, Deserialize)] @@ -302,7 +309,7 @@ impl From for ItemConfig { ability2: ability_drain.next(), ability3: ability_drain.next(), block_ability: None, - dodge_ability: Some(get_roll()), + dodge_ability: Some(CharacterAbility::default_roll()), }; } @@ -310,19 +317,6 @@ impl From for ItemConfig { } } -fn get_roll() -> CharacterAbility { - CharacterAbility::Roll { - energy_cost: 100, - buildup_duration: Duration::from_millis(100), - movement_duration: Duration::from_millis(250), - recover_duration: Duration::from_millis(150), - roll_strength: 2.5, - buildup_iframes: true, - movement_iframes: true, - recover_iframes: false, - } -} - #[derive(Arraygen, Clone, PartialEq, Default, Debug, Serialize, Deserialize)] #[gen_array(pub fn get_armor: &Option)] pub struct Loadout { @@ -488,18 +482,12 @@ impl From<(&CharacterAbility, AbilityKey)> for CharacterState { movement_duration, recover_duration, roll_strength, - buildup_iframes, - movement_iframes, - recover_iframes, } => CharacterState::Roll(roll::Data { static_data: roll::StaticData { buildup_duration: *buildup_duration, movement_duration: *movement_duration, recover_duration: *recover_duration, roll_strength: *roll_strength, - buildup_iframes: *buildup_iframes, - movement_iframes: *movement_iframes, - recover_iframes: *recover_iframes, }, timer: Duration::default(), stage_section: StageSection::Buildup, diff --git a/common/src/comp/character_state.rs b/common/src/comp/character_state.rs index e90ad1d404..f1b7e91502 100644 --- a/common/src/comp/character_state.rs +++ b/common/src/comp/character_state.rs @@ -145,20 +145,6 @@ impl CharacterState { // Check if state is the same without looking at the inner data std::mem::discriminant(self) == std::mem::discriminant(other) } - - pub fn is_invincible(&self) -> bool { - if let CharacterState::Roll(data) = self { - use utils::StageSection; - match data.stage_section { - StageSection::Buildup => data.static_data.buildup_iframes, - StageSection::Movement => data.static_data.movement_iframes, - StageSection::Recover => data.static_data.recover_iframes, - _ => false, - } - } else { - false - } - } } impl Default for CharacterState { diff --git a/common/src/comp/phys.rs b/common/src/comp/phys.rs index c59c038333..902d391737 100644 --- a/common/src/comp/phys.rs +++ b/common/src/comp/phys.rs @@ -73,9 +73,9 @@ impl Collider { } } - pub fn get_z_limits(&self) -> (f32, f32) { + pub fn get_z_limits(&self, modifier: f32) -> (f32, f32) { match self { - Collider::Box { z_min, z_max, .. } => (*z_min, *z_max), + Collider::Box { z_min, z_max, .. } => (*z_min * modifier, *z_max * modifier), Collider::Point => (0.0, 0.0), } } diff --git a/common/src/states/roll.rs b/common/src/states/roll.rs index 587d314147..45decbb376 100644 --- a/common/src/states/roll.rs +++ b/common/src/states/roll.rs @@ -15,14 +15,8 @@ pub struct StaticData { pub movement_duration: Duration, /// How long it takes to recover from roll pub recover_duration: Duration, - /// How strong the roll is + /// Affects the speed and distance of the roll pub roll_strength: f32, - /// Whether you are immune to damage in buildup - pub buildup_iframes: bool, - /// Whether you are immune to damage in movement - pub movement_iframes: bool, - /// Whether you are immune to damage in recover - pub recover_iframes: bool, } #[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize)] diff --git a/common/src/sys/beam.rs b/common/src/sys/beam.rs index d5cd98eb98..0642e0e9f7 100644 --- a/common/src/sys/beam.rs +++ b/common/src/sys/beam.rs @@ -1,7 +1,7 @@ use crate::{ comp::{ - group, Beam, BeamSegment, Body, CharacterState, Energy, EnergyChange, EnergySource, Health, - HealthChange, HealthSource, Last, Loadout, Ori, Pos, Scale, + group, Beam, BeamSegment, Body, Energy, EnergyChange, EnergySource, Health, HealthChange, + HealthSource, Last, Loadout, Ori, Pos, Scale, }, event::{EventBus, ServerEvent}, state::{DeltaTime, Time}, @@ -34,7 +34,6 @@ impl<'a> System<'a> for Sys { ReadStorage<'a, Energy>, WriteStorage<'a, BeamSegment>, WriteStorage<'a, Beam>, - ReadStorage<'a, CharacterState>, ); fn run( @@ -57,7 +56,6 @@ impl<'a> System<'a> for Sys { energies, mut beam_segments, mut beams, - char_states, ): Self::SystemData, ) { let mut server_emitter = server_bus.emitter(); @@ -114,16 +112,7 @@ impl<'a> System<'a> for Sys { }; // Go through all other effectable entities - for ( - b, - uid_b, - pos_b, - last_pos_b_maybe, - scale_b_maybe, - health_b, - body_b, - char_state_b_maybe, - ) in ( + for (b, uid_b, pos_b, last_pos_b_maybe, scale_b_maybe, health_b, body_b) in ( &entities, &uids, &positions, @@ -132,7 +121,6 @@ impl<'a> System<'a> for Sys { scales.maybe(), &healths, &bodies, - char_states.maybe(), ) .join() { @@ -146,9 +134,6 @@ impl<'a> System<'a> for Sys { let rad_b = body_b.radius() * scale_b; let height_b = body_b.height() * scale_b; - // Check if entity is immune to damage - let is_invincible = char_state_b_maybe.map_or(false, |c_s| c_s.is_invincible()); - // Check if it is a hit let hit = entity != b && !health_b.is_dead @@ -175,9 +160,7 @@ impl<'a> System<'a> for Sys { for (target, damage) in beam_segment.damages.iter() { if let Some(target) = target { - if *target != target_group - || (!matches!(target, GroupTarget::InGroup) && is_invincible) - { + if *target != target_group { continue; } } diff --git a/common/src/sys/melee.rs b/common/src/sys/melee.rs index 0377765c63..f284cb390d 100644 --- a/common/src/sys/melee.rs +++ b/common/src/sys/melee.rs @@ -94,8 +94,8 @@ impl<'a> System<'a> for Sys { let scale_b = scale_b_maybe.map_or(1.0, |s| s.0); let rad_b = body_b.radius() * scale_b; - // Check if entity is invincible - let is_invincible = char_state_b_maybe.map_or(false, |c_s| c_s.is_invincible()); + // Check if entity is dodging + let is_dodge = char_state_b_maybe.map_or(false, |c_s| c_s.is_dodge()); // Check if it is a hit if entity != b @@ -119,7 +119,7 @@ impl<'a> System<'a> for Sys { for (target, damage) in attack.damages.iter() { if let Some(target) = target { if *target != target_group - || (!matches!(target, GroupTarget::InGroup) && is_invincible) + || (!matches!(target, GroupTarget::InGroup) && is_dodge) { continue; } diff --git a/common/src/sys/phys.rs b/common/src/sys/phys.rs index b9e75feb0d..f8d8bd5f3c 100644 --- a/common/src/sys/phys.rs +++ b/common/src/sys/phys.rs @@ -1,7 +1,7 @@ use crate::{ comp::{ - BeamSegment, Collider, Gravity, Mass, Mounting, Ori, PhysicsState, Pos, PreviousVelDtCache, - Projectile, Scale, Shockwave, Sticky, Vel, + BeamSegment, CharacterState, Collider, Gravity, Mass, Mounting, Ori, PhysicsState, Pos, + PreviousVelDtCache, Projectile, Scale, Shockwave, Sticky, Vel, }, event::{EventBus, ServerEvent}, metrics::{PhysicsMetrics, SysMetrics}, @@ -71,6 +71,7 @@ impl<'a> System<'a> for Sys { ReadStorage<'a, Projectile>, ReadStorage<'a, BeamSegment>, ReadStorage<'a, Shockwave>, + ReadStorage<'a, CharacterState>, ); #[allow(clippy::or_fun_call)] // TODO: Pending review in #587 @@ -99,6 +100,7 @@ impl<'a> System<'a> for Sys { projectiles, beams, shockwaves, + char_states, ): Self::SystemData, ) { let start_time = std::time::Instant::now(); @@ -183,17 +185,38 @@ impl<'a> System<'a> for Sys { // TODO: if we need to avoid collisions for other things consider moving whether it // should interact into the collider component or into a separate component projectiles.maybe(), + char_states.maybe(), ) .par_join() - .filter(|(_, _, _, _, _, _, _, _, sticky, physics, _)| { + .filter(|(_, _, _, _, _, _, _, _, sticky, physics, _, _)| { sticky.is_none() || (physics.on_wall.is_none() && !physics.on_ground) }) - .map(|(e, p, v, vd, s, m, c, _, _, ph, pr)| (e, p, v, vd, s, m, c, ph, pr)) - .fold(PhysicsMetrics::default, - |mut metrics,(entity, pos, vel, vel_dt, scale, mass, collider, physics, projectile)| { + .map(|(e, p, v, vd, s, m, c, _, _, ph, pr, c_s)| (e, p, v, vd, s, m, c, ph, pr, c_s)) + .fold( + PhysicsMetrics::default, + |mut metrics, + ( + entity, + pos, + vel, + vel_dt, + scale, + mass, + collider, + physics, + projectile, + char_state_maybe, + )| { let scale = scale.map(|s| s.0).unwrap_or(1.0); let radius = collider.map(|c| c.get_radius()).unwrap_or(0.5); - let z_limits = collider.map(|c| c.get_z_limits()).unwrap_or((-0.5, 0.5)); + let modifier = if char_state_maybe.map_or(false, |c_s| c_s.is_dodge()) { + 0.5 + } else { + 1.0 + }; + let z_limits = collider + .map(|c| c.get_z_limits(modifier)) + .unwrap_or((-0.5 * modifier, 0.5 * modifier)); let mass = mass.map(|m| m.0).unwrap_or(scale); // Resets touch_entities in physics @@ -215,6 +238,7 @@ impl<'a> System<'a> for Sys { _, _, _, + char_state_other_maybe, ) in ( &entities, &uids, @@ -227,6 +251,7 @@ impl<'a> System<'a> for Sys { !&mountings, !&beams, !&shockwaves, + char_states.maybe(), ) .join() { @@ -240,13 +265,22 @@ impl<'a> System<'a> for Sys { let collision_dist = scale * radius + scale_other * radius_other; // Sanity check: skip colliding entities that are too far from each other - if (pos.0 - pos_other.0).xy().magnitude() > (vel_dt.0 - vel_dt_other.0).xy().magnitude() + collision_dist { + if (pos.0 - pos_other.0).xy().magnitude() + > (vel_dt.0 - vel_dt_other.0).xy().magnitude() + collision_dist + { continue; } + let modifier_other = + if char_state_other_maybe.map_or(false, |c_s| c_s.is_dodge()) { + 0.5 + } else { + 1.0 + }; + let z_limits_other = collider_other - .map(|c| c.get_z_limits()) - .unwrap_or((-0.5, 0.5)); + .map(|c| c.get_z_limits(modifier_other)) + .unwrap_or((-0.5 * modifier_other, 0.5 * modifier_other)); let mass_other = mass_other.map(|m| m.0).unwrap_or(scale_other); //This check after the pos check, as we currently don't have that many // massless entites [citation needed] @@ -257,7 +291,8 @@ impl<'a> System<'a> for Sys { metrics.entity_entity_collision_checks += 1; const MIN_COLLISION_DIST: f32 = 0.3; - let increments = ((vel_dt.0 - vel_dt_other.0).magnitude() / MIN_COLLISION_DIST) + let increments = ((vel_dt.0 - vel_dt_other.0).magnitude() + / MIN_COLLISION_DIST) .max(1.0) .ceil() as usize; let step_delta = 1.0 / increments as f32; @@ -300,11 +335,11 @@ impl<'a> System<'a> for Sys { metrics }, ) - .reduce(PhysicsMetrics::default, |old, new| { - PhysicsMetrics { - entity_entity_collision_checks: old.entity_entity_collision_checks + new.entity_entity_collision_checks, - entity_entity_collisions: old.entity_entity_collisions + new.entity_entity_collisions, - } + .reduce(PhysicsMetrics::default, |old, new| PhysicsMetrics { + entity_entity_collision_checks: old.entity_entity_collision_checks + + new.entity_entity_collision_checks, + entity_entity_collisions: old.entity_entity_collisions + + new.entity_entity_collisions, }); physics_metrics.entity_entity_collision_checks = metrics.entity_entity_collision_checks; physics_metrics.entity_entity_collisions = metrics.entity_entity_collisions; diff --git a/common/src/sys/projectile.rs b/common/src/sys/projectile.rs index 023a7ce998..0e4a8b2a07 100644 --- a/common/src/sys/projectile.rs +++ b/common/src/sys/projectile.rs @@ -1,8 +1,8 @@ use crate::{ comp::{ buff::{BuffChange, BuffSource}, - projectile, CharacterState, EnergyChange, EnergySource, Group, HealthSource, Loadout, Ori, - PhysicsState, Pos, Projectile, Vel, + projectile, EnergyChange, EnergySource, Group, HealthSource, Loadout, Ori, PhysicsState, + Pos, Projectile, Vel, }, event::{EventBus, LocalEvent, ServerEvent}, metrics::SysMetrics, @@ -35,7 +35,6 @@ impl<'a> System<'a> for Sys { WriteStorage<'a, Projectile>, ReadStorage<'a, Loadout>, ReadStorage<'a, Group>, - ReadStorage<'a, CharacterState>, ); fn run( @@ -54,7 +53,6 @@ impl<'a> System<'a> for Sys { mut projectiles, loadouts, groups, - char_states, ): Self::SystemData, ) { let start_time = std::time::Instant::now(); @@ -104,18 +102,6 @@ impl<'a> System<'a> for Sys { continue; } - // Checks if entity is immune to damage - // TODO: When projectiles are reduced down to a collection of (target, effect)s, - // move this check there so that future projectiles intended for allies cannot - // be dodged by those allies - let entity_invincible = uid_allocator - .retrieve_entity_internal(other.into()) - .and_then(|e| char_states.get(e)) - .map_or(false, |c_s| c_s.is_invincible()); - if entity_invincible { - continue; - } - for effect in projectile.hit_entity.drain(..) { match effect { projectile::Effect::Damage(target, damage) => { diff --git a/common/src/sys/shockwave.rs b/common/src/sys/shockwave.rs index 5d02caa340..de13dfc17a 100644 --- a/common/src/sys/shockwave.rs +++ b/common/src/sys/shockwave.rs @@ -1,7 +1,7 @@ use crate::{ comp::{ - group, Body, CharacterState, Health, HealthSource, Last, Loadout, Ori, PhysicsState, Pos, - Scale, Shockwave, ShockwaveHitEntities, + group, Body, Health, HealthSource, Last, Loadout, Ori, PhysicsState, Pos, Scale, Shockwave, + ShockwaveHitEntities, }, event::{EventBus, LocalEvent, ServerEvent}, state::{DeltaTime, Time}, @@ -36,7 +36,6 @@ impl<'a> System<'a> for Sys { ReadStorage<'a, PhysicsState>, WriteStorage<'a, Shockwave>, WriteStorage<'a, ShockwaveHitEntities>, - ReadStorage<'a, CharacterState>, ); fn run( @@ -60,7 +59,6 @@ impl<'a> System<'a> for Sys { physics_states, mut shockwaves, mut shockwave_hit_lists, - char_states, ): Self::SystemData, ) { let mut server_emitter = server_bus.emitter(); @@ -133,7 +131,6 @@ impl<'a> System<'a> for Sys { health_b, body_b, physics_state_b, - char_state_b_maybe, ) in ( &entities, &uids, @@ -144,7 +141,6 @@ impl<'a> System<'a> for Sys { &healths, &bodies, &physics_states, - char_states.maybe(), ) .join() { @@ -180,9 +176,6 @@ impl<'a> System<'a> for Sys { GroupTarget::OutOfGroup }; - // Check if entity is immune to damage - let is_invincible = char_state_b_maybe.map_or(false, |c_s| c_s.is_invincible()); - // Check if it is a hit let hit = entity != b && !health_b.is_dead @@ -200,9 +193,7 @@ impl<'a> System<'a> for Sys { if hit { for (target, damage) in shockwave.damages.iter() { if let Some(target) = target { - if *target != target_group - || (!matches!(target, GroupTarget::InGroup) && is_invincible) - { + if *target != target_group { continue; } } diff --git a/server/src/events/entity_manipulation.rs b/server/src/events/entity_manipulation.rs index 05c42172fd..91717fd532 100644 --- a/server/src/events/entity_manipulation.rs +++ b/server/src/events/entity_manipulation.rs @@ -647,14 +647,7 @@ pub fn handle_explosion( .get(entity_b) .map_or(false, |h| !h.is_dead); - let is_invincible = ecs - .read_storage::() - .get(entity_b) - .map_or(false, |c_s| c_s.is_invincible()); - - if is_alive - && (matches!(target, Some(GroupTarget::InGroup)) || !is_invincible) - { + if is_alive { effect.modify_strength(strength); server.state().apply_effect(entity_b, effect, owner); // Apply energy change diff --git a/voxygen/src/audio/sfx/event_mapper/movement/tests.rs b/voxygen/src/audio/sfx/event_mapper/movement/tests.rs index c054cd9e2a..0001b867ff 100644 --- a/voxygen/src/audio/sfx/event_mapper/movement/tests.rs +++ b/voxygen/src/audio/sfx/event_mapper/movement/tests.rs @@ -158,9 +158,6 @@ fn maps_roll() { movement_duration: Duration::default(), recover_duration: Duration::default(), roll_strength: 0.0, - buildup_iframes: false, - movement_iframes: false, - recover_iframes: false, }, timer: Duration::default(), stage_section: states::utils::StageSection::Buildup,