Skip to content

Latest commit

 

History

History
540 lines (404 loc) · 21.8 KB

README.md

File metadata and controls

540 lines (404 loc) · 21.8 KB

ECS

Build Status codecov

Macro powered entity component system for haxe. Many of the existing ECS libraries were very verbose and / or required extending / implementing some type for all components. I wasn't a fan of this so made my own.

All components and families are resolved at compile time allowing entities and components to be stored in flat arrays. No dynamic lookup, reflection, or anything like that is used at runtime, so it should be reasonably performant.

Inspired / ideas stolen from clay_ecs and baldrick.

Requires Haxe 4.2 as well as the safety and bits library.

Quick Example

Components and resources can be plain old haxe classes as well as ints, floats, bools, and strings. No interface needs to be implemented or class extended.

package components;

class Position
{
    public var x : Float;

    public var y : Float;

    public function new()
    {
        x = 0;
        y = 0;
    }
}

class Velocity
{
    public var x : Float;

    public var y : Float;

    public function new(_x, _y)
    {
        x = _x;
        y = _y;
    }
}

Families are groups of components we are interested in, as components are added and removed from entities the families keep track of which entities currently fit the request. Systems must extend ecs.System and the easiest way to define a family is to create a variable with the @:fastFamily meta, this should be an object declaration defining the component types you want on entities and the name you want them accessible by.

package systems;

import ecs.System;
import components.Components.Position;
import components.Components.Velocity;

class VelocitySystem extends System
{
    @:fastFamily var movables : { pos : Position, vel : Velocity };

    override function update(_dt : Float)
    {
        iterate(movables, {
            pos.x += vel.x * _dt;
            pos.y += vel.y * _dt;

            trace('${ pos.x },${ pos.y }');
        });
    }
}

The iterate macro function allows you to run a block of code for each entity in a specified family. Notice that pos and vel which were defined the the movables family are being used in the iterate function. When giving a name to a component in a family definition a variable of that name with that component for the current entity will be accessible in a iterate function for that family.

To bring everything together we need to create a universe object, this will hold all entities, components, resources, and systems. Universe creation and phases are covered in detail further down in this document.

import ecs.Universe;
import systems.VelocitySystem;
import components.Components.Position;
import components.Components.Velocity;

function main()
{
    final universe = Universe.create({
        entities : 1024,
        phases   : [
            {
                name    : 'game',
                systems : [ VelocitySystem ]
            }
        ]
    });

    final object = universe.createEntity();

    universe.setComponents(object,
        Position,
        new Velocity(1, 1));

    for (_ in 0...120)
    {
        universe.update(1 / 60);
    }
}

The function setComponents is a macro which eases the process of adding multiple components and notifying the required families about changes. If a component has a constructor with no parameters you can just enter its type and the new call will be generated for you. Constructors, fields, and function calls are also allowed in the macro. If you add a component which isn't used by any family the compiler will emit a warning and that expression will be skipped.

The universe can then be ticked forward by calling the update function. Phases and systems are iterated in the order they are defined at universe creation when using universe.update.

This example can be found in the sample directory of this repository, it can be ran with haxe run.hxml, the universe will be ticked forward 120 times and the velocity system will update the entities position and print out the result along the way.

Advance Usage

Iterate

The iterate macro is the main way to execute code with each entities components in a given family, it automates the process of getting the components using the names provided when defining the family. In situations where you don't actually care about the entity itself you can use the following syntax.

iterate(someFamily, {
    // code here is ran for each entity found in `someFamily`.
});

Alternativly lambda function syntax can be used.

iterate(someFamily, () -> {
    // code here is ran for each entity found in `someFamily`.
});

If you do need to access the entity whos components are currently being accessed then you can use lambda function syntax with a single parameter which will then be accessible in the block and contain the current entity.

iterate(someFamily, entity -> {
    // `entity` is the entity which has the components currently being accessed.
});

⚠️ While the last two of these iterate examples used lambda function syntax no function is actually generate or called! All iterate macro functions generate a for loop so passing a real function into iterate is not valid.

It is also valid to nest iterate calls, just make sure that families do not have any component name collisions.

