mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
Polymorph effect now more correctly handles when multiple polymorph buffs are applied.
This commit is contained in:
parent
d1a7884ac8
commit
f40ef0d5dd
@ -242,6 +242,13 @@ pub enum ServerEvent {
|
|||||||
entity: EcsEntity,
|
entity: EcsEntity,
|
||||||
stance: comp::Stance,
|
stance: comp::Stance,
|
||||||
},
|
},
|
||||||
|
ChangeBody {
|
||||||
|
entity: EcsEntity,
|
||||||
|
new_body: comp::Body,
|
||||||
|
},
|
||||||
|
RemoveLightEmitter {
|
||||||
|
entity: EcsEntity,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct EventBus<E> {
|
pub struct EventBus<E> {
|
||||||
|
@ -42,26 +42,20 @@ pub struct ReadData<'a> {
|
|||||||
buffs: ReadStorage<'a, Buffs>,
|
buffs: ReadStorage<'a, Buffs>,
|
||||||
auras: ReadStorage<'a, Auras>,
|
auras: ReadStorage<'a, Auras>,
|
||||||
positions: ReadStorage<'a, Pos>,
|
positions: ReadStorage<'a, Pos>,
|
||||||
|
bodies: ReadStorage<'a, Body>,
|
||||||
|
light_emitters: ReadStorage<'a, LightEmitter>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
pub struct Sys;
|
pub struct Sys;
|
||||||
impl<'a> System<'a> for Sys {
|
impl<'a> System<'a> for Sys {
|
||||||
type SystemData = (
|
type SystemData = (ReadData<'a>, WriteStorage<'a, Stats>);
|
||||||
ReadData<'a>,
|
|
||||||
WriteStorage<'a, Stats>,
|
|
||||||
WriteStorage<'a, Body>,
|
|
||||||
WriteStorage<'a, LightEmitter>,
|
|
||||||
);
|
|
||||||
|
|
||||||
const NAME: &'static str = "buff";
|
const NAME: &'static str = "buff";
|
||||||
const ORIGIN: Origin = Origin::Common;
|
const ORIGIN: Origin = Origin::Common;
|
||||||
const PHASE: Phase = Phase::Create;
|
const PHASE: Phase = Phase::Create;
|
||||||
|
|
||||||
fn run(
|
fn run(job: &mut Job<Self>, (read_data, mut stats): Self::SystemData) {
|
||||||
job: &mut Job<Self>,
|
|
||||||
(read_data, mut stats, mut bodies, mut light_emitters): Self::SystemData,
|
|
||||||
) {
|
|
||||||
let mut server_emitter = read_data.server_bus.emitter();
|
let mut server_emitter = read_data.server_bus.emitter();
|
||||||
let dt = read_data.dt.0;
|
let dt = read_data.dt.0;
|
||||||
// Set to false to avoid spamming server
|
// Set to false to avoid spamming server
|
||||||
@ -73,9 +67,9 @@ impl<'a> System<'a> for Sys {
|
|||||||
job.cpu_stats.measure(ParMode::Rayon);
|
job.cpu_stats.measure(ParMode::Rayon);
|
||||||
let to_put_out_campfires = (
|
let to_put_out_campfires = (
|
||||||
&read_data.entities,
|
&read_data.entities,
|
||||||
&bodies,
|
&read_data.bodies,
|
||||||
&read_data.physics_states,
|
&read_data.physics_states,
|
||||||
&light_emitters, //to improve iteration speed
|
&read_data.light_emitters, //to improve iteration speed
|
||||||
)
|
)
|
||||||
.par_join()
|
.par_join()
|
||||||
.map_init(
|
.map_init(
|
||||||
@ -117,19 +111,20 @@ impl<'a> System<'a> for Sys {
|
|||||||
// slower than parallel checking above
|
// slower than parallel checking above
|
||||||
for e in to_put_out_campfires {
|
for e in to_put_out_campfires {
|
||||||
{
|
{
|
||||||
bodies
|
server_emitter.emit(ServerEvent::ChangeBody {
|
||||||
.get_mut(e)
|
entity: e,
|
||||||
.map(|mut body| *body = Body::Object(object::Body::Campfire));
|
new_body: Body::Object(object::Body::Campfire),
|
||||||
light_emitters.remove(e);
|
});
|
||||||
|
server_emitter.emit(ServerEvent::RemoveLightEmitter { entity: e });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (entity, buff_comp, mut stat, mut body, health, energy, physics_state) in (
|
for (entity, buff_comp, mut stat, body, health, energy, physics_state) in (
|
||||||
&read_data.entities,
|
&read_data.entities,
|
||||||
&read_data.buffs,
|
&read_data.buffs,
|
||||||
&mut stats,
|
&mut stats,
|
||||||
&mut bodies,
|
&read_data.bodies,
|
||||||
&read_data.healths,
|
&read_data.healths,
|
||||||
&read_data.energies,
|
&read_data.energies,
|
||||||
read_data.physics_states.maybe(),
|
read_data.physics_states.maybe(),
|
||||||
@ -331,7 +326,7 @@ impl<'a> System<'a> for Sys {
|
|||||||
// Call to reset stats to base values
|
// Call to reset stats to base values
|
||||||
stat.reset_temp_modifiers();
|
stat.reset_temp_modifiers();
|
||||||
|
|
||||||
let mut body_override = stat.original_body;
|
let mut body_override = None;
|
||||||
|
|
||||||
// Iterator over the lists of buffs by kind
|
// Iterator over the lists of buffs by kind
|
||||||
let mut buff_kinds = buff_comp
|
let mut buff_kinds = buff_comp
|
||||||
@ -371,6 +366,7 @@ impl<'a> System<'a> for Sys {
|
|||||||
kind_start_time,
|
kind_start_time,
|
||||||
&read_data,
|
&read_data,
|
||||||
&mut stat,
|
&mut stat,
|
||||||
|
body,
|
||||||
&mut body_override,
|
&mut body_override,
|
||||||
health,
|
health,
|
||||||
energy,
|
energy,
|
||||||
@ -387,8 +383,9 @@ impl<'a> System<'a> for Sys {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Update body if needed.
|
// Update body if needed.
|
||||||
if body_override != *body {
|
let new_body = body_override.unwrap_or(stat.original_body);
|
||||||
*body = body_override;
|
if new_body != *body {
|
||||||
|
server_emitter.emit(ServerEvent::ChangeBody { entity, new_body });
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove buffs that expire
|
// Remove buffs that expire
|
||||||
@ -416,6 +413,8 @@ impl<'a> System<'a> for Sys {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Globally disable this clippy lint
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn execute_effect(
|
fn execute_effect(
|
||||||
effect: &BuffEffect,
|
effect: &BuffEffect,
|
||||||
buff_kind: BuffKind,
|
buff_kind: BuffKind,
|
||||||
@ -423,7 +422,8 @@ fn execute_effect(
|
|||||||
buff_kind_start_time: Time,
|
buff_kind_start_time: Time,
|
||||||
read_data: &ReadData,
|
read_data: &ReadData,
|
||||||
stat: &mut Stats,
|
stat: &mut Stats,
|
||||||
body_override: &mut Body,
|
current_body: &Body,
|
||||||
|
body_override: &mut Option<Body>,
|
||||||
health: &Health,
|
health: &Health,
|
||||||
energy: &Energy,
|
energy: &Energy,
|
||||||
entity: Entity,
|
entity: Entity,
|
||||||
@ -618,6 +618,16 @@ fn execute_effect(
|
|||||||
BuffEffect::CriticalChance(cc) => {
|
BuffEffect::CriticalChance(cc) => {
|
||||||
stat.crit_chance_modifier *= *cc;
|
stat.crit_chance_modifier *= *cc;
|
||||||
},
|
},
|
||||||
BuffEffect::BodyChange(b) => *body_override = *b,
|
BuffEffect::BodyChange(b) => {
|
||||||
|
// For when an entity is under the effects of multiple de/buffs that change the
|
||||||
|
// body, to avoid flickering between many bodies only change the body if the
|
||||||
|
// override body is not equal to the current body. (If the buff that caused the
|
||||||
|
// current body is still active, body override will eventually pick up on it,
|
||||||
|
// otherwise this will end up with a new body, though random depending on
|
||||||
|
// iteration order)
|
||||||
|
if Some(current_body) != body_override.as_ref() {
|
||||||
|
*body_override = Some(*b)
|
||||||
|
}
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -1465,3 +1465,22 @@ pub fn handle_stance_change(server: &mut Server, entity: EcsEntity, new_stance:
|
|||||||
*stance = new_stance;
|
*stance = new_stance;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn handle_change_body(server: &mut Server, entity: EcsEntity, new_body: comp::Body) {
|
||||||
|
if let Some(mut body) = server
|
||||||
|
.state
|
||||||
|
.ecs_mut()
|
||||||
|
.write_storage::<comp::Body>()
|
||||||
|
.get_mut(entity)
|
||||||
|
{
|
||||||
|
*body = new_body;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn handle_remove_light_emitter(server: &mut Server, entity: EcsEntity) {
|
||||||
|
server
|
||||||
|
.state
|
||||||
|
.ecs_mut()
|
||||||
|
.write_storage::<comp::LightEmitter>()
|
||||||
|
.remove(entity);
|
||||||
|
}
|
||||||
|
@ -10,11 +10,12 @@ use entity_creation::{
|
|||||||
handle_shockwave, handle_shoot,
|
handle_shockwave, handle_shoot,
|
||||||
};
|
};
|
||||||
use entity_manipulation::{
|
use entity_manipulation::{
|
||||||
handle_aura, handle_bonk, handle_buff, handle_change_ability, handle_combo_change,
|
handle_aura, handle_bonk, handle_buff, handle_change_ability, handle_change_body,
|
||||||
handle_delete, handle_destroy, handle_energy_change, handle_entity_attacked_hook,
|
handle_combo_change, handle_delete, handle_destroy, handle_energy_change,
|
||||||
handle_explosion, handle_health_change, handle_knockback, handle_land_on_ground,
|
handle_entity_attacked_hook, handle_explosion, handle_health_change, handle_knockback,
|
||||||
handle_make_admin, handle_parry_hook, handle_poise, handle_respawn, handle_stance_change,
|
handle_land_on_ground, handle_make_admin, handle_parry_hook, handle_poise,
|
||||||
handle_teleport_to, handle_update_map_marker,
|
handle_remove_light_emitter, handle_respawn, handle_stance_change, handle_teleport_to,
|
||||||
|
handle_update_map_marker,
|
||||||
};
|
};
|
||||||
use group_manip::handle_group;
|
use group_manip::handle_group;
|
||||||
use information::handle_site_info;
|
use information::handle_site_info;
|
||||||
@ -303,6 +304,12 @@ impl Server {
|
|||||||
ServerEvent::ChangeStance { entity, stance } => {
|
ServerEvent::ChangeStance { entity, stance } => {
|
||||||
handle_stance_change(self, entity, stance)
|
handle_stance_change(self, entity, stance)
|
||||||
},
|
},
|
||||||
|
ServerEvent::ChangeBody { entity, new_body } => {
|
||||||
|
handle_change_body(self, entity, new_body)
|
||||||
|
},
|
||||||
|
ServerEvent::RemoveLightEmitter { entity } => {
|
||||||
|
handle_remove_light_emitter(self, entity)
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user