2020-02-16 20:04:06 +00:00
|
|
|
use crate::{client::Client, Server, SpawnPoint, StateExt};
|
|
|
|
use common::{
|
2020-03-18 19:37:11 +00:00
|
|
|
assets,
|
|
|
|
comp::{self, object, Body, HealthChange, HealthSource, Item, Player, Stats},
|
2020-02-16 20:04:06 +00:00
|
|
|
msg::ServerMsg,
|
|
|
|
state::BlockChange,
|
|
|
|
sync::{Uid, WorldSyncExt},
|
|
|
|
terrain::{Block, TerrainGrid},
|
|
|
|
vol::{ReadVol, Vox},
|
|
|
|
};
|
|
|
|
use log::error;
|
2020-03-22 19:39:50 +00:00
|
|
|
use specs::{join::Join, Entity as EcsEntity, WorldExt};
|
2020-03-23 17:13:14 +00:00
|
|
|
use vek::Vec3;
|
2020-03-22 19:39:50 +00:00
|
|
|
|
|
|
|
const BLOCK_EFFICIENCY: f32 = 0.9;
|
|
|
|
const BLOCK_ANGLE: f32 = 180.0;
|
2020-02-16 20:04:06 +00:00
|
|
|
|
|
|
|
pub fn handle_damage(server: &Server, uid: Uid, change: HealthChange) {
|
|
|
|
let state = &server.state;
|
|
|
|
let ecs = state.ecs();
|
|
|
|
if let Some(entity) = ecs.entity_from_uid(uid.into()) {
|
|
|
|
if let Some(stats) = ecs.write_storage::<Stats>().get_mut(entity) {
|
|
|
|
stats.health.change_by(change);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn handle_destroy(server: &mut Server, entity: EcsEntity, cause: HealthSource) {
|
|
|
|
let state = server.state_mut();
|
|
|
|
|
|
|
|
// Chat message
|
|
|
|
if let Some(player) = state.ecs().read_storage::<Player>().get(entity) {
|
2020-03-20 13:26:18 +00:00
|
|
|
let msg = if let HealthSource::Attack { by }
|
|
|
|
| HealthSource::Projectile { owner: Some(by) } = cause
|
|
|
|
{
|
2020-02-16 20:04:06 +00:00
|
|
|
state.ecs().entity_from_uid(by.into()).and_then(|attacker| {
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.read_storage::<Player>()
|
|
|
|
.get(attacker)
|
|
|
|
.map(|attacker_alias| {
|
|
|
|
format!("{} was killed by {}", &player.alias, &attacker_alias.alias)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
.unwrap_or(format!("{} died", &player.alias));
|
|
|
|
|
|
|
|
state.notify_registered_clients(ServerMsg::kill(msg));
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Give EXP to the killer if entity had stats
|
|
|
|
let mut stats = state.ecs().write_storage::<Stats>();
|
|
|
|
if let Some(entity_stats) = stats.get(entity).cloned() {
|
2020-03-20 13:26:18 +00:00
|
|
|
if let HealthSource::Attack { by } | HealthSource::Projectile { owner: Some(by) } =
|
|
|
|
cause
|
|
|
|
{
|
2020-02-16 20:04:06 +00:00
|
|
|
state.ecs().entity_from_uid(by.into()).map(|attacker| {
|
|
|
|
if let Some(attacker_stats) = stats.get_mut(attacker) {
|
|
|
|
// TODO: Discuss whether we should give EXP by Player
|
|
|
|
// Killing or not.
|
|
|
|
attacker_stats
|
|
|
|
.exp
|
|
|
|
.change_by((entity_stats.level.level() * 10) as i64);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<Client>()
|
|
|
|
.get_mut(entity)
|
|
|
|
.is_some()
|
|
|
|
{
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage()
|
|
|
|
.insert(entity, comp::Vel(Vec3::zero()))
|
|
|
|
.err()
|
|
|
|
.map(|err| error!("Failed to set zero vel on dead client: {:?}", err));
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage()
|
|
|
|
.insert(entity, comp::ForceUpdate)
|
|
|
|
.err()
|
|
|
|
.map(|err| error!("Failed to insert ForceUpdate on dead client: {:?}", err));
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<comp::Energy>()
|
|
|
|
.get_mut(entity)
|
|
|
|
.map(|energy| energy.set_to(energy.maximum(), comp::EnergySource::Revive));
|
|
|
|
let _ = state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<comp::CharacterState>()
|
|
|
|
.insert(entity, comp::CharacterState::default());
|
|
|
|
} else {
|
2020-03-20 10:25:53 +00:00
|
|
|
if state.ecs().read_storage::<comp::Agent>().contains(entity) {
|
|
|
|
// Replace npc with loot
|
|
|
|
let _ = state
|
|
|
|
.ecs()
|
|
|
|
.write_storage()
|
|
|
|
.insert(entity, Body::Object(object::Body::Pouch));
|
|
|
|
let _ = state.ecs().write_storage().insert(
|
|
|
|
entity,
|
|
|
|
assets::load_expect_cloned::<Item>("common.items.cheese"),
|
|
|
|
);
|
|
|
|
state.ecs().write_storage::<comp::Stats>().remove(entity);
|
|
|
|
state.ecs().write_storage::<comp::Agent>().remove(entity);
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<comp::LightEmitter>()
|
|
|
|
.remove(entity);
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<comp::CharacterState>()
|
|
|
|
.remove(entity);
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<comp::Controller>()
|
|
|
|
.remove(entity);
|
|
|
|
} else {
|
|
|
|
if let Err(err) = state.delete_entity_recorded(entity) {
|
|
|
|
error!("Failed to delete destroyed entity: {:?}", err);
|
|
|
|
}
|
|
|
|
}
|
2020-03-18 19:37:11 +00:00
|
|
|
|
|
|
|
// TODO: Add Delete(time_left: Duration) component
|
|
|
|
/*
|
2020-02-16 20:04:06 +00:00
|
|
|
// If not a player delete the entity
|
|
|
|
if let Err(err) = state.delete_entity_recorded(entity) {
|
|
|
|
error!("Failed to delete destroyed entity: {:?}", err);
|
|
|
|
}
|
2020-03-18 19:37:11 +00:00
|
|
|
*/
|
2020-02-16 20:04:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn handle_land_on_ground(server: &Server, entity: EcsEntity, vel: Vec3<f32>) {
|
|
|
|
let state = &server.state;
|
|
|
|
if vel.z <= -37.0 {
|
|
|
|
if let Some(stats) = state.ecs().write_storage::<comp::Stats>().get_mut(entity) {
|
|
|
|
let falldmg = (vel.z / 2.5) as i32;
|
|
|
|
if falldmg < 0 {
|
|
|
|
stats.health.change_by(comp::HealthChange {
|
|
|
|
amount: falldmg,
|
|
|
|
cause: comp::HealthSource::World,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn handle_respawn(server: &Server, entity: EcsEntity) {
|
|
|
|
let state = &server.state;
|
|
|
|
|
|
|
|
// Only clients can respawn
|
|
|
|
if state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<Client>()
|
|
|
|
.get_mut(entity)
|
|
|
|
.is_some()
|
|
|
|
{
|
|
|
|
let respawn_point = state
|
|
|
|
.read_component_cloned::<comp::Waypoint>(entity)
|
|
|
|
.map(|wp| wp.get_pos())
|
|
|
|
.unwrap_or(state.ecs().read_resource::<SpawnPoint>().0);
|
|
|
|
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<comp::Stats>()
|
|
|
|
.get_mut(entity)
|
|
|
|
.map(|stats| stats.revive());
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage::<comp::Pos>()
|
|
|
|
.get_mut(entity)
|
|
|
|
.map(|pos| pos.0 = respawn_point);
|
|
|
|
state
|
|
|
|
.ecs()
|
|
|
|
.write_storage()
|
|
|
|
.insert(entity, comp::ForceUpdate)
|
|
|
|
.err()
|
|
|
|
.map(|err| {
|
|
|
|
error!(
|
|
|
|
"Error inserting ForceUpdate component when respawning client: {:?}",
|
|
|
|
err
|
|
|
|
)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-22 19:39:50 +00:00
|
|
|
pub fn handle_explosion(server: &Server, pos: Vec3<f32>, power: f32, owner: Option<Uid>) {
|
|
|
|
// Go through all other entities
|
2020-03-23 11:50:08 +00:00
|
|
|
let hit_range = 2.0 * power;
|
2020-03-22 19:39:50 +00:00
|
|
|
let ecs = &server.state.ecs();
|
2020-03-23 17:13:14 +00:00
|
|
|
for (pos_b, ori_b, character_b, stats_b) in (
|
2020-03-22 19:39:50 +00:00
|
|
|
&ecs.read_storage::<comp::Pos>(),
|
|
|
|
&ecs.read_storage::<comp::Ori>(),
|
|
|
|
&ecs.read_storage::<comp::CharacterState>(),
|
|
|
|
&mut ecs.write_storage::<comp::Stats>(),
|
|
|
|
)
|
|
|
|
.join()
|
|
|
|
{
|
2020-03-23 11:50:08 +00:00
|
|
|
let distance_squared = pos.distance_squared(pos_b.0);
|
2020-03-22 19:39:50 +00:00
|
|
|
// Check if it is a hit
|
|
|
|
if !stats_b.is_dead
|
|
|
|
// Spherical wedge shaped attack field
|
|
|
|
// RADIUS
|
2020-03-23 11:50:08 +00:00
|
|
|
&& distance_squared < hit_range.powi(2)
|
2020-03-22 19:39:50 +00:00
|
|
|
{
|
|
|
|
// Weapon gives base damage
|
2020-03-23 17:13:14 +00:00
|
|
|
let mut dmg = ((1.0 - distance_squared / hit_range.powi(2)) * power * 10.0) as u32;
|
2020-03-22 19:39:50 +00:00
|
|
|
|
|
|
|
if rand::random() {
|
|
|
|
dmg += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Block
|
|
|
|
if character_b.is_block()
|
|
|
|
&& ori_b.0.angle_between(pos - pos_b.0) < BLOCK_ANGLE.to_radians() / 2.0
|
|
|
|
{
|
|
|
|
dmg = (dmg as f32 * (1.0 - BLOCK_EFFICIENCY)) as u32
|
|
|
|
}
|
|
|
|
|
|
|
|
stats_b.health.change_by(HealthChange {
|
|
|
|
amount: -(dmg as i32),
|
|
|
|
cause: HealthSource::Projectile { owner },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-16 20:04:06 +00:00
|
|
|
const RAYS: usize = 500;
|
|
|
|
|
2020-03-23 17:13:14 +00:00
|
|
|
// Color terrain
|
|
|
|
let mut touched_blocks = Vec::new();
|
2020-02-16 20:04:06 +00:00
|
|
|
for _ in 0..RAYS {
|
|
|
|
let dir = Vec3::new(
|
|
|
|
rand::random::<f32>() - 0.5,
|
|
|
|
rand::random::<f32>() - 0.5,
|
|
|
|
rand::random::<f32>() - 0.5,
|
|
|
|
)
|
|
|
|
.normalized();
|
|
|
|
|
2020-03-23 17:13:14 +00:00
|
|
|
let _ = ecs
|
|
|
|
.read_resource::<TerrainGrid>()
|
|
|
|
.ray(pos, pos + dir * power * 2.5)
|
|
|
|
.until(|_| rand::random::<f32>() < 0.05)
|
|
|
|
.for_each(|pos| touched_blocks.push(pos))
|
|
|
|
.cast();
|
|
|
|
}
|
|
|
|
|
|
|
|
let terrain = ecs.read_resource::<TerrainGrid>();
|
|
|
|
let mut block_change = ecs.write_resource::<BlockChange>();
|
|
|
|
for pos in touched_blocks {
|
|
|
|
if let Ok(block) = terrain.get(pos) {
|
|
|
|
if let Some(mut color) = block.get_color() {
|
|
|
|
color[0] /= 2;
|
|
|
|
color[1] /= 3;
|
|
|
|
color[2] /= 3;
|
|
|
|
block_change.set(pos, Block::new(block.kind(), color));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Destroy terrain
|
|
|
|
for _ in 0..RAYS {
|
|
|
|
let dir = Vec3::new(
|
|
|
|
rand::random::<f32>() - 0.5,
|
|
|
|
rand::random::<f32>() - 0.5,
|
|
|
|
rand::random::<f32>() - 0.5,
|
|
|
|
)
|
|
|
|
.normalized();
|
2020-02-16 20:04:06 +00:00
|
|
|
|
|
|
|
let _ = ecs
|
|
|
|
.read_resource::<TerrainGrid>()
|
2020-03-22 19:39:50 +00:00
|
|
|
.ray(pos, pos + dir * power)
|
2020-02-16 20:04:06 +00:00
|
|
|
.until(|_| rand::random::<f32>() < 0.05)
|
|
|
|
.for_each(|pos| block_change.set(pos, Block::empty()))
|
|
|
|
.cast();
|
|
|
|
}
|
|
|
|
}
|