diff options
author | Alec Thomas <alec@swapoff.org> | 2015-07-18 21:15:23 -0400 |
---|---|---|
committer | Alec Thomas <alec@swapoff.org> | 2015-07-18 21:16:53 -0400 |
commit | c4d518bbdd136dde3826ff2e0e51ae84d7613720 (patch) | |
tree | bef10aad31d50b113a7876fceaf2e35a89fa7b11 | |
parent | 243bee5ae16d33ba24d4247ec399df750aa07699 (diff) |
Add `.each<C...>([](C &c...) {})` iteration.
May break code that uses `EntityManager::View`, as this is now a
template type `EntityManager::View<C...>`.
Fixes #62.
-rw-r--r-- | README.md | 28 | ||||
-rw-r--r-- | entityx/Entity.h | 42 | ||||
-rw-r--r-- | entityx/Entity_test.cc | 37 | ||||
-rw-r--r-- | entityx/System_test.cc | 6 | ||||
-rw-r--r-- | examples/example.cc | 83 |
5 files changed, 135 insertions, 61 deletions
@@ -143,7 +143,23 @@ entity.assign<Position>(1.0f, 2.0f); #### Querying entities and their components -To query all entities with a set of components assigned, use ``entityx::EntityManager::entities_with_components()``. This method will return only those entities that have *all* of the specified components associated with them, assigning each component pointer to the corresponding component instance: +To query all entities with a set of components assigned you can use two +methods. Both methods will return only those entities that have *all* of the +specified components associated with them. + +`entityx::EntityManager::each(f) provides functional-style iteration over +`entity components. The callback for `each()` can optionally accept an Entity as +`its first argument. + +```c++ +entities.each<Position, Direction>([](Entity entity, Position &position, Direction &direction) { + // Do things with entity, position and direction. +};) +``` + + +For iterator-style traversal of components, use +``entityx::EntityManager::entities_with_components()``: ```c++ ComponentHandle<Position> position; @@ -189,12 +205,10 @@ A basic movement system might be implemented with something like the following: ```c++ struct MovementSystem : public System<MovementSystem> { void update(entityx::EntityManager &es, entityx::EventManager &events, TimeDelta dt) override { - ComponentHandle<Position> position; - ComponentHandle<Direction> direction; - for (Entity entity : es.entities_with_components(position, direction)) { - position->x += direction->x * dt; - position->y += direction->y * dt; - } + es.each<Position, Direction>([dt](Position &position, Direction &direction) { + position.x += direction.x * dt; + position.y += direction.y * dt; + }); }; }; ``` diff --git a/entityx/Entity.h b/entityx/Entity.h index 493f948..9f44106 100644 --- a/entityx/Entity.h +++ b/entityx/Entity.h @@ -412,7 +412,6 @@ class EntityManager : entityx::help::NonCopyable { void next_entity(Entity &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); } @@ -429,7 +428,29 @@ class EntityManager : entityx::help::NonCopyable { ComponentMask mask_; }; - typedef BaseView<false> View; + template <bool All, typename ... Components> + class TypedView: public BaseView<All> { + public: + template <typename T> struct identity { typedef T type; }; + + void each(typename identity<std::function<void(Components&...)>>::type f) { + for (auto it : *this) + f(*(it.template component<Components>().get())...); + } + + void each(typename identity<std::function<void(Entity entity, Components&...)>>::type f) { + for (auto it : *this) + f(it, *(it.template component<Components>().get())...); + } + + private: + friend class EntityManager; + + explicit TypedView(EntityManager *manager) : BaseView<All>(manager) {} + TypedView(EntityManager *manager, ComponentMask mask) : BaseView<All>(manager, mask) {} + }; + + template <typename ... Components> using View = TypedView<false, Components...>; typedef BaseView<true> DebugView; template <typename ... Components> @@ -443,6 +464,7 @@ class EntityManager : entityx::help::NonCopyable { unpack_<0, Components...>(entity); } + private: template <int N, typename C> void unpack_(entityx::Entity &entity) const { @@ -691,9 +713,21 @@ class EntityManager : entityx::help::NonCopyable { * @endcode */ template <typename ... Components> - View entities_with_components() { + View<Components...> entities_with_components() { auto mask = component_mask<Components ...>(); - return View(this, mask); + return View<Components...>(this, mask); + } + + template <typename T> struct identity { typedef T type; }; + + template <typename ... Components> + void each(typename identity<std::function<void(Components&...)>>::type f) { + return entities_with_components<Components...>().each(f); + } + + template <typename ... Components> + void each(typename identity<std::function<void(Entity entity, Components&...)>>::type f) { + return entities_with_components<Components...>().each(f); } /** diff --git a/entityx/Entity_test.cc b/entityx/Entity_test.cc index f5667e2..a79becb 100644 --- a/entityx/Entity_test.cc +++ b/entityx/Entity_test.cc @@ -604,3 +604,40 @@ TEST_CASE_METHOD(EntityManagerFixture, "TestConstComponentsNotInstantiatedTwice" REQUIRE(b.component<const Position>()->x == 1); REQUIRE(b.component<const Position>()->y == 2); } + +TEST_CASE_METHOD(EntityManagerFixture, "TestEntityManagerEach") { + Entity a = em.create(); + a.assign<Position>(1, 2); + int count = 0; + em.each<Position>([&count](Position &position) { + count++; + REQUIRE(position.x == 1); + REQUIRE(position.y == 2); + }); + REQUIRE(count == 1); +} + +TEST_CASE_METHOD(EntityManagerFixture, "TestViewEach") { + Entity a = em.create(); + a.assign<Position>(1, 2); + int count = 0; + em.entities_with_components<Position>().each([&count](Position &position) { + count++; + REQUIRE(position.x == 1); + REQUIRE(position.y == 2); + }); + REQUIRE(count == 1); +} + +TEST_CASE_METHOD(EntityManagerFixture, "TestViewEachWithEntity") { + Entity a = em.create(); + a.assign<Position>(1, 2); + int count = 0; + em.entities_with_components<Position>().each([&count](Entity entity, Position &position) { + count++; + REQUIRE(position.x == 1); + REQUIRE(position.y == 2); + REQUIRE(*entity.component<Position>().get() == position); + }); + REQUIRE(count == 1); +} diff --git a/entityx/System_test.cc b/entityx/System_test.cc index 5810357..71ef183 100644 --- a/entityx/System_test.cc +++ b/entityx/System_test.cc @@ -43,8 +43,7 @@ class MovementSystem : public System<MovementSystem> { explicit MovementSystem(string label = "") : label(label) {} void update(EntityManager &es, EventManager &events, TimeDelta) override { - EntityManager::View entities = - es.entities_with_components<Position, Direction>(); + auto entities = es.entities_with_components<Position, Direction>(); ComponentHandle<Position> position; ComponentHandle<Direction> direction; for (auto entity : entities) { @@ -60,8 +59,7 @@ class MovementSystem : public System<MovementSystem> { class CounterSystem : public System<CounterSystem> { public: void update(EntityManager &es, EventManager &events, TimeDelta) override { - EntityManager::View entities = - es.entities_with_components<Counter>(); + auto entities = es.entities_with_components<Counter>(); Counter::Handle counter; for (auto entity : entities) { entity.unpack<Counter>(counter); diff --git a/examples/example.cc b/examples/example.cc index 7488a74..7474f7f 100644 --- a/examples/example.cc +++ b/examples/example.cc @@ -87,7 +87,7 @@ public: void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { int c = 0; ex::ComponentHandle<Collideable> collideable; - for (ex::Entity entity : es.entities_with_components<Collideable>()) c++; + es.each<Collideable>([&](Collideable&) { ++c; }); for (int i = 0; i < count - c; i++) { ex::Entity entity = es.create(); @@ -117,11 +117,10 @@ private: // Updates a body's position and rotation. struct BodySystem : public ex::System<BodySystem> { void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { - ex::ComponentHandle<Body> body; - for (ex::Entity entity : es.entities_with_components(body)) { - body->position += body->direction * static_cast<float>(dt); - body->rotation += body->rotationd * dt; - } + es.each<Body>([dt](Body &body) { + body.position += body.direction * static_cast<float>(dt); + body.rotation += body.rotationd * dt; + }); }; }; @@ -132,15 +131,14 @@ public: explicit BounceSystem(sf::RenderTarget &target) : size(target.getSize()) {} void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { - ex::ComponentHandle<Body> body; - for (ex::Entity entity : es.entities_with_components(body)) { - if (body->position.x + body->direction.x < 0 || - body->position.x + body->direction.x >= size.x) - body->direction.x = -body->direction.x; - if (body->position.y + body->direction.y < 0 || - body->position.y + body->direction.y >= size.y) - body->direction.y = -body->direction.y; - } + es.each<Body>([this](Body &body) { + if (body.position.x + body.direction.x < 0 || + body.position.x + body.direction.x >= size.x) + body.direction.x = -body.direction.x; + if (body.position.y + body.direction.y < 0 || + body.position.y + body.direction.y >= size.y) + body.direction.y = -body.direction.y; + }); } private: @@ -185,15 +183,13 @@ private: } void collect(ex::EntityManager &entities) { - ex::ComponentHandle<Body> body; - ex::ComponentHandle<Collideable> collideable; - for (ex::Entity entity : entities.entities_with_components(body, collideable)) { + entities.each<Body, Collideable>([this](ex::Entity entity, Body &body, Collideable &collideable) { unsigned int - left = static_cast<int>(body->position.x - collideable->radius) / PARTITIONS, - top = static_cast<int>(body->position.y - collideable->radius) / PARTITIONS, - right = static_cast<int>(body->position.x + collideable->radius) / PARTITIONS, - bottom = static_cast<int>(body->position.y + collideable->radius) / PARTITIONS; - Candidate candidate {body->position, collideable->radius, entity}; + left = static_cast<int>(body.position.x - collideable.radius) / PARTITIONS, + top = static_cast<int>(body.position.y - collideable.radius) / PARTITIONS, + right = static_cast<int>(body.position.x + collideable.radius) / PARTITIONS, + bottom = static_cast<int>(body.position.y + collideable.radius) / PARTITIONS; + Candidate candidate {body.position, collideable.radius, entity}; unsigned int slots[4] = { left + top * size.x, right + top * size.x, @@ -204,7 +200,7 @@ private: if (slots[0] != slots[1]) grid[slots[1]].push_back(candidate); if (slots[1] != slots[2]) grid[slots[2]].push_back(candidate); if (slots[2] != slots[3]) grid[slots[3]].push_back(candidate); - } + }); } void collide(ex::EventManager &events) { @@ -232,15 +228,14 @@ private: class ParticleSystem : public ex::System<ParticleSystem> { public: void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { - ex::ComponentHandle<Particle> particle; - for (ex::Entity entity : es.entities_with_components(particle)) { - particle->alpha -= particle->d * dt; - if (particle->alpha <= 0) { + es.each<Particle>([dt](ex::Entity entity, Particle &particle) { + particle.alpha -= particle.d * dt; + if (particle.alpha <= 0) { entity.destroy(); } else { - particle->colour.a = particle->alpha; + particle.colour.a = particle.alpha; } - } + }); } }; @@ -251,15 +246,13 @@ public: void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { sf::VertexArray vertices(sf::Quads); - ex::ComponentHandle<Particle> particle; - ex::ComponentHandle<Body> body; - for (ex::Entity entity : es.entities_with_components(body, particle)) { - float r = particle->radius; - vertices.append(sf::Vertex(body->position + sf::Vector2f(-r, -r), particle->colour)); - vertices.append(sf::Vertex(body->position + sf::Vector2f(r, -r), particle->colour)); - vertices.append(sf::Vertex(body->position + sf::Vector2f(r, r), particle->colour)); - vertices.append(sf::Vertex(body->position + sf::Vector2f(-r, r), particle->colour)); - } + es.each<Particle, Body>([&vertices](Particle &particle, Body &body) { + const float r = particle.radius; + vertices.append(sf::Vertex(body.position + sf::Vector2f(-r, -r), particle.colour)); + vertices.append(sf::Vertex(body.position + sf::Vector2f(r, -r), particle.colour)); + vertices.append(sf::Vertex(body.position + sf::Vector2f(r, r), particle.colour)); + vertices.append(sf::Vertex(body.position + sf::Vector2f(-r, r), particle.colour)); + }); target.draw(vertices); } private: @@ -331,13 +324,11 @@ public: } void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { - ex::ComponentHandle<Body> body; - ex::ComponentHandle<Renderable> renderable; - for (ex::Entity entity : es.entities_with_components(body, renderable)) { - renderable->shape->setPosition(body->position); - renderable->shape->setRotation(body->rotation); - target.draw(*renderable->shape.get()); - } + es.each<Body, Renderable>([this](Body &body, Renderable &renderable) { + renderable.shape->setPosition(body.position); + renderable.shape->setRotation(body.rotation); + target.draw(*renderable.shape.get()); + }); last_update += dt; frame_count++; if (last_update >= 0.5) { |