In the below example both the bullets and enemies families request the Position and BBox component, but because we give them different names (bulletPos vs enemyPos and bulletBox vs enemyBox) we can safely nest iterations without name collisions.

class BulletCollisionSystem extends System
{
    @:fastFamily bullets : { bulletPos : Position, bulletBox : BBox, bullet : Bullet };

    @:fastFamily enemies : { enemyPos : Position, enemyBox : BBox, enemy : Enemy };

    override function update(_dt : Float)
    {
        iterate(bullets, {
            iterate(enemies, {
                // check bounding box collision between enemies and bullets.
            });
        });
    }
} 

Resources

Resources are components which are attached to the universe instead of entities. They can be required by families and are very useful for data which could be considered "singleton" in nature (e.g. current level data).

class MySystem extends System
{
    @:fullFamily var myFamily : {
        requires : { comp : SomeComp },
        resources : { myRes : MyResourceType }
    };

    override function update(_dt : Float)
    {
        setup(myFamily, {
            // myRes is avaialble in this block.

            iterate(myFamily, {
                // do stuff with the components and resources.
            });
        });
    }
}

Using the fullFamily meta instead of fastFamily allows us to define resources which are required for the family to run. In the same way that iterate automatically creates variables for components setup create variables for any resources in that family.

If the resources requested by a family are not currently all in the universe then the code block passed to setup will not be ran.

Fetch

While the iterate macro provides a safe way to operate on every entity in a family, sometimes you may want to just operate on a single, know entity. In this case fetch operates in much the same way except that it takes an entity as an argument as well as the family. If the provided entity is not part of that family then the code in the final block expression is not executed.

fetch(someFamily, someEntity, {
    // code here is ran if `someEntity` is currently within `someFamily`
});

This function sets up variables based on the family component names just like iterate and is perfect for performing setup or teardown code when an entity is added or removed from a family.

class SpriteSystem extends System
{
    @:fullFamily var sprites : {
        requires : { sprite : Sprite },
        resources : { scene : GameScene }
    };

    override function onAdded()
    {
        sprites.onEntityAdded.subscribe(entity -> {
            setup(sprites, {
                fetch(sprites, entity, {
                    scene.addSprite(sprite);
                });
            });
        });

        sprites.onEntityRemoved.subscribe(entity -> {
            setup(sprites, {
                fetch(sprites, entity, {
                    scene.removeSprite(sprite);
                });
            });
        });
    }
}

Family Definition

FastFamily

The fastFamily meta provides an easy way to define a family which only requires components. Variables tagged with this meta must then be assigned an anonymous object type. Using object declaration local variable generation can be skipped on a per-component basis by using _ as the name.

ℹ️ fastFamily does not provide any runtime speed increases over fullFamily, the fast comes from the fact that its faster to type when your family only needs components.

class MySystem extends System
{
    @:fastFamily myFamily1 : { pos : Position, vel : Velocity, _ : Sprite };

    override function update(_dt : Float)
    {
        iterate(myFamily1, {
            // `pos` and `vel` are two local variables accessible in this block.
            // No local variable for the `Sprite` component will be generated.
        });
    }
}

FullFamily

The fullFamily meta allows defining families which require both components and resources. Variables tagged with this meta must be an object declaration which has a requires and resources field. The requires field is for defining what components are needed and follows all the same rules outlined in the above fastFamily section.

class MySystem extends System
{
    @:fastFamily myFamily : {
        requires : { pos : Position, vel : Velocity, _ : Sprite },
        resources : { myRes : MyResource }
    };

    override function update(_dt : Float)
    {
        setup(myFamily, {
            // This block will only be executed if `MyResource` is attached to the universe.
            // a local variable `myRes` will also be accessible here.

            iterate(myFamily, {
                // `pos` and `vel` are two local variables accessible in this block.
                // No local variable for the `Sprite` component will be generated.
            });
        });
    }
}

Family Activation

You may find yourself wanting to run pre and post iterate code for the family as a whole in the update function. The following shows an example of this.

class SpriteDrawerSystem extends System
{
    @:fullFamily var sprites : {
        requires : { pos : Position, origin : Origin, spr : Sprite },
        resources : { painter : Painter }
    }

