mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
200 lines
7.5 KiB
Rust
200 lines
7.5 KiB
Rust
|
use crate::{chunk_generator::ChunkGenerator, client::Client, Tick};
|
||
|
use common::{
|
||
|
comp::{self, Player, Pos},
|
||
|
event::{EventBus, ServerEvent},
|
||
|
msg::ServerMsg,
|
||
|
state::TerrainChanges,
|
||
|
terrain::TerrainGrid,
|
||
|
};
|
||
|
use rand::Rng;
|
||
|
use specs::{Join, Read, ReadStorage, System, Write, WriteExpect, WriteStorage};
|
||
|
use std::sync::Arc;
|
||
|
use vek::*;
|
||
|
|
||
|
/// This system will handle loading generated chunks and unloading uneeded chunks.
|
||
|
/// 1. Inserts newly generated chunks into the TerrainGrid
|
||
|
/// 2. Sends new chunks to neaby clients
|
||
|
/// 3. Handles the chunk's supplement (e.g. npcs)
|
||
|
/// 4. Removes chunks outside the range of players
|
||
|
pub struct Sys;
|
||
|
impl<'a> System<'a> for Sys {
|
||
|
type SystemData = (
|
||
|
Read<'a, EventBus<ServerEvent>>,
|
||
|
Read<'a, Tick>,
|
||
|
WriteExpect<'a, ChunkGenerator>,
|
||
|
WriteExpect<'a, TerrainGrid>,
|
||
|
Write<'a, TerrainChanges>,
|
||
|
ReadStorage<'a, Pos>,
|
||
|
ReadStorage<'a, Player>,
|
||
|
WriteStorage<'a, Client>,
|
||
|
);
|
||
|
|
||
|
fn run(
|
||
|
&mut self,
|
||
|
(
|
||
|
server_emitter,
|
||
|
tick,
|
||
|
mut chunk_generator,
|
||
|
mut terrain,
|
||
|
mut terrain_changes,
|
||
|
positions,
|
||
|
players,
|
||
|
mut clients,
|
||
|
): Self::SystemData,
|
||
|
) {
|
||
|
// Fetch any generated `TerrainChunk`s and insert them into the terrain.
|
||
|
// Also, send the chunk data to anybody that is close by.
|
||
|
'insert_terrain_chunks: while let Some((key, res)) = chunk_generator.recv_new_chunk() {
|
||
|
let (chunk, supplement) = match res {
|
||
|
Ok((chunk, supplement)) => (chunk, supplement),
|
||
|
Err(entity) => {
|
||
|
if let Some(client) = clients.get_mut(entity) {
|
||
|
client.notify(ServerMsg::TerrainChunkUpdate {
|
||
|
key,
|
||
|
chunk: Err(()),
|
||
|
});
|
||
|
}
|
||
|
continue 'insert_terrain_chunks;
|
||
|
}
|
||
|
};
|
||
|
// Send the chunk to all nearby players.
|
||
|
for (view_distance, pos, client) in (&players, &positions, &mut clients)
|
||
|
.join()
|
||
|
.filter_map(|(player, pos, client)| {
|
||
|
player.view_distance.map(|vd| (vd, pos, client))
|
||
|
})
|
||
|
{
|
||
|
let chunk_pos = terrain.pos_key(pos.0.map(|e| e as i32));
|
||
|
let adjusted_dist_sqr = (Vec2::from(chunk_pos) - Vec2::from(key))
|
||
|
.map(|e: i32| (e.abs() as u32).checked_sub(2).unwrap_or(0))
|
||
|
.magnitude_squared();
|
||
|
|
||
|
if adjusted_dist_sqr <= view_distance.pow(2) {
|
||
|
client.notify(ServerMsg::TerrainChunkUpdate {
|
||
|
key,
|
||
|
chunk: Ok(Box::new(chunk.clone())),
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: code duplication for chunk insertion between here and state.rs
|
||
|
// Insert the chunk into terrain changes
|
||
|
if terrain.insert(key, Arc::new(chunk)).is_some() {
|
||
|
terrain_changes.modified_chunks.insert(key);
|
||
|
} else {
|
||
|
terrain_changes.new_chunks.insert(key);
|
||
|
}
|
||
|
|
||
|
// Handle chunk supplement
|
||
|
for npc in supplement.npcs {
|
||
|
let (mut stats, mut body) = if rand::random() {
|
||
|
let stats = comp::Stats::new(
|
||
|
"Humanoid".to_string(),
|
||
|
Some(comp::Item::Tool {
|
||
|
kind: comp::item::Tool::Sword,
|
||
|
power: 5,
|
||
|
stamina: 0,
|
||
|
strength: 0,
|
||
|
dexterity: 0,
|
||
|
intelligence: 0,
|
||
|
}),
|
||
|
);
|
||
|
let body = comp::Body::Humanoid(comp::humanoid::Body::random());
|
||
|
(stats, body)
|
||
|
} else {
|
||
|
let stats = comp::Stats::new("Wolf".to_string(), None);
|
||
|
let body = comp::Body::QuadrupedMedium(comp::quadruped_medium::Body::random());
|
||
|
(stats, body)
|
||
|
};
|
||
|
let mut scale = 1.0;
|
||
|
|
||
|
// TODO: Remove this and implement scaling or level depending on stuff like species instead
|
||
|
stats.level.set_level(rand::thread_rng().gen_range(1, 3));
|
||
|
|
||
|
if npc.boss {
|
||
|
if rand::random::<f32>() < 0.8 {
|
||
|
stats = comp::Stats::new(
|
||
|
"Humanoid".to_string(),
|
||
|
Some(comp::Item::Tool {
|
||
|
kind: comp::item::Tool::Sword,
|
||
|
power: 10,
|
||
|
stamina: 0,
|
||
|
strength: 0,
|
||
|
dexterity: 0,
|
||
|
intelligence: 0,
|
||
|
}),
|
||
|
);
|
||
|
body = comp::Body::Humanoid(comp::humanoid::Body::random());
|
||
|
}
|
||
|
stats.level.set_level(rand::thread_rng().gen_range(10, 50));
|
||
|
scale = 2.5 + rand::random::<f32>();
|
||
|
}
|
||
|
|
||
|
stats.update_max_hp();
|
||
|
stats
|
||
|
.health
|
||
|
.set_to(stats.health.maximum(), comp::HealthSource::Revive);
|
||
|
server_emitter.emit(ServerEvent::CreateNpc {
|
||
|
pos: Pos(npc.pos),
|
||
|
stats,
|
||
|
body,
|
||
|
agent: comp::Agent::enemy(),
|
||
|
scale: comp::Scale(scale),
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Remove chunks that are too far from players.
|
||
|
let mut chunks_to_remove = Vec::new();
|
||
|
terrain
|
||
|
.iter()
|
||
|
.map(|(k, _)| k)
|
||
|
// Don't every chunk every tick (spread over 16 ticks)
|
||
|
.filter(|k| k.x.abs() as u64 % 4 + k.y.abs() as u64 % 8 * 4 == tick.0 % 16)
|
||
|
// There shouldn't be to many pending chunks so we will just check them all
|
||
|
.chain(chunk_generator.pending_chunks())
|
||
|
.for_each(|chunk_key| {
|
||
|
let mut should_drop = true;
|
||
|
|
||
|
// For each player with a position, calculate the distance.
|
||
|
for (player, pos) in (&players, &positions).join() {
|
||
|
if player
|
||
|
.view_distance
|
||
|
.map(|vd| chunk_in_vd(pos.0, chunk_key, &terrain, vd))
|
||
|
.unwrap_or(false)
|
||
|
{
|
||
|
should_drop = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if should_drop {
|
||
|
chunks_to_remove.push(chunk_key);
|
||
|
}
|
||
|
});
|
||
|
for key in chunks_to_remove {
|
||
|
// TODO: code duplication for chunk insertion between here and state.rs
|
||
|
if terrain.remove(key).is_some() {
|
||
|
terrain_changes.removed_chunks.insert(key);
|
||
|
}
|
||
|
|
||
|
chunk_generator.cancel_if_pending(key);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub fn chunk_in_vd(
|
||
|
player_pos: Vec3<f32>,
|
||
|
chunk_pos: Vec2<i32>,
|
||
|
terrain: &TerrainGrid,
|
||
|
vd: u32,
|
||
|
) -> bool {
|
||
|
let player_chunk_pos = terrain.pos_key(player_pos.map(|e| e as i32));
|
||
|
|
||
|
let adjusted_dist_sqr = Vec2::from(player_chunk_pos - chunk_pos)
|
||
|
.map(|e: i32| (e.abs() as u32).checked_sub(2).unwrap_or(0))
|
||
|
.magnitude_squared();
|
||
|
|
||
|
adjusted_dist_sqr <= vd.pow(2)
|
||
|
}
|