Systems
Core behavior in Excalibur is implemented by Systems. Systems process all entities that have matching component types and perform some action.
Examples of systems are the GraphicsSystem, MotionSystem, CollisionSystem, and DebugSystem
System
Each Excalibur System loops through all entities in System.priority
order. There are a list of built in system priorities SystemPriority that can be used to assign priority. Lower number means higher priority which runs first.
typescript
/*** Higher priorities run earlier than others in the system update*/export const SystemPriority = {Highest: -Infinity,Higher: -5,Average: 0,Lower: 5,Lowest: Infinity} as const;
typescript
/*** Higher priorities run earlier than others in the system update*/export const SystemPriority = {Highest: -Infinity,Higher: -5,Average: 0,Lower: 5,Lowest: Infinity} as const;
System Types
There are two system types update and draw.
SystemType.Update type systems all run before SystemType.Draw type systems.
Lifecycle
Systems execute in this order
- one-time
constructor(world: World)
- Initialize your query here
- one-time
initialize(world: World, scene: Scene)
- Initialize anything that relies on the scene/engine here
- every frame
preupdate(scene: Scene, elapsedMs: number)
- Run setup code for the update
- every frame
update(delta: number)
- Update all your entities here
- every frame
postupdate(scene: Scene, elapsedMs: number)
- Run cleanup code for the update
Built in Systems
Excalibur has a few built in systems that are used to enable the default behavior. These Systems
are defined at Scene constructor time.
MotionSystem
The motion system implements motion on entities, like Actors moving with velocity and acceleration.
This system makes use of the TransformComponent and MotionComponent to implement motion.
If a BodyComponent is present that will be used to apply sleep or global acceleration Physics.acc to all CollisionType.Active bodies.
CollisionSystem
The collision system uses the TransformComponent, MotionComponent, and ColliderComponent to implement collision detection and resolution behavior.
GraphicsSystem
The Excalibur GraphicsSystem takes any entity with a TransformComponent and a GraphicsComponent and draws it to the screen using the ExcaliburGraphicsContext.
DebugSystem
The debug system is slightly odd, it operates on all entities with a TransformComponent to display debug information to the screen when Engine.showDebug is enabled.
Read more about the debug options here Debug
Implementing your own Components & Systems
To build your own component, extend the Excalibur Component abstract class and pick a unique type name (duplicate type names will cause problems).
For custom component types it is recommended you prefix your types, like type = 'myCustomPrefixTransform'
In this example, we create a "search" type component, that searches for a target position. Notice how this component implementation is mostly data.
typescript
class SearchComponent extends ex.Component {constructor(public target: ex.Vector) {super();}}class SearchSystem extends ex.System {query: Query<typeof SearchComponent>;constructor(world: World) {this.query = world.query([TransformComponent, SearchComponent]);}// Lower numbers mean higher priority// 99 is low prioritypublic priority = 99;// Run this system in the "update" phasepublic systemType = ex.SystemType.Updateprivate _searchSpeed = 100 // pixels/secpublic update(delta: number) {for (let entity of this.query.entities) {const target = entity.get(SearchComponent).target;const transform = entity.get(ex.TransformComponent);const direction = target.sub(transform.pos);const motion = direction.normalize().scale(this._searchSpeed);// Moves these entities towards the target at 10 pixels per secondtransform.pos = transform.pos.add(motion.scale(delta / 1000))}}}const scene = new ex.Scene();scene.world.add(new SearchSystem());// Actors come with batteries included built in featuresconst actor = new ex.Actor({pos: ex.vec(100, 100),width: 30,height: 30,color: ex.Color.Red});actor.addComponent(new SearchComponent(ex.vec(600, 400)));
typescript
class SearchComponent extends ex.Component {constructor(public target: ex.Vector) {super();}}class SearchSystem extends ex.System {query: Query<typeof SearchComponent>;constructor(world: World) {this.query = world.query([TransformComponent, SearchComponent]);}// Lower numbers mean higher priority// 99 is low prioritypublic priority = 99;// Run this system in the "update" phasepublic systemType = ex.SystemType.Updateprivate _searchSpeed = 100 // pixels/secpublic update(delta: number) {for (let entity of this.query.entities) {const target = entity.get(SearchComponent).target;const transform = entity.get(ex.TransformComponent);const direction = target.sub(transform.pos);const motion = direction.normalize().scale(this._searchSpeed);// Moves these entities towards the target at 10 pixels per secondtransform.pos = transform.pos.add(motion.scale(delta / 1000))}}}const scene = new ex.Scene();scene.world.add(new SearchSystem());// Actors come with batteries included built in featuresconst actor = new ex.Actor({pos: ex.vec(100, 100),width: 30,height: 30,color: ex.Color.Red});actor.addComponent(new SearchComponent(ex.vec(600, 400)));
Any entity that has the new component attached will be processed by the new system once added to the world!