    override function update(_dt : Float)
    {
        setup(sprites, {
            painter.begin();

            iterate(sprites, {
                painter.drawSprite(
                    spr.id,
                    pos.x,
                    pos.y,
                    origin.x,
                    origin.y);
            });

            painter.end();
        });
    }
}

The above is an example of what a drawing system might look like and how it would interface with some imaginary game engine. Here the Painter resource is some object from the game engine which allows efficient drawing through batching, but in order to do that you need to make a begin and end call.

The code within the setup block will only be ran if all the resources requested by the family we're setting up are currently attached to the systems universe. This macro also handles creating local variables with the names specified in the family definition so we can safely access our resources. We can also then iterate over that same family allowing us to run pre and post iterate code.

Phases

Phases are logical collections of systems, phases are defined at universe creation and cannot be modified afterwards. While they can't be modified phases and individual systems within can be enabled or disabled at will.

final universe = Universe.create({
    entities : 1024,
    phases : [
        {
            name : 'game-logic',
            systems : [
                KeyboardMovementSystem,
                GravitySystem,
                VelocitySystem,
                CollisionDetectionSystem
            ]
        },
        {
            name : 'rendering',
            systems : [
                SpriteDrawingSystem
            ]
        }
    ]
})

In the above example our universe is created with two phases, calling universe.update will then update all enabled phases and systems in the order defined in the universe. If you want more control over when phases are updated (e.g. you want a rendering or network phase to update at a different rate from the game simulation) you can use the universe.getPhase function to retrieve a phase based on its name.

final phase = universe.getPhase('game-logic');

phase.update(1 / 30);

Using the returned phase you can update, enable, and disable the system. Calling update on a phase which has been disabled will result in no systems being updated.

You can also fetch specific systems from a phase with the phase.getSystem function. This function takes in the type of the system you want from the phase.

final system = phase.getSystem(VelocitySystem);

The phase.enableSystem and phase.disableSystem also take in the type of the specific system to enable or disable.

phase.disableSystem(VelocitySystem);
phase.enableSystem(VelocitySystem);

ℹ️ Information about disabled systems is preserved when disabling an entire phase. If you specifically disable a system then disable the phase its in and then at a late time re-enable the phase, that system which was specifically disabled won't be enabled.

⚠️ The getPhase, getSystem, enableSystem, and disableSystem are runtime functions and will throw an exception the specific phase or system is not found. These also search through all phases and systems performing type checks, so the performance characteristics of these functions may not be great. These may be changed to macro functions in the future, but for now I recommend pre-fetching any specific phases or systems up front and keep your own references to them.

OnEnabled and OnDisabled

The ecs.System type contains onEnabled and onDisabled functions which can be overridden to add custom code for when a system is enabled or disabled in a phase. It is also perfectly safe to access families, components, and resource from whithin these functions.

class MySystem extends System
{
    @:fastFamily var myFamily : { _ : SomeComponent };

    override function onEnabled()
    {
        trace(myFamily.isActive());
    }

    override function onDisabled()
    {
        trace(myFamily.isActive());
    }
}

OnEntityAdded and OnEntityRemoved

Families also expose two signals you can subscribe to for when entities are added and removed from the family.

class MySystem extends System
{
    @:fastFamily var myFamily : { _ : SomeComponent };

    override function onAdded()
    {
        myFamily.onEntityAdded.subscribe(added);
        myFamily.onEntityRemoved.subscribe(removed);
    }

    function added(_entity)
    {
        trace('${ _entity } was added to myFamily');
    }

    function removed(_entity)
    {
        trace('${ _entity } was removed from myFamily');
    }
}

All components and resources requested by a family are guarenteed to still be accessible from within onEntityAdded and onEntityRemoved subscribers.

ℹ️ If a family has 10 entities in it and a resource it requires is removed from the universe all subscribers to that families onEntityRemoved signal will recieve 10 notifications, one for each entity being removed. In the same fashion when a resource is added any subscribers to a family which now has entities will recieve a notification for each entity added.

Defines

ecs.invalidationFile

For better compatibility with the haxe compilation server a file is used which allows systems to trigger the core ecs classes for wiping from the compilation cache. By default the folder of the output file is used. The compilation server does not define an output file so the invalidation file is placed in the projects root folder.

