An ECS (Entity Component System) implementation written in TypeScript.
npm i cat-herder
There is a umd build available:
<script src="https://unpkg.com/cat-herder@latest/dist/cat-herder.umd.js"></script>
import { World, Entity } from 'cat-herder';
// Component factories are of form (...args: any) => Record<string, any>
const Name = (name: string) => ({ name });
const Velocity = (vx: number, vy: number) => ({ vx, vy });
const world = World({});
world
.register(Name)
.register(Velocity);
// Entities
const bob = world.entity()
.with(Name)("Bob")
.build();
const roger = world.entity()
.with(Name)("Roger")
.with(Velocity)(0, 1)
.build();
// Querying
for (const [name] of world.query(Name).result()) {
console.log(name.name); // "Bob", "Roger"
}
for (const [name] of world.query(Name).not(Velocity).result()) {
console.log(name.name); // "Bob"
}
for (const [name, velocity] of world.query(Name, Velocity).result()) {
console.log(name.name); // "Roger"
console.log(velocity); // { vx: 0, vy: 1 }
}
World expects initial shared resources to be T.
T defaults to Record<string, any>
.
interface IResources {
time: number,
someOtherResource?: ISomeOtherResource,
}
const world = World<IResources>({ time: Date.now() });
Creates an entity with attached components.
const bob = world
.entity()
.with(Name)("Bob")
.with(Position)(10, 10)
.with(Velocity)(1, 2)
.build();
Removes entity and any attached components.
world.delete(bob);
Makes component available to attach to entities.
Component factories must have form (...args: any) => Record<string, any>
Attempting to attach an unregistered component will throw an error.
Support for class-based components is on the roadmap.
const Name = (name: string) => ({ name });
world.register(Name);
world
.entity()
.with(Name)("Bob") // will give appropriate type hints with TS
.build();
Attaches component to entity.
Attempting to attach an unregistered component will throw an error.
Attempting to attach to an unknown entity will throw an error.
const myEntity = world.entity().build();
world.add(Name, myEntity)("Roger");
Returns given component for entity. Will return null if not found.
Attempting to retrieve an unregistered component will throw an error.
Attempting to retrieve from an unknown entity will throw an error.
world.get(Name, myEntity);
Removes component from entity.
Attempting to remove an unregistered component will throw an error.
Attempting to remove from an unknown entity will throw an error.
world.remove(Name, myEntity);
Returns results for all entities with matching components.
Will exclude any entities with components in the not
clause.
You can retrieve entity by using the Entity
component.
import { Entity, World } from 'cat-herder';
const Name = (name: string) => ({ name });
const Position = (vx: number, vy: number) => ({ vx, vy });
const Dead = () => ({}); // Tag component (doesn't contain any actual data)
const world = World({})
.register(Name)
.register(Position)
.register(Dead);
const mario = world
.entity()
.with(Name)("Mario")
.with(Position)(10, 10)
.build();
const luigi = world
.entity()
.with(Name)("Luigi")
.with(Position)(5, 10)
.build();
const toad = world
.entity()
.with(Name)("Toad")
.with(Position)(15, 10)
.build();
// oops, Luigi had an accident
world.add(Dead, luigi);
for (const [entity, name, pos] of world.query(Entity, Name, Position).not(Dead).result()) {
// Contains valid type hints in TS
console.log(name.name); // Mario ; Toad
console.log(`${pos.x}, ${pos.y}`); // 10, 10 ; 15 , 10
}
Experimental version of query which doesn't require the .result()
call before iteration.
It accesses data lazily, so you can terminate lookups on demand.
Must call .collect()
to access as array.
Will throw if .collect()
or .not()
are called after iteration has started.
for (const [life] of world.query_iter(Life, Ally)) {
if (life <= 0) {
world.resources.game_over = true;
break;
}
}
// to access as array
const allies_count = world.query_iter(Ally).collect().length;
Registers a function which should run each game tick.
They occur in the order registered.
// system setup
import { System, Resource } from "cat-herder";
function movementSystem(world: IWorld) {
const { time } = world.resources;
for (
const [pos, vel] of
world.query(Position, Velocity).result()
) {
pos.x += vel.vx * time.elapsed;
pos.y += vel.vy * time.elapsed;
}
}
// init
world.system(movementSystem);
// each game tick
world.update();
Trigger all registered systems.
world.update();
Initial state should be passed in to the World on creation.
Use it as a dictionary of shared resources.
const world = World({
time: Date.now(),
delta: 1,
})
world.system(world => {
const time = Date.now();
const delta = (time - world.resources.time) * 60 / 1000;
world.resources.time = time;
world.resources.delta = delta;
});
For now, cat-herder requires BitSet.
https://github.com/schtauffen/ts-ecs
This is an early project and breaking changes are to be expected.