From f800dbc034e7a70a613bab8cd9d147be4f6e88b6 Mon Sep 17 00:00:00 2001 From: Clyne Sullivan Date: Tue, 10 Jan 2017 21:26:13 -0500 Subject: windows build --- entityx/Entity.h | 1104 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 1104 insertions(+) create mode 100644 entityx/Entity.h (limited to 'entityx/Entity.h') diff --git a/entityx/Entity.h b/entityx/Entity.h new file mode 100644 index 0000000..3217ae5 --- /dev/null +++ b/entityx/Entity.h @@ -0,0 +1,1104 @@ +/* + * 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 "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; + + + +/** 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; + + 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) { + for (auto it : *this) + f(it, *(it.template component().get())...); + } + + 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) { + return entities_with_components().each(f); + } + + /** + * 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()); + } +}; +} + -- cgit v1.2.3