2019-10-20 07:20:21 +00:00
|
|
|
use super::SysTimer;
|
2019-10-20 05:19:50 +00:00
|
|
|
use crate::{chunk_generator::ChunkGenerator, client::Client, Tick};
|
|
|
|
use common::{
|
2019-10-22 18:18:40 +00:00
|
|
|
assets,
|
2020-05-15 17:34:05 +00:00
|
|
|
comp::{self, item, Alignment, CharacterAbility, ItemConfig, Player, Pos},
|
2019-10-20 05:19:50 +00:00
|
|
|
event::{EventBus, ServerEvent},
|
2020-04-23 16:00:48 +00:00
|
|
|
generation::get_npc_name,
|
2019-10-20 05:19:50 +00:00
|
|
|
msg::ServerMsg,
|
2020-04-23 14:01:37 +00:00
|
|
|
npc::NPC_NAMES,
|
2019-10-20 05:19:50 +00:00
|
|
|
state::TerrainChanges,
|
|
|
|
terrain::TerrainGrid,
|
2020-07-07 01:21:14 +00:00
|
|
|
LoadoutBuilder,
|
2019-10-20 05:19:50 +00:00
|
|
|
};
|
2020-04-20 00:17:54 +00:00
|
|
|
use rand::Rng;
|
2019-10-20 05:19:50 +00:00
|
|
|
use specs::{Join, Read, ReadStorage, System, Write, WriteExpect, WriteStorage};
|
2020-03-16 13:27:52 +00:00
|
|
|
use std::{sync::Arc, time::Duration};
|
2019-10-20 05:19:50 +00:00
|
|
|
use vek::*;
|
|
|
|
|
2020-01-27 16:48:42 +00:00
|
|
|
/// This system will handle loading generated chunks and unloading
|
|
|
|
/// uneeded chunks.
|
2019-10-20 05:19:50 +00:00
|
|
|
/// 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 {
|
2020-06-10 19:47:36 +00:00
|
|
|
#[allow(clippy::type_complexity)] // TODO: Pending review in #587
|
2019-10-20 05:19:50 +00:00
|
|
|
type SystemData = (
|
|
|
|
Read<'a, EventBus<ServerEvent>>,
|
|
|
|
Read<'a, Tick>,
|
2019-10-20 07:20:21 +00:00
|
|
|
Write<'a, SysTimer<Self>>,
|
2019-10-20 05:19:50 +00:00
|
|
|
WriteExpect<'a, ChunkGenerator>,
|
|
|
|
WriteExpect<'a, TerrainGrid>,
|
|
|
|
Write<'a, TerrainChanges>,
|
|
|
|
ReadStorage<'a, Pos>,
|
|
|
|
ReadStorage<'a, Player>,
|
|
|
|
WriteStorage<'a, Client>,
|
|
|
|
);
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&mut self,
|
|
|
|
(
|
2020-03-22 04:49:32 +00:00
|
|
|
server_event_bus,
|
2019-10-20 05:19:50 +00:00
|
|
|
tick,
|
2019-10-20 07:20:21 +00:00
|
|
|
mut timer,
|
2019-10-20 05:19:50 +00:00
|
|
|
mut chunk_generator,
|
|
|
|
mut terrain,
|
|
|
|
mut terrain_changes,
|
|
|
|
positions,
|
|
|
|
players,
|
|
|
|
mut clients,
|
|
|
|
): Self::SystemData,
|
|
|
|
) {
|
2019-10-20 07:20:21 +00:00
|
|
|
timer.start();
|
|
|
|
|
2020-03-22 04:49:32 +00:00
|
|
|
let mut server_emitter = server_event_bus.emitter();
|
|
|
|
|
2019-10-20 05:19:50 +00:00
|
|
|
// 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;
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-10-20 05:19:50 +00:00
|
|
|
};
|
|
|
|
// 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));
|
2019-12-23 06:02:00 +00:00
|
|
|
// Subtract 2 from the offset before computing squared magnitude
|
|
|
|
// 1 since chunks need neighbors to be meshed
|
|
|
|
// 1 to act as a buffer if the player moves in that direction
|
2020-06-30 14:56:49 +00:00
|
|
|
let adjusted_dist_sqr = (chunk_pos - key)
|
|
|
|
.map(|e: i32| (e.abs() as u32).saturating_sub(2))
|
2019-10-20 05:19:50 +00:00
|
|
|
.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
|
2020-01-25 02:15:15 +00:00
|
|
|
for entity in supplement.entities {
|
2020-04-17 20:58:36 +00:00
|
|
|
if entity.is_waypoint {
|
2020-01-25 02:15:15 +00:00
|
|
|
server_emitter.emit(ServerEvent::CreateWaypoint(entity.pos));
|
2020-04-17 20:58:36 +00:00
|
|
|
continue;
|
|
|
|
}
|
2020-03-16 13:27:52 +00:00
|
|
|
|
2020-04-18 20:26:43 +00:00
|
|
|
let mut body = entity.body;
|
2020-06-30 14:56:49 +00:00
|
|
|
let name = entity.name.unwrap_or_else(|| "Unnamed".to_string());
|
2020-04-17 20:58:36 +00:00
|
|
|
let alignment = entity.alignment;
|
2020-04-18 20:26:43 +00:00
|
|
|
let main_tool = entity.main_tool;
|
|
|
|
let mut stats = comp::Stats::new(name, body);
|
2020-05-04 16:59:32 +00:00
|
|
|
// let damage = stats.level.level() as i32; TODO: Make NPC base damage
|
|
|
|
// non-linearly depend on their level
|
2019-10-20 05:19:50 +00:00
|
|
|
|
2020-04-17 20:58:36 +00:00
|
|
|
let active_item =
|
2020-04-18 20:26:43 +00:00
|
|
|
if let Some(item::ItemKind::Tool(tool)) = main_tool.as_ref().map(|i| &i.kind) {
|
2020-04-17 20:58:36 +00:00
|
|
|
let mut abilities = tool.get_abilities();
|
|
|
|
let mut ability_drain = abilities.drain(..);
|
2019-10-20 05:19:50 +00:00
|
|
|
|
2020-04-18 20:26:43 +00:00
|
|
|
main_tool.map(|item| comp::ItemConfig {
|
2020-04-17 20:58:36 +00:00
|
|
|
item,
|
|
|
|
ability1: ability_drain.next(),
|
|
|
|
ability2: ability_drain.next(),
|
|
|
|
ability3: ability_drain.next(),
|
|
|
|
block_ability: None,
|
2020-08-06 08:04:03 +00:00
|
|
|
dodge_ability: Some(comp::CharacterAbility::Roll),
|
2020-04-17 20:58:36 +00:00
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Some(ItemConfig {
|
|
|
|
// We need the empty item so npcs can attack
|
2020-08-01 20:08:30 +00:00
|
|
|
item: assets::load_expect_cloned("common.items.weapons.empty.empty"),
|
2020-08-06 08:04:03 +00:00
|
|
|
ability1: Some(CharacterAbility::BasicMelee {
|
|
|
|
energy_cost: 0,
|
|
|
|
buildup_duration: Duration::from_millis(0),
|
|
|
|
recover_duration: Duration::from_millis(400),
|
|
|
|
base_healthchange: -60,
|
|
|
|
range: 5.0,
|
|
|
|
max_angle: 80.0,
|
|
|
|
}),
|
2020-04-17 20:58:36 +00:00
|
|
|
ability2: None,
|
|
|
|
ability3: None,
|
|
|
|
block_ability: None,
|
|
|
|
dodge_ability: None,
|
|
|
|
})
|
|
|
|
};
|
2020-03-14 21:33:20 +00:00
|
|
|
|
2020-04-17 20:58:36 +00:00
|
|
|
let mut loadout = match alignment {
|
|
|
|
comp::Alignment::Npc => comp::Loadout {
|
|
|
|
active_item,
|
|
|
|
second_item: None,
|
2020-04-30 20:43:24 +00:00
|
|
|
shoulder: None,
|
2020-04-17 20:58:36 +00:00
|
|
|
chest: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
match rand::thread_rng().gen_range(0, 10) {
|
|
|
|
0 => "common.items.armor.chest.worker_green_0",
|
|
|
|
1 => "common.items.armor.chest.worker_green_1",
|
|
|
|
2 => "common.items.armor.chest.worker_red_0",
|
|
|
|
3 => "common.items.armor.chest.worker_red_1",
|
|
|
|
4 => "common.items.armor.chest.worker_purple_0",
|
|
|
|
5 => "common.items.armor.chest.worker_purple_1",
|
|
|
|
6 => "common.items.armor.chest.worker_yellow_0",
|
|
|
|
7 => "common.items.armor.chest.worker_yellow_1",
|
|
|
|
8 => "common.items.armor.chest.worker_orange_0",
|
|
|
|
_ => "common.items.armor.chest.worker_orange_1",
|
|
|
|
},
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
belt: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.belt.leather_0",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
2020-04-30 20:43:24 +00:00
|
|
|
hand: None,
|
2020-04-17 20:58:36 +00:00
|
|
|
pants: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.pants.worker_blue_0",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
foot: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
match rand::thread_rng().gen_range(0, 2) {
|
|
|
|
0 => "common.items.armor.foot.leather_0",
|
|
|
|
_ => "common.items.armor.starter.sandals_0",
|
|
|
|
},
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
back: None,
|
|
|
|
ring: None,
|
|
|
|
neck: None,
|
|
|
|
lantern: None,
|
|
|
|
head: None,
|
|
|
|
tabard: None,
|
|
|
|
},
|
|
|
|
comp::Alignment::Enemy => comp::Loadout {
|
|
|
|
active_item,
|
|
|
|
second_item: None,
|
|
|
|
shoulder: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.shoulder.cultist_shoulder_purple",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
chest: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.chest.cultist_chest_purple",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
belt: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.belt.cultist_belt",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
hand: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.hand.cultist_hands_purple",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
pants: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.pants.cultist_legs_purple",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
|
|
|
foot: Some(assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.armor.foot.cultist_boots",
|
2020-04-17 20:58:36 +00:00
|
|
|
)),
|
2020-07-23 12:10:13 +00:00
|
|
|
back: Some(assets::load_expect_cloned(
|
2020-07-23 13:01:39 +00:00
|
|
|
"common.items.armor.back.dungeon_purple-0",
|
2020-07-23 12:10:13 +00:00
|
|
|
)),
|
2020-04-17 20:58:36 +00:00
|
|
|
ring: None,
|
|
|
|
neck: None,
|
2020-04-30 20:43:24 +00:00
|
|
|
lantern: Some(assets::load_expect_cloned("common.items.lantern.black_0")),
|
2020-04-17 20:58:36 +00:00
|
|
|
head: None,
|
|
|
|
tabard: None,
|
|
|
|
},
|
2020-07-09 00:04:25 +00:00
|
|
|
_ => LoadoutBuilder::animal(entity.body).build(),
|
2020-04-17 20:58:36 +00:00
|
|
|
};
|
2019-10-20 05:19:50 +00:00
|
|
|
|
2020-05-15 15:05:50 +00:00
|
|
|
let mut scale = entity.scale;
|
2020-03-15 13:34:17 +00:00
|
|
|
|
2020-04-17 20:58:36 +00:00
|
|
|
// TODO: Remove this and implement scaling or level depending on stuff like
|
|
|
|
// species instead
|
2020-05-15 15:05:50 +00:00
|
|
|
stats.level.set_level(
|
|
|
|
entity.level.unwrap_or_else(|| {
|
|
|
|
(rand::thread_rng().gen_range(1, 9) as f32 * scale) as u32
|
|
|
|
}),
|
|
|
|
);
|
2020-03-15 13:34:17 +00:00
|
|
|
|
2020-04-17 20:58:36 +00:00
|
|
|
// Replace stuff if it's a boss
|
|
|
|
if entity.is_giant {
|
|
|
|
if rand::random::<f32>() < 0.65 {
|
|
|
|
let body_new = comp::humanoid::Body::random();
|
|
|
|
body = comp::Body::Humanoid(body_new);
|
2020-05-15 17:33:34 +00:00
|
|
|
let adjective = if let Alignment::Enemy = entity.alignment {
|
|
|
|
"Angry"
|
|
|
|
} else {
|
|
|
|
"Gentle"
|
|
|
|
};
|
2020-04-17 20:58:36 +00:00
|
|
|
stats = comp::Stats::new(
|
2020-05-15 17:34:05 +00:00
|
|
|
format!(
|
|
|
|
"{} Giant {}",
|
|
|
|
adjective,
|
2020-05-29 18:23:00 +00:00
|
|
|
get_npc_name(&NPC_NAMES.humanoid, body_new.species)
|
2020-05-15 17:34:05 +00:00
|
|
|
),
|
2020-04-17 20:58:36 +00:00
|
|
|
body,
|
|
|
|
);
|
2020-01-25 02:15:15 +00:00
|
|
|
}
|
2020-04-17 20:58:36 +00:00
|
|
|
loadout = comp::Loadout {
|
|
|
|
active_item: Some(comp::ItemConfig {
|
|
|
|
item: assets::load_expect_cloned(
|
2020-04-30 20:43:24 +00:00
|
|
|
"common.items.weapons.sword.zweihander_sword_0",
|
2020-04-17 20:58:36 +00:00
|
|
|
),
|
2020-08-06 08:04:03 +00:00
|
|
|
ability1: Some(CharacterAbility::BasicMelee {
|
|
|
|
energy_cost: 0,
|
|
|
|
buildup_duration: Duration::from_millis(800),
|
|
|
|
recover_duration: Duration::from_millis(200),
|
|
|
|
base_healthchange: -100,
|
|
|
|
range: 3.5,
|
|
|
|
max_angle: 60.0,
|
|
|
|
}),
|
2020-04-17 20:58:36 +00:00
|
|
|
ability2: None,
|
|
|
|
ability3: None,
|
|
|
|
block_ability: None,
|
|
|
|
dodge_ability: None,
|
|
|
|
}),
|
|
|
|
second_item: None,
|
|
|
|
shoulder: Some(assets::load_expect_cloned(
|
|
|
|
"common.items.armor.shoulder.plate_0",
|
|
|
|
)),
|
|
|
|
chest: Some(assets::load_expect_cloned(
|
|
|
|
"common.items.armor.chest.plate_green_0",
|
|
|
|
)),
|
|
|
|
belt: Some(assets::load_expect_cloned(
|
|
|
|
"common.items.armor.belt.plate_0",
|
|
|
|
)),
|
|
|
|
hand: Some(assets::load_expect_cloned(
|
|
|
|
"common.items.armor.hand.plate_0",
|
|
|
|
)),
|
|
|
|
pants: Some(assets::load_expect_cloned(
|
|
|
|
"common.items.armor.pants.plate_green_0",
|
|
|
|
)),
|
|
|
|
foot: Some(assets::load_expect_cloned(
|
|
|
|
"common.items.armor.foot.plate_0",
|
|
|
|
)),
|
|
|
|
back: None,
|
|
|
|
ring: None,
|
|
|
|
neck: None,
|
|
|
|
lantern: None,
|
|
|
|
head: None,
|
|
|
|
tabard: None,
|
|
|
|
};
|
|
|
|
|
|
|
|
stats.level.set_level(rand::thread_rng().gen_range(30, 35));
|
|
|
|
scale = 2.0 + rand::random::<f32>();
|
|
|
|
}
|
|
|
|
|
2020-07-09 00:04:25 +00:00
|
|
|
stats.update_max_hp(stats.body_type);
|
2020-04-17 20:58:36 +00:00
|
|
|
|
|
|
|
stats
|
|
|
|
.health
|
|
|
|
.set_to(stats.health.maximum(), comp::HealthSource::Revive);
|
|
|
|
|
2020-05-26 02:45:13 +00:00
|
|
|
let can_speak = alignment == comp::Alignment::Npc;
|
|
|
|
|
2020-04-17 20:58:36 +00:00
|
|
|
// TODO: This code sets an appropriate base_damage for the enemy. This doesn't
|
|
|
|
// work because the damage is now saved in an ability
|
|
|
|
/*
|
|
|
|
if let Some(item::ItemKind::Tool(item::ToolData { base_damage, .. })) =
|
|
|
|
&mut loadout.active_item.map(|i| i.item.kind)
|
|
|
|
{
|
|
|
|
*base_damage = stats.level.level() as u32 * 3;
|
2020-01-20 16:42:35 +00:00
|
|
|
}
|
2020-04-17 20:58:36 +00:00
|
|
|
*/
|
|
|
|
server_emitter.emit(ServerEvent::CreateNpc {
|
|
|
|
pos: Pos(entity.pos),
|
|
|
|
stats,
|
|
|
|
loadout,
|
|
|
|
body,
|
2020-07-05 12:39:28 +00:00
|
|
|
agent: if entity.has_agency {
|
|
|
|
Some(comp::Agent::new(entity.pos, can_speak))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
},
|
2020-04-17 20:58:36 +00:00
|
|
|
alignment,
|
|
|
|
scale: comp::Scale(scale),
|
2020-05-15 15:05:50 +00:00
|
|
|
drop_item: entity.loot_drop,
|
2020-04-17 20:58:36 +00:00
|
|
|
})
|
2019-10-20 05:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove chunks that are too far from players.
|
|
|
|
let mut chunks_to_remove = Vec::new();
|
|
|
|
terrain
|
|
|
|
.iter()
|
|
|
|
.map(|(k, _)| k)
|
2019-10-26 02:00:30 +00:00
|
|
|
// Don't check every chunk every tick (spread over 16 ticks)
|
|
|
|
.filter(|k| k.x.abs() as u64 % 4 + (k.y.abs() as u64 % 4) * 4 == tick.0 % 16)
|
2019-10-20 05:19:50 +00:00
|
|
|
// 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);
|
|
|
|
}
|
2019-10-20 07:20:21 +00:00
|
|
|
|
|
|
|
timer.end()
|
2019-10-20 05:19:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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));
|
|
|
|
|
2020-06-30 14:56:49 +00:00
|
|
|
let adjusted_dist_sqr = (player_chunk_pos - chunk_pos)
|
|
|
|
.map(|e: i32| (e.abs() as u32).saturating_sub(2))
|
2019-10-20 05:19:50 +00:00
|
|
|
.magnitude_squared();
|
|
|
|
|
|
|
|
adjusted_dist_sqr <= vd.pow(2)
|
|
|
|
}
|