/* * Copyright (C) 2012 Alec Thomas * All rights reserved. * * This software is licensed as described in the file COPYING, which * you should have received as part of this distribution. * * Author: Alec Thomas */ #pragma once #include #include #include #include #include "entityx/config.h" #include "entityx/Entity.h" #include "entityx/Event.h" #include "entityx/help/NonCopyable.h" namespace entityx { class SystemManager; /** * Base System class. Generally should not be directly used, instead see System. */ class BaseSystem : entityx::help::NonCopyable { public: typedef size_t Family; virtual ~BaseSystem(); /** * Called once all Systems have been added to the SystemManager. * * Typically used to set up event handlers. */ virtual void configure(EntityManager &entities, EventManager &events) { (void)entities; configure(events); } /** * Legacy configure(). Called by default implementation of configure(EntityManager&, EventManager&). */ virtual void configure(EventManager &events) { (void)events; } /** * Apply System behavior. * * Called every game step. */ virtual void update(EntityManager &entities, EventManager &events, TimeDelta dt) = 0; static Family family_counter_; protected: }; /** * Use this class when implementing Systems. * * struct MovementSystem : public System { * void update(EntityManager &entities, EventManager &events, TimeDelta dt) { * // Do stuff to/with entities... * } * } */ template class System : public BaseSystem { public: virtual ~System() {} private: friend class SystemManager; static Family family() { static Family family = family_counter_++; return family; } }; class SystemManager : entityx::help::NonCopyable { public: SystemManager(EntityManager &entity_manager, EventManager &event_manager) : entity_manager_(entity_manager), event_manager_(event_manager) {} /** * Add a System to the SystemManager. * * Must be called before Systems can be used. * * eg. * std::shared_ptr movement = entityx::make_shared(); * system.add(movement); */ template void add(std::shared_ptr system) { systems_.insert(std::make_pair(S::family(), system)); } /** * Add a System to the SystemManager. * * Must be called before Systems can be used. * * eg. * auto movement = system.add(); */ template std::shared_ptr add(Args && ... args) { std::shared_ptr s(new S(std::forward(args) ...)); add(s); return s; } /** * Retrieve the registered System instance, if any. * * std::shared_ptr collisions = systems.system(); * * @return System instance or empty shared_std::shared_ptr. */ template std::shared_ptr system() { auto it = systems_.find(S::family()); assert(it != systems_.end()); return it == systems_.end() ? std::shared_ptr() : std::shared_ptr(std::static_pointer_cast(it->second)); } /** * Call the System::update() method for a registered system. */ template void update(TimeDelta dt) { assert(initialized_ && "SystemManager::configure() not called"); std::shared_ptr s = system(); s->update(entity_manager_, event_manager_, dt); } /** * Call System::update() on all registered systems. * * The order which the registered systems are updated is arbitrary but consistent, * meaning the order which they will be updated cannot be specified, but that order * will stay the same as long no systems are added or removed. * * If the order in which systems update is important, use SystemManager::update() * to manually specify the update order. EntityX does not yet support a way of * specifying priority for update_all(). */ void update_all(TimeDelta dt); /** * Configure the system. Call after adding all Systems. * * This is typically used to set up event handlers. */ void configure(); private: bool initialized_ = false; EntityManager &entity_manager_; EventManager &event_manager_; std::unordered_map> systems_; }; } // namespace entityx