Flax is a performant and easy to use Entity Component System.
The world is organized by simple identifiers known as an Entity
, which can have any number of components attached to them.
```rust
// Declare static components
component! {
health: f32,
regen: f32,
pos: (f32, f32),
player: (),
items: Vec
let mut world = World::new();
// Spawn an entity let p = EntityBuilder::new() .set(health(), 50.0) .tag(player()) .set(pos(), (0.0, 0.0)) .set(regen(), 1.0) .set_default(items()) .spawn(&mut world);
let mut query = Query::new((health().mutable(), regen()));
// Apply health regen for all match entites for (health, regen) in query.iter(&world) { health = (health + regen).min(100.0); }
```
Compared to other ecs implementations, a component is simply another Entity
identifier to which data is attached. This means the same "type" can be added to
an entity multiple times.
A limitation of existing implementations such as specs, planck, or hecs is that newtype wrappers need to be created to allow components of the same inner type to coexist.
This leads to having to forward all trait implementations trough e.g
derive-more
or dereferencing the newtypes during usage.
This can lead to situations such as this:
```rust
let vel = world.get::
pos = Position(pos + *vel * dt); ```
Instead of this:
```rust let vel = world.get(velocity(), entity); let mut pos = world.get_mut(position(), entity); let dt = 0.1;
*pos = *pos + *vel; ```
On a further note, since the components have to be declared beforehand (not
always true, more on that later), it limits the amount of types which can be
inserted as components. This fixes subtle bugs which come by having the type
dictate the component, such as inserting an Arc<Type>
instead of just Type
,
which leads to subsequent systems not finding the Type
on the entity.
Having statically declared componenents makes the rust type system disallow these cases and catches these bugs earlier.
During development of a game in school I used the hecs ECS. It is an awesome library, and the author Ralith has been awesome in bringing some pull requests in.
Despite this, I often made subtle bugs with similar. The game engine was
cluttered with gigantic newtypes for Velocity
, Position
with many deref
coercions in order to coexist.
This library makes use of unsafe for type erasure and the allocation in storage of ComponentBuffers and Archetypes.
As such, there are tests covering most, if not all of the unsafe parts of the code, both directly through unit tests and indirectly using integration tests.