X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=source%2Fgame%2Fstage.cpp;h=3888410a43c8bc976034b753693b56ab35f16d72;hb=e4f03880d49bdbe0c7269be0f40f23b197bcea77;hp=1c56c7e8a159abe3e41f32bb3b4b629eb4ffb533;hpb=f298027c2042b63cec903c98dfc97e792a4f923f;p=libs%2Fgame.git diff --git a/source/game/stage.cpp b/source/game/stage.cpp index 1c56c7e..3888410 100644 --- a/source/game/stage.cpp +++ b/source/game/stage.cpp @@ -1,4 +1,7 @@ #include "stage.h" +#include +#include +#include "accessguard.h" #include "camera.h" #include "root.h" #include "system.h" @@ -7,11 +10,13 @@ using namespace std; namespace Msp::Game { -Stage::Stage(DataFile::Collection &r): +Stage::Stage(Reflection::Reflector &f, DataFile::Collection &r): + reflector(f), resources(r), event_source(event_bus), event_observer(event_bus), - root(std::make_unique(*this)) + root(std::make_unique(*this)), + scheduler(reflector) { event_observer.observe([this](auto &e){ if(!active_camera) @@ -26,7 +31,9 @@ Stage::~Stage() void Stage::remove_system(System &s) { + scheduler.remove_system(s); erase_if(systems, [&s](auto &p){ return p.get()==&s; }); + pending_reschedule = true; } void Stage::set_active_camera(Handle c) @@ -35,10 +42,51 @@ void Stage::set_active_camera(Handle c) event_source.emit(active_camera); } +void Stage::synthesize_initial_events(EventObserver &target_observer) +{ + synthesize_initial_events(Handle::from_object(root.get()), target_observer); + if(active_camera) + event_source.emit_to(target_observer, active_camera); +} + +void Stage::synthesize_initial_events(Handle entity, EventObserver &target_observer) +{ + for(Handle c: entity->get_components()) + event_source.emit_to(target_observer, c); + for(Handle e: entity->get_children()) + synthesize_initial_events(e, target_observer); + event_source.emit_to(target_observer, entity); +} + void Stage::tick(Time::TimeDelta dt) { + if(pending_reschedule) + { + scheduler.schedule(); + pending_reschedule = false; + } + + { +#ifdef DEBUG + AccessGuard::BlockForScope _block; +#endif + for(const SystemScheduler::Group &g: scheduler.get_groups()) + for(System *s: g.systems) + { + System::Activator act(*s); + try + { + s->tick(dt); + } + catch(const invalid_access &exc) + { + throw invalid_access(format("%s by %s", exc.what(), Debug::demangle(typeid(*s).name()))); + } + } + } + for(const auto &s: systems) - s->tick(dt); + s->deferred_tick(); } } // namespace Msp::Game