2020-12-04 22:24:56 +00:00
|
|
|
use common::{
|
|
|
|
comp::{
|
2021-01-18 22:58:56 +00:00
|
|
|
aura::{AuraChange, AuraKey, AuraKind, AuraTarget},
|
|
|
|
buff,
|
|
|
|
group::Group,
|
|
|
|
Auras, BuffKind, Buffs, CharacterState, Health, Pos,
|
2020-12-04 22:24:56 +00:00
|
|
|
},
|
|
|
|
event::{EventBus, ServerEvent},
|
|
|
|
resources::DeltaTime,
|
2021-01-18 22:58:56 +00:00
|
|
|
uid::UidAllocator,
|
2020-12-04 22:24:56 +00:00
|
|
|
};
|
2021-02-22 19:14:10 +00:00
|
|
|
use specs::{
|
|
|
|
saveload::MarkerAllocator, shred::ResourceId, Entities, Join, Read, ReadStorage, System,
|
|
|
|
SystemData, World, WriteStorage,
|
|
|
|
};
|
2020-12-04 22:24:56 +00:00
|
|
|
use std::time::Duration;
|
|
|
|
|
2021-02-22 19:14:10 +00:00
|
|
|
#[derive(SystemData)]
|
2021-02-22 21:02:37 +00:00
|
|
|
pub struct ReadData<'a> {
|
2021-02-22 19:14:10 +00:00
|
|
|
entities: Entities<'a>,
|
|
|
|
dt: Read<'a, DeltaTime>,
|
|
|
|
server_bus: Read<'a, EventBus<ServerEvent>>,
|
|
|
|
uid_allocator: Read<'a, UidAllocator>,
|
|
|
|
positions: ReadStorage<'a, Pos>,
|
|
|
|
char_states: ReadStorage<'a, CharacterState>,
|
|
|
|
healths: ReadStorage<'a, Health>,
|
|
|
|
groups: ReadStorage<'a, Group>,
|
|
|
|
buffs: ReadStorage<'a, Buffs>,
|
|
|
|
}
|
|
|
|
|
2020-12-04 22:24:56 +00:00
|
|
|
pub struct Sys;
|
|
|
|
impl<'a> System<'a> for Sys {
|
2021-02-22 21:02:37 +00:00
|
|
|
type SystemData = (ReadData<'a>, WriteStorage<'a, Auras>);
|
2020-12-04 22:24:56 +00:00
|
|
|
|
2021-02-22 21:02:37 +00:00
|
|
|
fn run(&mut self, (read_data, mut auras): Self::SystemData) {
|
|
|
|
let mut server_emitter = read_data.server_bus.emitter();
|
|
|
|
let dt = read_data.dt.0;
|
2020-12-04 22:24:56 +00:00
|
|
|
|
|
|
|
auras.set_event_emission(false);
|
|
|
|
|
|
|
|
// Iterate through all entities with an aura
|
2021-02-22 21:02:37 +00:00
|
|
|
for (entity, pos, mut auras_comp) in
|
|
|
|
(&read_data.entities, &read_data.positions, &mut auras).join()
|
2021-02-22 19:14:10 +00:00
|
|
|
{
|
2020-12-04 22:24:56 +00:00
|
|
|
let mut expired_auras = Vec::<AuraKey>::new();
|
|
|
|
// Iterate through the auras attached to this entity
|
|
|
|
for (key, aura) in auras_comp.auras.iter_mut() {
|
|
|
|
// Tick the aura and subtract dt from it
|
|
|
|
if let Some(remaining_time) = &mut aura.duration {
|
|
|
|
if let Some(new_duration) =
|
2021-02-22 19:14:10 +00:00
|
|
|
remaining_time.checked_sub(Duration::from_secs_f32(dt))
|
2020-12-04 22:24:56 +00:00
|
|
|
{
|
|
|
|
*remaining_time = new_duration;
|
|
|
|
} else {
|
|
|
|
*remaining_time = Duration::default();
|
|
|
|
expired_auras.push(key);
|
|
|
|
}
|
|
|
|
}
|
2021-02-22 19:14:10 +00:00
|
|
|
for (target, target_pos, target_buffs, health) in (
|
2021-02-22 21:02:37 +00:00
|
|
|
&read_data.entities,
|
|
|
|
&read_data.positions,
|
|
|
|
&read_data.buffs,
|
|
|
|
&read_data.healths,
|
2021-01-14 12:21:55 +00:00
|
|
|
)
|
|
|
|
.join()
|
2020-12-04 22:24:56 +00:00
|
|
|
{
|
|
|
|
// Ensure entity is within the aura radius
|
|
|
|
if target_pos.0.distance_squared(pos.0) < aura.radius.powi(2) {
|
2021-01-18 22:58:56 +00:00
|
|
|
if let AuraTarget::GroupOf(uid) = aura.target {
|
2021-02-22 21:02:37 +00:00
|
|
|
let same_group = read_data
|
2021-02-22 19:14:10 +00:00
|
|
|
.uid_allocator
|
2021-01-18 22:58:56 +00:00
|
|
|
.retrieve_entity_internal(uid.into())
|
2021-02-22 21:02:37 +00:00
|
|
|
.and_then(|e| read_data.groups.get(e))
|
2021-01-18 22:58:56 +00:00
|
|
|
.map_or(false, |owner_group| {
|
2021-02-22 21:02:37 +00:00
|
|
|
Some(owner_group) == read_data.groups.get(target)
|
2021-01-18 22:58:56 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
if !same_group {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-04 22:24:56 +00:00
|
|
|
// TODO: When more aura kinds (besides Buff) are
|
|
|
|
// implemented, match on them here
|
|
|
|
match aura.aura_kind {
|
|
|
|
AuraKind::Buff {
|
|
|
|
kind,
|
|
|
|
data,
|
|
|
|
category,
|
|
|
|
source,
|
|
|
|
} => {
|
|
|
|
// Checks if the buff is not active so it isn't applied
|
|
|
|
// every tick, but rather only once it runs out
|
|
|
|
// TODO: Check for stronger buff of same kind so it can replace
|
|
|
|
// active buff.
|
|
|
|
if !target_buffs.contains(kind) {
|
|
|
|
// Conditions for different buffs are in this match
|
|
|
|
// statement
|
|
|
|
let apply_buff = match kind {
|
2021-01-14 12:21:55 +00:00
|
|
|
BuffKind::CampfireHeal => {
|
|
|
|
matches!(
|
2021-02-22 21:02:37 +00:00
|
|
|
read_data.char_states.get(target),
|
2021-01-14 12:21:55 +00:00
|
|
|
Some(CharacterState::Sit)
|
|
|
|
) && health.current() < health.maximum()
|
|
|
|
},
|
2020-12-04 22:24:56 +00:00
|
|
|
// Add other specific buff conditions here
|
|
|
|
_ => true,
|
|
|
|
};
|
|
|
|
if apply_buff {
|
|
|
|
use buff::*;
|
|
|
|
server_emitter.emit(ServerEvent::Buff {
|
2021-02-22 19:14:10 +00:00
|
|
|
entity: target,
|
2020-12-04 22:24:56 +00:00
|
|
|
buff_change: BuffChange::Add(Buff::new(
|
|
|
|
kind,
|
|
|
|
data,
|
|
|
|
vec![category],
|
|
|
|
source,
|
|
|
|
)),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !expired_auras.is_empty() {
|
|
|
|
server_emitter.emit(ServerEvent::Aura {
|
|
|
|
entity,
|
|
|
|
aura_change: AuraChange::RemoveByKey(expired_auras),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
auras.set_event_emission(true);
|
|
|
|
}
|
|
|
|
}
|