1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118
use crate::{persistence::character_updater, sys::SysScheduler};
use common::{
comp::{
pet::{is_tameable, Pet},
ActiveAbilities, Alignment, Body, Inventory, MapMarker, Presence, PresenceKind, SkillSet,
Stats, Waypoint,
},
uid::Uid,
};
use common_ecs::{Job, Origin, Phase, System};
use specs::{Join, LendJoin, ReadStorage, Write, WriteExpect};
use tracing::error;
#[derive(Default)]
pub struct Sys;
impl<'a> System<'a> for Sys {
type SystemData = (
ReadStorage<'a, Alignment>,
ReadStorage<'a, Body>,
ReadStorage<'a, Presence>,
ReadStorage<'a, SkillSet>,
ReadStorage<'a, Inventory>,
ReadStorage<'a, Uid>,
ReadStorage<'a, Waypoint>,
ReadStorage<'a, MapMarker>,
ReadStorage<'a, Pet>,
ReadStorage<'a, Stats>,
ReadStorage<'a, ActiveAbilities>,
WriteExpect<'a, character_updater::CharacterUpdater>,
Write<'a, SysScheduler<Self>>,
);
const NAME: &'static str = "persistence";
const ORIGIN: Origin = Origin::Server;
const PHASE: Phase = Phase::Create;
fn run(
_job: &mut Job<Self>,
(
alignments,
bodies,
presences,
player_skill_set,
player_inventories,
uids,
player_waypoints,
map_markers,
pets,
stats,
active_abilities,
mut updater,
mut scheduler,
): Self::SystemData,
) {
if scheduler.should_run() {
updater.batch_update(
(
&presences,
&player_skill_set,
&player_inventories,
&uids,
player_waypoints.maybe(),
&active_abilities,
map_markers.maybe(),
)
.join()
.filter_map(
|(
presence,
skill_set,
inventory,
player_uid,
waypoint,
active_abilities,
map_marker,
)| match presence.kind {
PresenceKind::LoadingCharacter(_char_id) => {
error!(
"Unexpected state when persisting characters! Some of the \
components required above should only be present after a \
character is loaded!"
);
None
},
PresenceKind::Character(id) => {
let pets = (&alignments, &bodies, &stats, &pets)
.join()
.filter_map(|(alignment, body, stats, pet)| match alignment {
// Don't try to persist non-tameable pets (likely spawned
// using /spawn) since there isn't any code to handle
// persisting them
Alignment::Owned(ref pet_owner)
if pet_owner == player_uid && is_tameable(body) =>
{
Some(((*pet).clone(), *body, stats.clone()))
},
_ => None,
})
.collect();
Some((
id,
skill_set.clone(),
inventory.clone(),
pets,
waypoint.cloned(),
active_abilities.clone(),
map_marker.cloned(),
))
},
PresenceKind::Spectator | PresenceKind::Possessor => None,
},
),
);
}
}
}