Events are messages sent used to notify about state changes of an Entity.
These events can be consumed in two different ways.
Process events directly by an event handler subscribed to an event like entity.OnComponentChanged.
Or by recording all events using an EventRecorder and process recorded events later within a Query.
Entity changes
If changing an entity by adding or removing components, tags, scripts or child entities events are emitted.
An application can subscribe to these events like shown in the example.
Emitting these type of events increase code decoupling.
Without events these modifications need to be notified by direct method calls.
The build-in events can be subscribed on EntityStore and on Entity level like shown in the example below.
public static void AddEventHandlers()
{
var store = new EntityStore();
var entity = store.CreateEntity();
entity.OnComponentChanged += ev => { Console.WriteLine(ev); }; // > entity: 1 - event > Add Component: [MyComponent]
entity.OnTagsChanged += ev => { Console.WriteLine(ev); }; // > entity: 1 - event > Add Tags: [#MyTag1]
entity.OnScriptChanged += ev => { Console.WriteLine(ev); }; // > entity: 1 - event > Add Script: [*MyScript]
entity.OnChildEntitiesChanged += ev => { Console.WriteLine(ev); }; // > entity: 1 - event > Add Child[0] = 2
entity.AddComponent(new MyComponent());
entity.AddTag<MyTag1>();
entity.AddScript(new MyScript());
entity.AddChild(store.CreateEntity());
}
Component events
Event handlers for component changes notify one of the following ev.Action
Add - a component was newly added to the entity.
Update - the component value changed.
Remove - the component was removed.
In case of Update or Remove the handler provide access to the old component value.
public static void ComponentEvents()
{
var store = new EntityStore();
var entity = store.CreateEntity();
entity.OnComponentChanged += ev =>
{
if (ev.Type == typeof(EntityName)) {
string log = ev.Action switch
{
Add => $"new: {ev.Component<EntityName>()}",
Update => $"new: {ev.Component<EntityName>()} old: {ev.OldComponent<EntityName>()}",
Remove => $"old: {ev.OldComponent<EntityName>()}",
_ => null
};
Console.WriteLine($"entity: {ev.Entity.Id} - {ev.Action} {log}");
}
};
entity.AddComponent(new EntityName("Peter"));
entity.AddComponent(new EntityName("Paul"));
entity.RemoveComponent<EntityName>();
}
An EventRecorder record all component and tag changes of an EntityStore when Enabled.
A recorder is required for queries using EventFilter's.
To clear all recorded events use store.EventRecorder.ClearEvents().
In a game loop this is typically performed at the beginning of a new frame.
To stop recording events entirely use store.EventRecorder.Enabled = false.
EventFilter
The intended use-case for EventFilter's are queries.
When iterating the entities of a query result it can be checked if an entity was changed by an operation matching the specified filters.
E.g a specific component or tag was added.
EventFilter's can be used on its own or within a query, see the examples below.
All events that need to be filtered - like added/removed components/tags - can be added to the EventFilter.
public static void FilterEntityEvents()
{
var store = new EntityStore();
store.EventRecorder.Enabled = true; // required for EventFilter
var entity1 = store.CreateEntity(1);
var entity2 = store.CreateEntity(2);
var entity3 = store.CreateEntity(3);
entity1.AddComponent(new Position());
entity2.AddTag<MyTag1>();
var query = store.Query();
query.EventFilter.ComponentAdded<Position>();
query.EventFilter.TagAdded<MyTag1>();
foreach (var entity in store.Entities) {
bool hasEvent = query.HasEvent(entity.Id);
Console.WriteLine($"{entity,-20} hasEvent: {hasEvent}");
}
// id: 3 [] hasEvent: False
// id: 1 [Position] hasEvent: True
// id: 2 [#MyTag1] hasEvent: True
store.EventRecorder.ClearEvents(); // typically called on new frame
foreach (var entity in store.Entities) {
bool hasEvent = query.HasEvent(entity.Id);
Console.WriteLine($"{entity,-20} hasEvent: {hasEvent}");
}
// id: 3 [] hasEvent: False
// id: 1 [Position] hasEvent: False
// id: 2 [#MyTag1] hasEvent: False
}
Signals
Signals are similar to events.
They are used to send and subscribecustom events on entity level in an application.
To prevent heap allocations signal types must be structs.
The use of signals is intended for scenarios when something happens occasionally.
This avoids the need to check a state every frame to detect a specific condition.
For example signals could be used to react on collisions between entities.
Signal handlers can be added as shown below and removed if needed.
var handler = entity.AddSignalHandler<MyEvent2>(signal => { ... });
entity.RemoveSignalHandler(handler);
Multiple signal handlers can be added to an entity and are automatically removed if the entity is deleted.
public struct CollisionSignal {
public Entity other;
}
public static void AddSignalHandler()
{
var store = new EntityStore();
var player = store.CreateEntity(1);
player.AddSignalHandler<CollisionSignal>(signal => {
Console.WriteLine($"collision signal - entity: {signal.Entity.Id} other: {signal.Event.other.Id}");
});
var npc = store.CreateEntity(2);
// ... detect collision. e.g. with a collision system. In case of collision:
player.EmitSignal(new CollisionSignal{ other = npc });
}
Log Output
collision signal - entity: 1 other: 2
Note Avoid emitting signals inside a query loop.
When using a query loop to detect collisions signals should not be emitted directly.
The the event handler may perform a structural change - e.g removing or adding a components.
Doing this will invalidate the query loop.
To avoid this detected collisions can be stored inside the query loop in a List<Event, CollisionSignal>.
After the collision loop finishes collision events can be emitted and are allowed to perform structural changes.