If you want to place the invalidation file else where this define allows you to specify the location. -D ecs.invalidationFile=my/custom/path/.ecs_invalidation

ecs.no_debug_output

When the --debug flag is used ecs related debug information is printed to stdout, if you do not want this output in debug mode this define will stop it from being output.

Implementation Details

The ecs.Entity type is an abstract around an int, -1 is a special reserved integer ID representing none / a null entity.

Each unique component and resource across the entire compiled project is given an ID by hashing its ComplexType, IDs start at 0 and increment each time a new component or resource is found. Components and resources both have their own ID lists.

For each component a ecs.Components<T> object is created at universe creation, these objects contain a haxe.ds.Vector to store the components, this vector is pre-allocated with enough space to store a component of its type for each entity.
Since entities are just integer abstracts they are used to index into these vectors when fetching a component.
The ecs.ComponentManager has a haxe.ds.Vector containing all of these ecs.Components<T> objects, indexing into this vector with the ID given to a component during compilation will get you the ecs.Components<T> object for that component type. It also stores a haxe.ds.Vector of bit flags, one for every possible entity in the universe. These bit flags track which components an entity currently has. When a component is added or removed from an entity the bits are changed accordingly, the unique ID for each component is also used to set, unset, and check these bits.

Resources are handled in a similar way, ecs.ResourceManager keeps a haxe.ds.Vector allocated to the number of unique resources found during family compilation. It also keeps a bit flag, but just the one as resources are not stored per entity, but per universe.

For each family defined a corresponding ecs.Family object is created. This object contains a sparse set which holds all entities currently in this family. It also contains a components and resource bit flag mask based on the components and resources requested, these masks are used for checking against the component masks in ecs.ComponentManager and resource mask in ecs.ResourceManager to see which entities fit the the family definition.

Currently even when a family doesn't have its required resources it keeps track of entities which fit the requested components in its sparse set, then, as and when resources are added and removed it notifies any subscribers whithout going through the expensive process or checking all entities to clear and repopulate the sparse set.

Systems have field variables injected into them based on all families defined, for each defined family in a system an ecs.Family variable is added which is assigned when the system is added to the universe. For each unique component in a system a variable of ecs.Components<T> is added for the component type, this is also assigned when the system is added to the universe. These ecs.Components<T> variables are named table$t where $t is the type name of that component. Below is an example system before and after macro code generation.

class MySystem extends System
{
    @:fastFamily var myFamily : { pos : Position, vel : Velocity };

    @:fastFamily var justPos : { pos : Position };
}
class MySystem extends System
{
    var myFamily : ecs.Family;
    var justPos : ecs.Family;

    var tablePosition : ecs.Components<Position>;
    var tableVelocity : ecs.Components<Velocity>;

    override function onAdded()
    {
        myFamily = families.get(0);
        justPos  = families.get(1);

        tablePosition = components.get(0);
        tableVelocity = components.get(1);
    }
}

Since this is all handled in macros the index / component ID is generated at compile time.

ℹ️ If you override the onAdded function these get calls are guarenteed to be injected before any of your code so will be accessible from it.

When calling iterate it generates a for loop with local variables based on the defined components in that family. So a complete before and after of the VelocitySystem shown at the top of this README and in the sample would look like this.

class VelocitySystem extends System
{
    @:fastFamily var movables : { pos : Position, vel : Velocity };

    override function update(_dt : Float)
    {
        iterate(movables, {
            pos.x += vel.x * _dt;
            pos.y += vel.y * _dt;

            trace('${ pos.x },${ pos.y }');
        });
    }
}
class VelocitySystem extends System
{
    var movables : ecs.Family;

    var tablePosition : ecs.Components<Position>;
    var tableVelocity : ecs.Components<Velocity>;

    override function onUpdate()
    {
        movables = families.get(0);

        tablePosition = components.get(0);
        tableVelocity = components.get(1);
    }

    override function update(_dt : Float)
    {
        for (_tmpEnt in movables)
        {
            final pos = tablePosition.get(_tmpEnt);
            final vel = tableVelocity.get(_tmpEnt);

            pos.x += vel.x * _dt;
            pos.y += vel.y * _dt;

            trace('${ pos.x },${ pos.y }');
        }
    }
}