]> git.tdb.fi Git - libs/game.git/blobdiff - source/game/stage.cpp
Report which system caused invalid access
[libs/game.git] / source / game / stage.cpp
index 8d781f28961fc77da688b640c9c579a44c9b4a28..f8dbd1a5b4fa21074b81d1f0d229d6f8e6ffe026 100644 (file)
@@ -1,22 +1,82 @@
 #include "stage.h"
+#include <msp/debug/demangle.h>
+#include <msp/strings/format.h>
+#include "accessguard.h"
+#include "camera.h"
 #include "root.h"
 #include "system.h"
 
+using namespace std;
+
 namespace Msp::Game {
 
-Stage::Stage():
+Stage::Stage(Reflection::Reflector &f, DataFile::Collection &r):
+       reflector(f),
+       resources(r),
        event_source(event_bus),
+       event_observer(event_bus),
        root(std::make_unique<Root>(*this))
-{ }
+{
+       event_observer.observe<Events::ComponentCreated>([this](auto &e){
+               if(!active_camera)
+                       if(Handle<Camera> camera = dynamic_handle_cast<Camera>(e.component))
+                               set_active_camera(camera);
+       });
+}
 
 // Hide unique_ptr destructors from the header
 Stage::~Stage()
 { }
 
+void Stage::remove_system(System &s)
+{
+       erase_if(systems, [&s](auto &p){ return p.get()==&s; });
+}
+
+void Stage::set_active_camera(Handle<Camera> c)
+{
+       active_camera = c;
+       event_source.emit<Events::CameraChanged>(active_camera);
+}
+
+void Stage::synthesize_initial_events(EventObserver &target_observer)
+{
+       synthesize_initial_events(Handle<Entity>::from_object(root.get()), target_observer);
+       if(active_camera)
+               event_source.emit_to<Events::CameraChanged>(target_observer, active_camera);
+}
+
+void Stage::synthesize_initial_events(Handle<Entity> entity, EventObserver &target_observer)
+{
+       for(Handle<Component> c: entity->get_components())
+               event_source.emit_to<Events::ComponentCreated>(target_observer, c);
+       for(Handle<Entity> e: entity->get_children())
+               synthesize_initial_events(e, target_observer);
+       event_source.emit_to<Events::EntityCreated>(target_observer, entity);
+}
+
 void Stage::tick(Time::TimeDelta dt)
 {
+       {
+#ifdef DEBUG
+               AccessGuard::BlockForScope _block;;
+#endif
+               for(const auto &s: 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