/* * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include "entityx/help/Pool.h" #include "entityx/config.h" #include "entityx/Event.h" #include "entityx/help/NonCopyable.h" namespace entityx { typedef std::uint32_t uint32_t; typedef std::uint64_t uint64_t; class EntityManager; template class ComponentHandle; struct Killed {}; /** A convenience handle around an Entity::Id. * * If an entity is destroyed, any copies will be invalidated. Use valid() to * check for validity before using. * * Create entities with `EntityManager`: * * Entity entity = entity_manager->create(); */ class Entity { public: struct Id { Id() : id_(0) {} explicit Id(uint64_t id) : id_(id) {} Id(uint32_t index, uint32_t version) : id_(uint64_t(index) | uint64_t(version) << 32UL) {} uint64_t id() const { return id_; } bool operator == (const Id &other) const { return id_ == other.id_; } bool operator != (const Id &other) const { return id_ != other.id_; } bool operator < (const Id &other) const { return id_ < other.id_; } uint32_t index() const { return id_ & 0xffffffffUL; } uint32_t version() const { return id_ >> 32; } private: friend class EntityManager; uint64_t id_; }; /** * Id of an invalid Entity. */ static const Id INVALID; Entity() = default; Entity(EntityManager *manager, Entity::Id id) : manager_(manager), id_(id) {} Entity(const Entity &other) = default; Entity &operator = (const Entity &other) = default; /** * Check if Entity handle is invalid. */ operator bool() const { return valid(); } bool operator == (const Entity &other) const { return other.manager_ == manager_ && other.id_ == id_; } bool operator != (const Entity &other) const { return !(other == *this); } bool operator < (const Entity &other) const { return other.id_ < id_; } /** * Is this Entity handle valid? * * In older versions of EntityX, there were no guarantees around entity * validity if a previously allocated entity slot was reassigned. That is no * longer the case: if a slot is reassigned, old Entity::Id's will be * invalid. */ bool valid() const; /** * Invalidate Entity handle, disassociating it from an EntityManager and invalidating its ID. * * Note that this does *not* affect the underlying entity and its * components. Use destroy() to destroy the associated Entity and components. */ void invalidate(); Id id() const { return id_; } template ComponentHandle assign(Args && ... args); template ComponentHandle assign_from_copy(const C &component); template ComponentHandle replace(Args && ... args); template void remove(); template ::value>::type> ComponentHandle component(); template ::value>::type> const ComponentHandle component() const; template std::tuple...> components(); template std::tuple...> components() const; template bool has_component() const; template void unpack(ComponentHandle &a, ComponentHandle & ... args); /** * Destroy and invalidate this Entity. */ void destroy(); std::bitset component_mask() const; void kill(); private: EntityManager *manager_ = nullptr; Entity::Id id_ = INVALID; }; /** * A ComponentHandle is a wrapper around an instance of a component. * * It provides safe access to components. The handle will be invalidated under * the following conditions: * * - If a component is removed from its host entity. * - If its host entity is destroyed. */ template class ComponentHandle { public: typedef C ComponentType; ComponentHandle() : manager_(nullptr) {} bool valid() const; operator bool() const; C *operator -> (); const C *operator -> () const; C &operator * (); const C &operator * () const; C *get(); const C *get() const; /** * Remove the component from its entity and destroy it. */ void remove(); /** * Returns the Entity associated with the component */ Entity entity(); bool operator == (const ComponentHandle &other) const { return manager_ == other.manager_ && id_ == other.id_; } bool operator != (const ComponentHandle &other) const { return !(*this == other); } private: friend class EntityManager; ComponentHandle(EM *manager, Entity::Id id) : manager_(manager), id_(id) {} EM *manager_; Entity::Id id_; }; /** * Base component class, only used for insertion into collections. * * Family is used for registration. */ struct BaseComponent { public: typedef size_t Family; // NOTE: Component memory is *always* managed by the EntityManager. // Use Entity::destroy() instead. void operator delete(void *p) { (void)p; fail(); } void operator delete[](void *p) { (void)p; fail(); } protected: static void fail() { #if defined(_HAS_EXCEPTIONS) || defined(__EXCEPTIONS) throw std::bad_alloc(); #else std::abort(); #endif } static Family family_counter_; }; /** * Component implementations should inherit from this. * * Components MUST provide a no-argument constructor. * Components SHOULD provide convenience constructors for initializing on assignment to an Entity::Id. * * This is a struct to imply that components should be data-only. * * Usage: * * struct Position : public Component { * Position(float x = 0.0f, float y = 0.0f) : x(x), y(y) {} * * float x, y; * }; * * family() is used for registration. */ template struct Component : public BaseComponent { public: typedef ComponentHandle Handle; typedef ComponentHandle ConstHandle; private: friend class EntityManager; /// Used internally for registration. static Family family(); }; /** * Emitted when an entity is added to the system. */ struct EntityCreatedEvent : public Event { explicit EntityCreatedEvent(Entity entity) : entity(entity) {} virtual ~EntityCreatedEvent(); Entity entity; }; /** * Called just prior to an entity being destroyed. */ struct EntityDestroyedEvent : public Event { explicit EntityDestroyedEvent(Entity entity) : entity(entity) {} virtual ~EntityDestroyedEvent(); Entity entity; }; /** * Emitted when any component is added to an entity. */ template struct ComponentAddedEvent : public Event> { ComponentAddedEvent(Entity entity, ComponentHandle component) : entity(entity), component(component) {} Entity entity; ComponentHandle component; }; /** * Emitted when any component is removed from an entity. */ template struct ComponentRemovedEvent : public Event> { ComponentRemovedEvent(Entity entity, ComponentHandle component) : entity(entity), component(component) {} Entity entity; ComponentHandle component; }; /** * Helper class to perform component operations in a typed manner. */ class BaseComponentHelper { public: virtual ~BaseComponentHelper() {} virtual void remove_component(Entity e) = 0; virtual void copy_component_to(Entity source, Entity target) = 0; }; template class ComponentHelper : public BaseComponentHelper { public: void remove_component(Entity e) override { e.remove(); } void copy_component_to(Entity source, Entity target) override { target.assign_from_copy(*(source.component().get())); } }; /** * Manages Entity::Id creation and component assignment. */ class EntityManager : entityx::help::NonCopyable { public: typedef std::bitset ComponentMask; explicit EntityManager(EventManager &event_manager); virtual ~EntityManager(); /// An iterator over a view of the entities in an EntityManager. /// If All is true it will iterate over all valid entities and will ignore the entity mask. template class ViewIterator : public std::iterator { public: Delegate &operator ++() { ++i_; next(); return *static_cast(this); } bool operator == (const Delegate& rhs) const { return i_ == rhs.i_; } bool operator != (const Delegate& rhs) const { return i_ != rhs.i_; } Entity operator * () { return Entity(manager_, manager_->create_id(i_)); } const Entity operator * () const { return Entity(manager_, manager_->create_id(i_)); } protected: ViewIterator(EntityManager *manager, uint32_t index) : manager_(manager), i_(index), capacity_(manager_->capacity()), free_cursor_(~0UL) { if (All) { std::sort(manager_->free_list_.begin(), manager_->free_list_.end()); free_cursor_ = 0; } } ViewIterator(EntityManager *manager, const ComponentMask mask, uint32_t index) : manager_(manager), mask_(mask), i_(index), capacity_(manager_->capacity()), free_cursor_(~0UL) { if (All) { std::sort(manager_->free_list_.begin(), manager_->free_list_.end()); free_cursor_ = 0; } } void next() { while (i_ < capacity_ && !predicate()) { ++i_; } if (i_ < capacity_) { Entity entity = manager_->get(manager_->create_id(i_)); static_cast(this)->next_entity(entity); } } inline bool predicate() { return (All && valid_entity()) || (manager_->entity_component_mask_[i_] & mask_) == mask_; } inline bool valid_entity() { const std::vector &free_list = manager_->free_list_; if (free_cursor_ < free_list.size() && free_list[free_cursor_] == i_) { ++free_cursor_; return false; } return true; } EntityManager *manager_; ComponentMask mask_; uint32_t i_; size_t capacity_; size_t free_cursor_; }; template class BaseView { public: class Iterator : public ViewIterator { public: Iterator(EntityManager *manager, const ComponentMask mask, uint32_t index) : ViewIterator(manager, mask, index) { ViewIterator::next(); } void next_entity(Entity &entity) { (void)entity; } }; Iterator begin() { return Iterator(manager_, mask_, 0); } Iterator end() { return Iterator(manager_, mask_, uint32_t(manager_->capacity())); } const Iterator begin() const { return Iterator(manager_, mask_, 0); } const Iterator end() const { return Iterator(manager_, mask_, manager_->capacity()); } private: friend class EntityManager; explicit BaseView(EntityManager *manager) : manager_(manager) { mask_.set(); } BaseView(EntityManager *manager, ComponentMask mask) : manager_(manager), mask_(mask) {} EntityManager *manager_; ComponentMask mask_; }; template class TypedView: public BaseView { public: template struct identity { typedef T type; }; void each(typename identity>::type f, bool dead = false) { static std::mutex locked; locked.lock(); for (Entity it : *this) { if (dead || !it.has_component()) f(it, *(it.template component().get())...); } locked.unlock(); } private: friend class EntityManager; explicit TypedView(EntityManager *manager) : BaseView(manager) {} TypedView(EntityManager *manager, ComponentMask mask) : BaseView(manager, mask) {} }; template using View = TypedView; typedef BaseView DebugView; template class UnpackingView { public: struct Unpacker { explicit Unpacker(ComponentHandle & ... handles) : handles(std::tuple & ...>(handles...)) {} void unpack(entityx::Entity &entity) const { unpack_<0, Components...>(entity); } private: template void unpack_(entityx::Entity &entity) const { std::get(handles) = entity.component(); } template void unpack_(entityx::Entity &entity) const { std::get(handles) = entity.component(); unpack_(entity); } std::tuple & ...> handles; }; class Iterator : public ViewIterator { public: Iterator(EntityManager *manager, const ComponentMask mask, uint32_t index, const Unpacker &unpacker) : ViewIterator(manager, mask, index), unpacker_(unpacker) { ViewIterator::next(); } void next_entity(Entity &entity) { unpacker_.unpack(entity); } private: const Unpacker &unpacker_; }; Iterator begin() { return Iterator(manager_, mask_, 0, unpacker_); } Iterator end() { return Iterator(manager_, mask_, static_cast(manager_->capacity()), unpacker_); } const Iterator begin() const { return Iterator(manager_, mask_, 0, unpacker_); } const Iterator end() const { return Iterator(manager_, mask_, static_cast(manager_->capacity()), unpacker_); } private: friend class EntityManager; UnpackingView(EntityManager *manager, ComponentMask mask, ComponentHandle & ... handles) : manager_(manager), mask_(mask), unpacker_(handles...) {} EntityManager *manager_; ComponentMask mask_; Unpacker unpacker_; }; /** * Number of managed entities. */ size_t size() const { return entity_component_mask_.size() - free_list_.size(); } /** * Current entity capacity. */ size_t capacity() const { return entity_component_mask_.size(); } /** * Return true if the given entity ID is still valid. */ bool valid(Entity::Id id) const { return id.index() < entity_version_.size() && entity_version_[id.index()] == id.version(); } /** * Create a new Entity::Id. * * Emits EntityCreatedEvent. */ Entity create() { uint32_t index, version; if (free_list_.empty()) { index = index_counter_++; accomodate_entity(index); version = entity_version_[index] = 1; } else { index = free_list_.back(); free_list_.pop_back(); version = entity_version_[index]; } Entity entity(this, Entity::Id(index, version)); event_manager_.emit(entity); return entity; } /** * Create a new Entity by copying another. Copy-constructs each component. * * Emits EntityCreatedEvent. */ Entity create_from_copy(Entity original) { assert(original.valid()); auto clone = create(); auto mask = original.component_mask(); for (size_t i = 0; i < component_helpers_.size(); i++) { BaseComponentHelper *helper = component_helpers_[i]; if (helper && mask.test(i)) helper->copy_component_to(original, clone); } return clone; } /** * Destroy an existing Entity::Id and its associated Components. * * Emits EntityDestroyedEvent. */ void destroy(Entity::Id entity) { assert_valid(entity); uint32_t index = entity.index(); auto mask = entity_component_mask_[index]; for (size_t i = 0; i < component_helpers_.size(); i++) { BaseComponentHelper *helper = component_helpers_[i]; if (helper && mask.test(i)) helper->remove_component(Entity(this, entity)); } event_manager_.emit(Entity(this, entity)); entity_component_mask_[index].reset(); entity_version_[index]++; free_list_.push_back(index); } Entity get(Entity::Id id) { assert_valid(id); return Entity(this, id); } /** * Create an Entity::Id for a slot. * * NOTE: Does *not* check for validity, but the Entity::Id constructor will * fail if the ID is invalid. */ Entity::Id create_id(uint32_t index) const { return Entity::Id(index, entity_version_[index]); } /** * Assign a Component to an Entity::Id, passing through Component constructor arguments. * * Position &position = em.assign(e, x, y); * * @returns Smart pointer to newly created component. */ template ComponentHandle assign(Entity::Id id, Args && ... args) { assert_valid(id); const BaseComponent::Family family = component_family(); assert(!entity_component_mask_[id.index()].test(family)); // Placement new into the component pool. Pool *pool = accomodate_component(); ::new(pool->get(id.index())) C(std::forward(args) ...); // Set the bit for this component. entity_component_mask_[id.index()].set(family); // Create and return handle. ComponentHandle component(this, id); event_manager_.emit>(Entity(this, id), component); return component; } /** * Remove a Component from an Entity::Id * * Emits a ComponentRemovedEvent event. */ template void remove(Entity::Id id) { assert_valid(id); const BaseComponent::Family family = component_family(); const uint32_t index = id.index(); // Find the pool for this component family. BasePool *pool = component_pools_[family]; ComponentHandle component(this, id); event_manager_.emit>(Entity(this, id), component); // Remove component bit. entity_component_mask_[id.index()].reset(family); // Call destructor. pool->destroy(index); } /** * Check if an Entity has a component. */ template bool has_component(Entity::Id id) const { assert_valid(id); size_t family = component_family(); // We don't bother checking the component mask, as we return a nullptr anyway. if (family >= component_pools_.size()) return false; BasePool *pool = component_pools_[family]; if (!pool || !entity_component_mask_[id.index()][family]) return false; return true; } /** * Retrieve a Component assigned to an Entity::Id. * * @returns Pointer to an instance of C, or nullptr if the Entity::Id does not have that Component. */ template ::value>::type> ComponentHandle component(Entity::Id id) { assert_valid(id); size_t family = component_family(); // We don't bother checking the component mask, as we return a nullptr anyway. if (family >= component_pools_.size()) return ComponentHandle(); BasePool *pool = component_pools_[family]; if (!pool || !entity_component_mask_[id.index()][family]) return ComponentHandle(); return ComponentHandle(this, id); } /** * Retrieve a Component assigned to an Entity::Id. * * @returns Component instance, or nullptr if the Entity::Id does not have that Component. */ template ::value>::type> const ComponentHandle component(Entity::Id id) const { assert_valid(id); size_t family = component_family(); // We don't bother checking the component mask, as we return a nullptr anyway. if (family >= component_pools_.size()) return ComponentHandle(); BasePool *pool = component_pools_[family]; if (!pool || !entity_component_mask_[id.index()][family]) return ComponentHandle(); return ComponentHandle(this, id); } template std::tuple...> components(Entity::Id id) { return std::make_tuple(component(id)...); } template std::tuple...> components(Entity::Id id) const { return std::make_tuple(component(id)...); } /** * Find Entities that have all of the specified Components. * * @code * for (Entity entity : entity_manager.entities_with_components()) { * ComponentHandle position = entity.component(); * ComponentHandle direction = entity.component(); * * ... * } * @endcode */ template View entities_with_components() { auto mask = component_mask(); return View(this, mask); } template struct identity { typedef T type; }; template void each(typename identity>::type f, bool dead = false) { return entities_with_components().each(f, dead); } /** * Find Entities that have all of the specified Components and assign them * to the given parameters. * * @code * ComponentHandle position; * ComponentHandle direction; * for (Entity entity : entity_manager.entities_with_components(position, direction)) { * // Use position and component here. * } * @endcode */ template UnpackingView entities_with_components(ComponentHandle & ... components) { auto mask = component_mask(); return UnpackingView(this, mask, components...); } /** * Iterate over all *valid* entities (ie. not in the free list). Not fast, * so should only be used for debugging. * * @code * for (Entity entity : entity_manager.entities_for_debugging()) {} * * @return An iterator view over all valid entities. */ DebugView entities_for_debugging() { return DebugView(this); } template void unpack(Entity::Id id, ComponentHandle &a) { assert_valid(id); a = component(id); } /** * Unpack components directly into pointers. * * Components missing from the entity will be set to nullptr. * * Useful for fast bulk iterations. * * ComponentHandle p; * ComponentHandle d; * unpack(e, p, d); */ template void unpack(Entity::Id id, ComponentHandle &a, ComponentHandle & ... args) { assert_valid(id); a = component(id); unpack(id, args ...); } /** * Destroy all entities and reset the EntityManager. */ void reset(); // Retrieve the component family for a type. template static BaseComponent::Family component_family() { return Component::type>::family(); } private: friend class Entity; template friend class ComponentHandle; inline void assert_valid(Entity::Id id) const { assert(id.index() < entity_component_mask_.size() && "Entity::Id ID outside entity vector range"); assert(entity_version_[id.index()] == id.version() && "Attempt to access Entity via a stale Entity::Id"); } template C *get_component_ptr(Entity::Id id) { assert(valid(id)); BasePool *pool = component_pools_[component_family()]; assert(pool); return static_cast(pool->get(id.index())); } template const C *get_component_ptr(Entity::Id id) const { assert_valid(id); BasePool *pool = component_pools_[component_family()]; assert(pool); return static_cast(pool->get(id.index())); } ComponentMask component_mask(Entity::Id id) { assert_valid(id); return entity_component_mask_.at(id.index()); } template ComponentMask component_mask() { ComponentMask mask; mask.set(component_family()); return mask; } template ComponentMask component_mask() { return component_mask() | component_mask(); } template ComponentMask component_mask(const ComponentHandle &c) { return component_mask(); } template ComponentMask component_mask(const ComponentHandle &c1, const ComponentHandle &... args) { return component_mask(); } inline void accomodate_entity(uint32_t index) { if (entity_component_mask_.size() <= index) { entity_component_mask_.resize(index + 1); entity_version_.resize(index + 1); for (BasePool *pool : component_pools_) if (pool) pool->expand(index + 1); } } template Pool *accomodate_component() { BaseComponent::Family family = component_family(); if (component_pools_.size() <= family) { component_pools_.resize(family + 1, nullptr); } if (!component_pools_[family]) { Pool *pool = new Pool(); pool->expand(index_counter_); component_pools_[family] = pool; } if (component_helpers_.size() <= family) { component_helpers_.resize(family + 1, nullptr); } if (!component_helpers_[family]) { ComponentHelper *helper = new ComponentHelper(); component_helpers_[family] = helper; } return static_cast*>(component_pools_[family]); } uint32_t index_counter_ = 0; EventManager &event_manager_; // Each element in component_pools_ corresponds to a Pool for a Component. // The index into the vector is the Component::family(). std::vector component_pools_; // Each element in component_helpers_ corresponds to a ComponentHelper for a Component type. // The index into the vector is the Component::family(). std::vector component_helpers_; // Bitmask of components associated with each entity. Index into the vector is the Entity::Id. std::vector entity_component_mask_; // Vector of entity version numbers. Incremented each time an entity is destroyed std::vector entity_version_; // List of available entity slots. std::vector free_list_; }; template BaseComponent::Family Component::family() { static Family family = family_counter_++; assert(family < entityx::MAX_COMPONENTS); return family; } template ComponentHandle Entity::assign(Args && ... args) { assert(valid()); return manager_->assign(id_, std::forward(args) ...); } template ComponentHandle Entity::assign_from_copy(const C &component) { assert(valid()); return manager_->assign(id_, std::forward(component)); } template ComponentHandle Entity::replace(Args && ... args) { assert(valid()); auto handle = component(); if (handle) { *(handle.get()) = C(std::forward(args) ...); } else { handle = manager_->assign(id_, std::forward(args) ...); } return handle; } template void Entity::remove() { assert(valid() && has_component()); manager_->remove(id_); } template ComponentHandle Entity::component() { assert(valid()); return manager_->component(id_); } template const ComponentHandle Entity::component() const { assert(valid()); return const_cast(manager_)->component(id_); } template std::tuple...> Entity::components() { assert(valid()); return manager_->components(id_); } template std::tuple...> Entity::components() const { assert(valid()); return const_cast(manager_)->components(id_); } template bool Entity::has_component() const { assert(valid()); return manager_->has_component(id_); } template void Entity::unpack(ComponentHandle &a, ComponentHandle & ... args) { assert(valid()); manager_->unpack(id_, a, args ...); } inline bool Entity::valid() const { return manager_ && manager_->valid(id_); } inline std::ostream &operator << (std::ostream &out, const Entity::Id &id) { out << "Entity::Id(" << id.index() << "." << id.version() << ")"; return out; } inline std::ostream &operator << (std::ostream &out, const Entity &entity) { out << "Entity(" << entity.id() << ")"; return out; } template inline ComponentHandle::operator bool() const { return valid(); } template inline bool ComponentHandle::valid() const { return manager_ && manager_->valid(id_) && manager_->template has_component(id_); } template inline C *ComponentHandle::operator -> () { assert(valid()); return manager_->template get_component_ptr(id_); } template inline const C *ComponentHandle::operator -> () const { assert(valid()); return manager_->template get_component_ptr(id_); } template inline C &ComponentHandle::operator * () { assert(valid()); return *manager_->template get_component_ptr(id_); } template inline const C &ComponentHandle::operator * () const { assert(valid()); return *manager_->template get_component_ptr(id_); } template inline C *ComponentHandle::get() { assert(valid()); return manager_->template get_component_ptr(id_); } template inline const C *ComponentHandle::get() const { assert(valid()); return manager_->template get_component_ptr(id_); } template inline void ComponentHandle::remove() { assert(valid()); manager_->template remove(id_); } template inline Entity ComponentHandle::entity() { assert(valid()); return manager_->get(id_); } } // namespace entityx namespace std { template <> struct hash { std::size_t operator () (const entityx::Entity &entity) const { return static_cast(entity.id().index() ^ entity.id().version()); } }; template <> struct hash { std::size_t operator () (const entityx::Entity &entity) const { return static_cast(entity.id().index() ^ entity.id().version()); } }; }