Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/jesstelford/pecs
Entity Component System (ECS) for PICO-8 & Picotron
https://github.com/jesstelford/pecs
entity-component-system game-development game-library lua pico-8 picotron
Last synced: 17 days ago
JSON representation
Entity Component System (ECS) for PICO-8 & Picotron
- Host: GitHub
- URL: https://github.com/jesstelford/pecs
- Owner: jesstelford
- License: mit
- Created: 2021-03-09T13:42:06.000Z (almost 4 years ago)
- Default Branch: main
- Last Pushed: 2024-07-21T18:15:16.000Z (6 months ago)
- Last Synced: 2025-01-02T17:15:30.099Z (25 days ago)
- Topics: entity-component-system, game-development, game-library, lua, pico-8, picotron
- Language: Lua
- Homepage:
- Size: 84 KB
- Stars: 61
- Watchers: 7
- Forks: 4
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE.txt
Awesome Lists containing this project
README
PECS
Entity Component System (ECS) for PICO-8 & Picotron in 567 tokens.
(Based on KatrinaKitten's excellent Tiny ECS Framework v1.1)
## Usage
Everything is part of a _World_. Create one with `pecs()`:
```lua
local world = pecs()
```_Components_ describe data containers that can be instantiated:
```lua
local Position = world.component()
local Velocity = world.component()
```An _Entity_ is a collection of instantiated _Components_.
```lua
local player = world.entity()
player += Position({ x=10, y=0 })
player += Velocity({ x=0, y=1 })
```All data within an _Entity_ can be accessed as long as you know the _Component_
it belongs to:```lua
print(player[Position].x, 10, 10, 7)
```_Systems_ allow specifying game logic (as a function) which applies to
_Entities_ that have a certain set of _Components_ (ie; a _filter_).The game logic function of a _System_ is executed once per matched _Entity_,
ensuring performance is maintained when there are many entities.
The function receives any arguments passed when calling the method. Useful for
passing in elapsed time, etc.```lua
local move = world.system({ Position, Velocity }, function(entity, ticks)
entity[Position].x += entity[Velocity].x * ticks
entity[Position].y += entity[Velocity].y * ticks
end)-- Run the system method against all matched entities
-- Any args passed will be available in the system callback function
local ticks = 1
move(ticks)
```## Example
```lua
local world = pecs()local Position = world.component()
local Velocity = world.component()local player = world.entity({ name="Jess" })
player += Position({ x=10, y=0 })
player += Velocity({ x=0, y=1 })local move = world.system({ Position, Velocity }, function(entity, ticks)
entity[Position].x += entity[Velocity].x * ticks
entity[Position].y += entity[Velocity].y * ticks
end)local lastTime = time()
function _update()
move(time() - lastTime)
lastTime = time()
endfunction _draw()
cls()
print(player[Position].x.." "..player[Position].y, 10, 10, 7)
end
```For more complete & practical examples, see the `example/` folder:
- **[`example/particles.p8`](./example/particles.p8)**: A Particle Emitter
showing how to spawn entities and adding/removing components (the type of
Emitter) on a button press.
- **[`example/camera-follow.p8`](./example/camera-follow.p8)**: A camera
follow/window technique built using Components & Systems. This example has a
visual representation of the "camera" to see the effect.## API
### `pecs() => World`
Everything in PECS happens within a world.
Can be called multiple times to create multiple worlds:
```lua
local world1 = pecs()
local world2 = pecs()
```Each world has its own _Components_ and _Entities_.
#### `World#update()`
Must be called at the start of each `_update()` before anything else.
#### `World#entity([attr[, Component, ...]]) => Entity`
```lua
local player = world.entity()local trap = world.entity({ type="spikes" })
local enemy = world.entity({}, Position({ x=10, y=10 }), Rotation({ angle=45 })
```##### Adding a Component to an Entity
```lua
player += Position({ x=100, y=20 })
```##### Removing a Component from an Entity
```lua
player -= Position
```#### `World#component([defaults]) => Component`
```lua
local Position = world.component()local Drawable = world.component({ color: 8 })
```#### `World#system(filter, callback) => Function`
Where `filter` is a table of Components, and `callback` is a function that's
passed the entity to operate on.Returns a function that when called will execute the `callback` once per Entity
that contains all the specified Components.When executing the function, any parameters are passed through to the
`callback`.```lua
local move = world.system({ Position, Velocity }, function(entity, ticks)
entity[Position].x += entity[Velocity].x * ticks
entity[Position].y += entity[Velocity].y * ticks
end)-- Run the system method against all matched entities
-- Any args passed will be available in the system callback function
local ticks = 1
move(ticks)
```Systems efficiently maintain a list of filtered entities that is only updated
when needed. It is safe to create many systems that operate over large lists of
Entities (within PICO-8's limits).#### `World#query(filter) => Table`
⚠️ _It is recommended to use `World#system` (which calls `query` internally)._
Where `filter` is a table of Components, eg; `{ Position, Velocity }`.
Return a reference to a filtered table of `Entity`s. The table is automatically
updated when new entities match or old entities no longer match. Modifying this
table can cause Systems to misbehave; treat it as read-only.```lua
local entities = world.query({ Position })
for _, ent in pairs(entities) do
printh(ent[Position].x .. "," ..ent[Position].y)
end
```Queries efficiently maintain a list of filtered entities that is only updated
when needed. It is safe to create many queries that operate over large lists of
Entities (within PICO-8's limits).#### `World#remove(Entity)`
Remove the given entity from the world.
Any Systems or Queries which previously matched this entity will no longer
operate on it.#### `World#queue(Function)`
Useful for delaying actions until the next turn of the update loop.
Particularly when the action would modify a list that's currently being iterated
on such as removing an item due to collision, or spawning new items.