From c4d518bbdd136dde3826ff2e0e51ae84d7613720 Mon Sep 17 00:00:00 2001 From: Alec Thomas Date: Sat, 18 Jul 2015 21:15:23 -0400 Subject: Add `.each([](C &c...) {})` iteration. May break code that uses `EntityManager::View`, as this is now a template type `EntityManager::View`. Fixes #62. --- README.md | 28 ++++++++++++----- entityx/Entity.h | 42 ++++++++++++++++++++++--- entityx/Entity_test.cc | 37 ++++++++++++++++++++++ entityx/System_test.cc | 6 ++-- examples/example.cc | 83 ++++++++++++++++++++++---------------------------- 5 files changed, 135 insertions(+), 61 deletions(-) diff --git a/README.md b/README.md index 7807f23..b450c18 100644 --- a/README.md +++ b/README.md @@ -143,7 +143,23 @@ entity.assign(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([](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; @@ -189,12 +205,10 @@ A basic movement system might be implemented with something like the following: ```c++ struct MovementSystem : public System { void update(entityx::EntityManager &es, entityx::EventManager &events, TimeDelta dt) override { - ComponentHandle position; - ComponentHandle direction; - for (Entity entity : es.entities_with_components(position, direction)) { - position->x += direction->x * dt; - position->y += direction->y * dt; - } + es.each([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 View; + template + class TypedView: public BaseView { + public: + template struct identity { typedef T type; }; + + void each(typename identity>::type f) { + for (auto it : *this) + f(*(it.template component().get())...); + } + + 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 @@ -443,6 +464,7 @@ class EntityManager : entityx::help::NonCopyable { unpack_<0, Components...>(entity); } + private: template void unpack_(entityx::Entity &entity) const { @@ -691,9 +713,21 @@ class EntityManager : entityx::help::NonCopyable { * @endcode */ template - View entities_with_components() { + View entities_with_components() { auto mask = component_mask(); - return View(this, mask); + return View(this, mask); + } + + template struct identity { typedef T type; }; + + template + void each(typename identity>::type f) { + return entities_with_components().each(f); + } + + template + void each(typename identity>::type f) { + return entities_with_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()->x == 1); REQUIRE(b.component()->y == 2); } + +TEST_CASE_METHOD(EntityManagerFixture, "TestEntityManagerEach") { + Entity a = em.create(); + a.assign(1, 2); + int count = 0; + em.each([&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(1, 2); + int count = 0; + em.entities_with_components().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(1, 2); + int count = 0; + em.entities_with_components().each([&count](Entity entity, Position &position) { + count++; + REQUIRE(position.x == 1); + REQUIRE(position.y == 2); + REQUIRE(*entity.component().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 { explicit MovementSystem(string label = "") : label(label) {} void update(EntityManager &es, EventManager &events, TimeDelta) override { - EntityManager::View entities = - es.entities_with_components(); + auto entities = es.entities_with_components(); ComponentHandle position; ComponentHandle direction; for (auto entity : entities) { @@ -60,8 +59,7 @@ class MovementSystem : public System { class CounterSystem : public System { public: void update(EntityManager &es, EventManager &events, TimeDelta) override { - EntityManager::View entities = - es.entities_with_components(); + auto entities = es.entities_with_components(); Counter::Handle counter; for (auto entity : entities) { entity.unpack(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; - for (ex::Entity entity : es.entities_with_components()) c++; + es.each([&](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 { void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { - ex::ComponentHandle body; - for (ex::Entity entity : es.entities_with_components(body)) { - body->position += body->direction * static_cast(dt); - body->rotation += body->rotationd * dt; - } + es.each([dt](Body &body) { + body.position += body.direction * static_cast(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; - 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([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; - ex::ComponentHandle collideable; - for (ex::Entity entity : entities.entities_with_components(body, collideable)) { + entities.each([this](ex::Entity entity, Body &body, Collideable &collideable) { unsigned int - left = static_cast(body->position.x - collideable->radius) / PARTITIONS, - top = static_cast(body->position.y - collideable->radius) / PARTITIONS, - right = static_cast(body->position.x + collideable->radius) / PARTITIONS, - bottom = static_cast(body->position.y + collideable->radius) / PARTITIONS; - Candidate candidate {body->position, collideable->radius, entity}; + left = static_cast(body.position.x - collideable.radius) / PARTITIONS, + top = static_cast(body.position.y - collideable.radius) / PARTITIONS, + right = static_cast(body.position.x + collideable.radius) / PARTITIONS, + bottom = static_cast(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 { public: void update(ex::EntityManager &es, ex::EventManager &events, ex::TimeDelta dt) override { - ex::ComponentHandle particle; - for (ex::Entity entity : es.entities_with_components(particle)) { - particle->alpha -= particle->d * dt; - if (particle->alpha <= 0) { + es.each([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; - ex::ComponentHandle 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([&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; - ex::ComponentHandle 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([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) { -- cgit v1.2.3