diff --git a/CHANGELOG.md b/CHANGELOG.md index cf2bf6b0ea..d8778c1399 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -81,6 +81,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Adjusted some keybindings - Consumables can now trigger multiple effects and buffs - Overhauled overworld spawns depending on chunk attributes +- Fixed a bug which caused campfires and other stuff to duplicate ### Removed diff --git a/common/src/comp/home_chunk.rs b/common/src/comp/home_chunk.rs new file mode 100644 index 0000000000..0389d2815a --- /dev/null +++ b/common/src/comp/home_chunk.rs @@ -0,0 +1,13 @@ +use specs::Component; +use specs_idvs::IdvStorage; +use vek::Vec2; + +/// This component exists in order to fix a bug that caused entitites +/// such as campfires to duplicate because the chunk was double-loaded. +/// See https://gitlab.com/veloren/veloren/-/merge_requests/1543 +#[derive(Copy, Clone, Default, Debug, PartialEq)] +pub struct HomeChunk(pub Vec2); + +impl Component for HomeChunk { + type Storage = IdvStorage; +} diff --git a/common/src/comp/mod.rs b/common/src/comp/mod.rs index 362d0ad06a..774f4fcb2a 100644 --- a/common/src/comp/mod.rs +++ b/common/src/comp/mod.rs @@ -10,6 +10,7 @@ mod controller; mod energy; pub mod group; mod health; +pub mod home_chunk; mod inputs; mod inventory; mod last; @@ -47,6 +48,7 @@ pub use controller::{ pub use energy::{Energy, EnergyChange, EnergySource}; pub use group::Group; pub use health::{Health, HealthChange, HealthSource}; +pub use home_chunk::HomeChunk; pub use inputs::CanBuild; pub use inventory::{ item, diff --git a/common/src/event.rs b/common/src/event.rs index fee8baf6a0..e9da671203 100644 --- a/common/src/event.rs +++ b/common/src/event.rs @@ -104,6 +104,7 @@ pub enum ServerEvent { agent: Option, alignment: comp::Alignment, scale: comp::Scale, + home_chunk: Option, drop_item: Option, }, CreateWaypoint(Vec3), diff --git a/server/src/events/entity_creation.rs b/server/src/events/entity_creation.rs index 6d8aa4b1de..e5cd9e7bfb 100644 --- a/server/src/events/entity_creation.rs +++ b/server/src/events/entity_creation.rs @@ -2,7 +2,7 @@ use crate::{sys, Server, StateExt}; use common::{ character::CharacterId, comp::{ - self, beam, shockwave, Agent, Alignment, Body, Gravity, Health, Item, ItemDrop, + self, beam, shockwave, Agent, Alignment, Body, Gravity, Health, HomeChunk, Item, ItemDrop, LightEmitter, Loadout, Ori, Pos, Projectile, Scale, Stats, Vel, WaypointArea, }, outcome::Outcome, @@ -49,6 +49,7 @@ pub fn handle_create_npc( alignment: Alignment, scale: Scale, drop_item: Option, + home_chunk: Option, ) { let group = match alignment { Alignment::Wild => None, @@ -83,6 +84,12 @@ pub fn handle_create_npc( entity }; + let entity = if let Some(home_chunk) = home_chunk { + entity.with(home_chunk) + } else { + entity + }; + entity.build(); } diff --git a/server/src/events/mod.rs b/server/src/events/mod.rs index edcc319842..a4707a971e 100644 --- a/server/src/events/mod.rs +++ b/server/src/events/mod.rs @@ -115,9 +115,11 @@ impl Server { agent, alignment, scale, + home_chunk, drop_item, } => handle_create_npc( self, pos, stats, health, loadout, body, agent, alignment, scale, drop_item, + home_chunk, ), ServerEvent::CreateWaypoint(pos) => handle_create_waypoint(self, pos), ServerEvent::ClientDisconnect(entity) => { diff --git a/server/src/lib.rs b/server/src/lib.rs index 1e68a88dee..68576c24ab 100644 --- a/server/src/lib.rs +++ b/server/src/lib.rs @@ -54,7 +54,6 @@ use common::{ }, outcome::Outcome, recipe::default_recipe_book, - spiral::Spiral2d, state::{State, TimeOfDay}, sync::WorldSyncExt, terrain::TerrainChunkSize, @@ -197,6 +196,7 @@ impl Server { state.ecs_mut().register::(); state.ecs_mut().register::(); state.ecs_mut().register::(); + state.ecs_mut().register::(); //Alias validator let banned_words_paths = &settings.banned_words_files; @@ -524,19 +524,17 @@ impl Server { &self.state.ecs().entities(), &self.state.ecs().read_storage::(), !&self.state.ecs().read_storage::(), + &self.state.ecs().read_storage::(), ) .join() - .filter(|(_, pos, _)| { + .filter(|(_, pos, _, home_chunk)| { let chunk_key = terrain.pos_key(pos.0.map(|e| e.floor() as i32)); - // Check not only this chunk, but also all neighbours to avoid immediate - // despawning if the entity walks outside of a valid chunk - // briefly. If the entity isn't even near a loaded chunk then we get - // rid of it. - Spiral2d::new() - .take(9) - .all(|offs| terrain.get_key(chunk_key + offs).is_none()) + // Check if both this chunk and the NPCs `home_chunk` is unloaded. If so, + // we delete them. We check for `home_chunk` in order to avoid duplicating + // the entity under some circumstances. + terrain.get_key(chunk_key).is_none() && terrain.get_key(home_chunk.0).is_none() }) - .map(|(entity, _, _)| entity) + .map(|(entity, _, _, _)| entity) .collect::>() }; diff --git a/server/src/sys/terrain.rs b/server/src/sys/terrain.rs index 634bca2b81..4cb75e0244 100644 --- a/server/src/sys/terrain.rs +++ b/server/src/sys/terrain.rs @@ -187,6 +187,7 @@ impl<'a> System<'a> for Sys { body, alignment, scale: comp::Scale(scale), + home_chunk: Some(comp::HomeChunk(key)), drop_item: entity.loot_drop, }) }