2019-11-29 06:04:37 +00:00
|
|
|
use super::{
|
|
|
|
sentinel::{DeletedEntities, TrackedComps},
|
|
|
|
SysTimer,
|
|
|
|
};
|
2019-10-15 04:06:14 +00:00
|
|
|
use crate::client::{self, Client, RegionSubscription};
|
|
|
|
use common::{
|
2020-03-20 14:45:36 +00:00
|
|
|
comp::{Ori, Player, Pos, Vel},
|
2019-10-15 04:06:14 +00:00
|
|
|
msg::ServerMsg,
|
|
|
|
region::{region_in_vd, regions_in_vd, Event as RegionEvent, RegionMap},
|
2019-11-24 20:12:03 +00:00
|
|
|
sync::Uid,
|
2019-10-15 04:06:14 +00:00
|
|
|
terrain::TerrainChunkSize,
|
|
|
|
vol::RectVolSize,
|
|
|
|
};
|
2019-11-29 06:04:37 +00:00
|
|
|
use specs::{
|
2019-11-30 06:41:20 +00:00
|
|
|
Entities, Join, ReadExpect, ReadStorage, System, SystemData, World, WorldExt, Write,
|
|
|
|
WriteStorage,
|
2019-11-29 06:04:37 +00:00
|
|
|
};
|
2020-06-21 14:26:06 +00:00
|
|
|
use tracing::{debug, error};
|
2019-10-15 04:06:14 +00:00
|
|
|
use vek::*;
|
|
|
|
|
|
|
|
/// This system will update region subscriptions based on client positions
|
|
|
|
pub struct Sys;
|
|
|
|
impl<'a> System<'a> for Sys {
|
2020-06-10 19:47:36 +00:00
|
|
|
#[allow(clippy::type_complexity)] // TODO: Pending review in #587
|
2019-10-15 04:06:14 +00:00
|
|
|
type SystemData = (
|
|
|
|
Entities<'a>,
|
|
|
|
ReadExpect<'a, RegionMap>,
|
2019-10-20 07:20:21 +00:00
|
|
|
Write<'a, SysTimer<Self>>,
|
2019-10-15 04:06:14 +00:00
|
|
|
ReadStorage<'a, Uid>,
|
|
|
|
ReadStorage<'a, Pos>,
|
2019-10-25 05:35:15 +00:00
|
|
|
ReadStorage<'a, Vel>,
|
|
|
|
ReadStorage<'a, Ori>,
|
2019-10-15 04:06:14 +00:00
|
|
|
ReadStorage<'a, Player>,
|
|
|
|
WriteStorage<'a, Client>,
|
|
|
|
WriteStorage<'a, RegionSubscription>,
|
2019-11-29 06:04:37 +00:00
|
|
|
Write<'a, DeletedEntities>,
|
2019-11-04 00:57:36 +00:00
|
|
|
TrackedComps<'a>,
|
2019-10-15 04:06:14 +00:00
|
|
|
);
|
|
|
|
|
2020-06-10 19:47:36 +00:00
|
|
|
#[allow(clippy::block_in_if_condition_stmt)] // TODO: Pending review in #587
|
2019-10-15 04:06:14 +00:00
|
|
|
fn run(
|
|
|
|
&mut self,
|
2019-10-25 05:35:15 +00:00
|
|
|
(
|
|
|
|
entities,
|
|
|
|
region_map,
|
|
|
|
mut timer,
|
|
|
|
uids,
|
|
|
|
positions,
|
|
|
|
velocities,
|
|
|
|
orientations,
|
|
|
|
players,
|
|
|
|
mut clients,
|
|
|
|
mut subscriptions,
|
2019-11-29 06:04:37 +00:00
|
|
|
mut deleted_entities,
|
2019-11-04 00:57:36 +00:00
|
|
|
tracked_comps,
|
2019-10-25 05:35:15 +00:00
|
|
|
): Self::SystemData,
|
2019-10-15 04:06:14 +00:00
|
|
|
) {
|
2019-10-20 07:20:21 +00:00
|
|
|
timer.start();
|
|
|
|
|
2019-10-15 04:06:14 +00:00
|
|
|
// To update subscriptions
|
|
|
|
// 1. Iterate through clients
|
|
|
|
// 2. Calculate current chunk position
|
|
|
|
// 3. If chunk is the same return, otherwise continue (use fuzzyiness)
|
|
|
|
// 4. Iterate through subscribed regions
|
|
|
|
// 5. Check if region is still in range (use fuzzyiness)
|
|
|
|
// 6. If not in range
|
|
|
|
// - remove from hashset
|
|
|
|
// - inform client of which entities to remove
|
|
|
|
// 7. Determine list of regions that are in range and iterate through it
|
|
|
|
// - check if in hashset (hash calc) if not add it
|
|
|
|
let mut regions_to_remove = Vec::new();
|
2019-10-26 02:00:30 +00:00
|
|
|
for (client, subscription, pos, vd, client_entity) in (
|
|
|
|
&mut clients,
|
|
|
|
&mut subscriptions,
|
|
|
|
&positions,
|
|
|
|
&players,
|
|
|
|
&entities,
|
|
|
|
)
|
|
|
|
.join()
|
2019-11-29 06:04:37 +00:00
|
|
|
.filter_map(|(client, s, pos, player, e)| {
|
|
|
|
if client.is_ingame() {
|
|
|
|
player.view_distance.map(|v| (client, s, pos, v, e))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
2019-10-15 04:06:14 +00:00
|
|
|
{
|
2019-11-29 06:04:37 +00:00
|
|
|
// Calculate current chunk
|
2019-10-15 04:06:14 +00:00
|
|
|
let chunk = (Vec2::<f32>::from(pos.0))
|
|
|
|
.map2(TerrainChunkSize::RECT_SIZE, |e, sz| e as i32 / sz as i32);
|
|
|
|
// Only update regions when moving to a new chunk
|
2020-02-01 20:39:39 +00:00
|
|
|
// uses a fuzzy border to prevent rapid triggering when moving along chunk
|
|
|
|
// boundaries
|
2019-10-15 04:06:14 +00:00
|
|
|
if chunk != subscription.fuzzy_chunk
|
|
|
|
&& (subscription
|
|
|
|
.fuzzy_chunk
|
|
|
|
.map2(TerrainChunkSize::RECT_SIZE, |e, sz| {
|
|
|
|
(e as f32 + 0.5) * sz as f32
|
|
|
|
})
|
|
|
|
- Vec2::from(pos.0))
|
|
|
|
.map2(TerrainChunkSize::RECT_SIZE, |e, sz| {
|
|
|
|
e.abs() > (sz / 2 + client::CHUNK_FUZZ) as f32
|
|
|
|
})
|
|
|
|
.reduce_or()
|
|
|
|
{
|
|
|
|
// Update current chunk
|
2019-11-30 05:33:17 +00:00
|
|
|
subscription.fuzzy_chunk = Vec2::<f32>::from(pos.0)
|
2019-10-15 04:06:14 +00:00
|
|
|
.map2(TerrainChunkSize::RECT_SIZE, |e, sz| e as i32 / sz as i32);
|
|
|
|
// Use the largest side length as our chunk size
|
|
|
|
let chunk_size = TerrainChunkSize::RECT_SIZE.reduce_max() as f32;
|
|
|
|
// Iterate through currently subscribed regions
|
|
|
|
for key in &subscription.regions {
|
|
|
|
// Check if the region is not within range anymore
|
|
|
|
if !region_in_vd(
|
|
|
|
*key,
|
|
|
|
pos.0,
|
|
|
|
(vd as f32 * chunk_size)
|
|
|
|
+ (client::CHUNK_FUZZ as f32 + client::REGION_FUZZ as f32 + chunk_size)
|
|
|
|
* 2.0f32.sqrt(),
|
|
|
|
) {
|
|
|
|
// Add to the list of regions to remove
|
|
|
|
regions_to_remove.push(*key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate through regions to remove
|
|
|
|
for key in regions_to_remove.drain(..) {
|
2019-11-29 06:04:37 +00:00
|
|
|
// Remove region from this client's set of subscribed regions
|
2019-10-15 04:06:14 +00:00
|
|
|
subscription.regions.remove(&key);
|
2020-02-01 20:39:39 +00:00
|
|
|
// Tell the client to delete the entities in that region if it exists in the
|
|
|
|
// RegionMap
|
2019-10-15 04:06:14 +00:00
|
|
|
if let Some(region) = region_map.get(key) {
|
2020-02-01 20:39:39 +00:00
|
|
|
// Process entity left events since they won't be processed during entity
|
|
|
|
// sync because this region is no longer subscribed to
|
2019-11-29 06:04:37 +00:00
|
|
|
// TODO: consider changing system ordering??
|
2019-10-15 04:06:14 +00:00
|
|
|
for event in region.events() {
|
|
|
|
match event {
|
2020-02-01 20:39:39 +00:00
|
|
|
RegionEvent::Entered(_, _) => {}, /* These don't need to be */
|
|
|
|
// processed because this
|
|
|
|
// region is being thrown out
|
|
|
|
// anyway
|
2019-10-15 04:06:14 +00:00
|
|
|
RegionEvent::Left(id, maybe_key) => {
|
|
|
|
// Lookup UID for entity
|
2019-11-29 06:04:37 +00:00
|
|
|
// Doesn't overlap with entity deletion in sync packages
|
|
|
|
// because the uid would not be available if the entity was
|
|
|
|
// deleted
|
2019-10-15 04:06:14 +00:00
|
|
|
if let Some(&uid) = uids.get(entities.entity(*id)) {
|
|
|
|
if !maybe_key
|
|
|
|
.as_ref()
|
2019-11-29 06:04:37 +00:00
|
|
|
// Don't need to check that this isn't also in the regions to remove since the entity will be removed when we get to that one
|
2019-10-15 04:06:14 +00:00
|
|
|
.map(|key| subscription.regions.contains(key))
|
|
|
|
.unwrap_or(false)
|
|
|
|
{
|
|
|
|
client.notify(ServerMsg::DeleteEntity(uid.into()));
|
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-10-15 04:06:14 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-29 06:04:37 +00:00
|
|
|
// Tell client to delete entities in the region
|
2019-10-15 04:06:14 +00:00
|
|
|
for (&uid, _) in (&uids, region.entities()).join() {
|
2019-11-29 06:04:37 +00:00
|
|
|
client.notify(ServerMsg::DeleteEntity(uid.into()));
|
2019-10-15 04:06:14 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
// Send deleted entities since they won't be processed for this client in entity
|
|
|
|
// sync
|
2019-11-29 06:04:37 +00:00
|
|
|
for uid in deleted_entities
|
|
|
|
.get_deleted_in_region(key)
|
|
|
|
.iter()
|
|
|
|
.flat_map(|v| v.iter())
|
|
|
|
{
|
|
|
|
client.notify(ServerMsg::DeleteEntity(*uid));
|
|
|
|
}
|
2019-10-15 04:06:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for key in regions_in_vd(
|
|
|
|
pos.0,
|
|
|
|
(vd as f32 * chunk_size)
|
|
|
|
+ (client::CHUNK_FUZZ as f32 + chunk_size) * 2.0f32.sqrt(),
|
|
|
|
) {
|
2019-11-29 06:04:37 +00:00
|
|
|
// Send client intial info about the entities in this region if it was not
|
|
|
|
// already within the set of subscribed regions
|
2020-06-18 21:25:48 +00:00
|
|
|
if subscription.regions.insert(key) {
|
2019-10-25 05:35:15 +00:00
|
|
|
if let Some(region) = region_map.get(key) {
|
2020-03-20 14:45:36 +00:00
|
|
|
for (pos, vel, ori, _, entity) in (
|
2019-10-25 05:35:15 +00:00
|
|
|
&positions,
|
|
|
|
velocities.maybe(),
|
|
|
|
orientations.maybe(),
|
|
|
|
region.entities(),
|
2019-10-26 02:00:30 +00:00
|
|
|
&entities,
|
2019-10-25 05:35:15 +00:00
|
|
|
)
|
|
|
|
.join()
|
2020-03-20 14:45:36 +00:00
|
|
|
.filter(|(_, _, _, _, e)| *e != client_entity)
|
2019-10-25 05:35:15 +00:00
|
|
|
{
|
2020-03-18 21:00:07 +00:00
|
|
|
// Send message to create entity and tracked components and physics
|
|
|
|
// components
|
2019-11-04 00:57:36 +00:00
|
|
|
client.notify(ServerMsg::CreateEntity(
|
2020-03-18 21:00:07 +00:00
|
|
|
tracked_comps.create_entity_package(
|
|
|
|
entity,
|
|
|
|
Some(*pos),
|
|
|
|
vel.copied(),
|
|
|
|
ori.copied(),
|
|
|
|
),
|
2019-11-04 00:57:36 +00:00
|
|
|
));
|
2019-10-25 05:35:15 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-15 04:06:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-10-20 07:20:21 +00:00
|
|
|
|
|
|
|
timer.end();
|
2019-10-15 04:06:14 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-29 06:04:37 +00:00
|
|
|
|
|
|
|
/// Initialize region subscription
|
|
|
|
pub fn initialize_region_subscription(world: &World, entity: specs::Entity) {
|
|
|
|
if let (Some(client_pos), Some(client_vd), Some(client)) = (
|
|
|
|
world.read_storage::<Pos>().get(entity),
|
|
|
|
world
|
|
|
|
.read_storage::<Player>()
|
|
|
|
.get(entity)
|
|
|
|
.map(|pl| pl.view_distance)
|
|
|
|
.and_then(|v| v),
|
|
|
|
world.write_storage::<Client>().get_mut(entity),
|
|
|
|
) {
|
|
|
|
let fuzzy_chunk = (Vec2::<f32>::from(client_pos.0))
|
|
|
|
.map2(TerrainChunkSize::RECT_SIZE, |e, sz| e as i32 / sz as i32);
|
|
|
|
let chunk_size = TerrainChunkSize::RECT_SIZE.reduce_max() as f32;
|
|
|
|
let regions = common::region::regions_in_vd(
|
|
|
|
client_pos.0,
|
|
|
|
(client_vd as f32 * chunk_size) as f32
|
|
|
|
+ (client::CHUNK_FUZZ as f32 + chunk_size) * 2.0f32.sqrt(),
|
|
|
|
);
|
|
|
|
|
|
|
|
let region_map = world.read_resource::<RegionMap>();
|
2019-11-30 06:41:20 +00:00
|
|
|
let tracked_comps = TrackedComps::fetch(world);
|
2019-11-29 06:04:37 +00:00
|
|
|
for key in ®ions {
|
|
|
|
if let Some(region) = region_map.get(*key) {
|
2020-03-20 14:45:36 +00:00
|
|
|
for (pos, vel, ori, _, entity) in (
|
2019-11-29 06:04:37 +00:00
|
|
|
&world.read_storage::<Pos>(), // We assume all these entities have a position
|
|
|
|
world.read_storage::<Vel>().maybe(),
|
|
|
|
world.read_storage::<Ori>().maybe(),
|
|
|
|
region.entities(),
|
|
|
|
&world.entities(),
|
|
|
|
)
|
|
|
|
.join()
|
|
|
|
{
|
2020-03-18 21:00:07 +00:00
|
|
|
// Send message to create entity and tracked components and physics components
|
2019-11-29 06:04:37 +00:00
|
|
|
client.notify(ServerMsg::CreateEntity(
|
2020-03-18 21:00:07 +00:00
|
|
|
tracked_comps.create_entity_package(
|
|
|
|
entity,
|
|
|
|
Some(*pos),
|
|
|
|
vel.copied(),
|
|
|
|
ori.copied(),
|
|
|
|
),
|
2019-11-29 06:04:37 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-21 21:47:49 +00:00
|
|
|
if let Err(e) = world.write_storage().insert(entity, RegionSubscription {
|
2020-02-01 20:39:39 +00:00
|
|
|
fuzzy_chunk,
|
|
|
|
regions,
|
|
|
|
}) {
|
2020-06-21 21:47:49 +00:00
|
|
|
error!(?e, "Failed to insert region subscription component");
|
2019-11-29 06:04:37 +00:00
|
|
|
}
|
|
|
|
} else {
|
2020-02-01 20:39:39 +00:00
|
|
|
debug!(
|
2020-06-21 21:47:49 +00:00
|
|
|
?entity,
|
2020-02-01 20:39:39 +00:00
|
|
|
"Failed to initialize region subcription. Couldn't retrieve all the neccesary \
|
|
|
|
components on the provided entity"
|
|
|
|
);
|
2019-11-29 06:04:37 +00:00
|
|
|
}
|
|
|
|
}
|