From 4b17cad7d4b837b02108f2fb296a344e2fce971e Mon Sep 17 00:00:00 2001 From: marked Date: Thu, 16 Jan 2025 20:44:09 +0100 Subject: [PATCH] fix: Make the pull script work with jecs 0.5.x, chore: Pull latest --- .luaurc | 6 +- .lune/pull.luau | 254 ++++++------- CHANGELOG.md | 204 +++++++++++ README.md | 129 ++++--- init.luau | 949 ++++++++++++++++++++++++++++++++++++------------ pesde.toml | 5 +- rokit.toml | 2 +- stylua.toml | 10 + 8 files changed, 1123 insertions(+), 436 deletions(-) create mode 100644 CHANGELOG.md create mode 100644 stylua.toml diff --git a/.luaurc b/.luaurc index 73598d2..07221f7 100644 --- a/.luaurc +++ b/.luaurc @@ -1,8 +1,8 @@ { "aliases": { - "jecs": "src", - "testkit": "testkit", + "jecs": "jecs", + "testkit": "test/testkit", "mirror": "mirror" }, "languageMode": "strict" -} \ No newline at end of file +} diff --git a/.lune/pull.luau b/.lune/pull.luau index 5c3a287..ff9973c 100644 --- a/.lune/pull.luau +++ b/.lune/pull.luau @@ -1,109 +1,109 @@ --!strict -local process = require("@lune/process") local fs = require("@lune/fs") local net = require("@lune/net") +local process = require("@lune/process") local serde = require("@lune/serde") print("-- 🟧 Starting pull of latest release...") type wally_manifest = { - package: { - name: string, - version: string, - registry: string, - realm: string, - license: string?, - exclude: { string }?, - include: { string }?, - }, - dependencies: { - [string]: string, - }, + package: { + name: string, + version: string, + registry: string, + realm: string, + license: string?, + exclude: { string }?, + include: { string }?, + }, + dependencies: { + [string]: string, + }, } type environment = "luau" | "lune" | "roblox" | "roblox_server" type pesde_dependency = { - name: string, - version: string, - index: string?, - target: environment?, + name: string, + version: string, + index: string?, + target: environment?, } | { - wally: string, - version: string, - index: string?, + wally: string, + version: string, + index: string?, } | { - repo: string, - rev: string, - path: string?, + repo: string, + rev: string, + path: string?, } type pesde_manifest = { - name: string, - version: string, - description: string?, - license: string?, - authors: { string }?, - repository: string?, - private: boolean?, - includes: { string }?, - pesde_version: string?, - workspace_members: { string }?, + name: string, + version: string, + description: string?, + license: string?, + authors: { string }?, + repository: string?, + private: boolean?, + includes: { string }?, + pesde_version: string?, + workspace_members: { string }?, - target: { - environment: environment, - lib: string, - bin: string?, - build_files: { string }?, - }, + target: { + environment: environment, + lib: string, + bin: string?, + build_files: { string }?, + }, - scripts: { - roblox_sync_config_generator: string?, - sourcemap_generator: string?, + scripts: { + roblox_sync_config_generator: string?, + sourcemap_generator: string?, - [string]: string, - }?, + [string]: string, + }?, - indices: { - default: string, + indices: { + default: string, - [string]: string, - }, - wally_indices: { - default: string, + [string]: string, + }, + wally_indices: { + default: string, - [string]: string, - }?, + [string]: string, + }?, - overrides: { - [string]: pesde_dependency, - }?, - patches: { - [string]: { [string]: string }, - }?, + overrides: { + [string]: pesde_dependency, + }?, + patches: { + [string]: { [string]: string }, + }?, - dependencies: { - [string]: pesde_dependency, - }?, - peer_dependencies: { - [string]: pesde_dependency, - }?, - dev_dependencies: { - [string]: pesde_dependency, - }?, + dependencies: { + [string]: pesde_dependency, + }?, + peer_dependencies: { + [string]: pesde_dependency, + }?, + dev_dependencies: { + [string]: pesde_dependency, + }?, } print("🟧 Fetching github token...") local github_token: string = process.args[1] if not github_token then - local env_exists = fs.metadata(".env").exists - if not env_exists then - error("Usage: lune run download-jecs [GITHUB_PAT]\nAlternatively, put the PAT in an .env file under GITHUB_PAT") - end + local env_exists = fs.metadata(".env").exists + if not env_exists then + error("Usage: lune run download-jecs [GITHUB_PAT]\nAlternatively, put the PAT in an .env file under GITHUB_PAT") + end - local env = serde.decode("toml", fs.readFile(".env")) - local pat = env.GITHUB_PAT or error("❌ Couldn't read GITHUB_PAT from .env") - github_token = pat + local env = serde.decode("toml", fs.readFile(".env")) + local pat = env.GITHUB_PAT or error("❌ Couldn't read GITHUB_PAT from .env") + github_token = pat end print("✅ Got github token.") @@ -111,7 +111,7 @@ print("✅ Got github token.") print("🟧 Fetching latest jecs wally manifest...") local pull_manifest_res = net.request("https://raw.githubusercontent.com/Ukendio/jecs/refs/heads/main/wally.toml") if not pull_manifest_res.ok then - error(`❌ Couldn't download jecs manifest: {pull_manifest_res.statusCode} {pull_manifest_res.statusMessage}`) + error(`❌ Couldn't download jecs manifest: {pull_manifest_res.statusCode} {pull_manifest_res.statusMessage}`) end local manifest_contents = pull_manifest_res.body local manifest: wally_manifest = serde.decode("toml", manifest_contents) or error("Couldn't decode manifest.") @@ -119,29 +119,29 @@ local jecs_version = manifest.package.version print(`✅ Got latest manifest. Version: {jecs_version}`) type gh_api_tag = { - ref: string, - node_id: string, - url: string, - object: { - sha: string, - type: string, - url: string, - }, + ref: string, + node_id: string, + url: string, + object: { + sha: string, + type: string, + url: string, + }, } print("🟧 Getting tag associated with version in manifest...") local response = net.request({ - url = `https://api.github.com/repos/ukendio/jecs/git/refs/tags/v{jecs_version}`, - method = "GET", - headers = { - Accept = "application/vnd.github+json", - Authorization = `Bearer {github_token}`, - ["X-GitHub-Api-Version"] = "2022-11-28", - }, + url = `https://api.github.com/repos/ukendio/jecs/git/refs/tags/v{jecs_version}`, + method = "GET", + headers = { + Accept = "application/vnd.github+json", + Authorization = `Bearer {github_token}`, + ["X-GitHub-Api-Version"] = "2022-11-28", + }, }) if not response.ok then - error(`❌ Github api response not ok:\n{response.statusCode} @ {response.statusMessage}\n{response.body}`) + error(`❌ Github api response not ok:\n{response.statusCode} @ {response.statusMessage}\n{response.body}`) end local gh_api_tag: gh_api_tag = serde.decode("json", response.body) @@ -149,28 +149,33 @@ print(`✅ Got associated tag: {gh_api_tag.object.sha}`) local URL = `https://raw.githubusercontent.com/Ukendio/jecs/{gh_api_tag.object.sha}/` local function download(path: string, output: string) - process.spawn("curl", { URL .. path, "-o", output }) + process.spawn("curl", { URL .. path, "-o", output }) end local function download_list(list: { { path: string, output: string } }) - local n = #list - print(`🟧 Downloading {n} files...`) + local n = #list + print(`🟧 Downloading {n} files...`) - for idx = 1, n do - local file = list[idx] + for idx = 1, n do + local file = list[idx] - print(`Downloading file {idx}/{n} - {file.path} -> {file.output}...`) - download(file.path, file.output) - end + if file.output == "$" then + file.output = file.path + end - print(`✅ Downloaded {n} files.`) + print(`Downloading file {idx}/{n} - {file.path} -> {file.output}...`) + download(file.path, file.output) + end + + print(`✅ Downloaded {n} files.`) end download_list({ - { path = "src/init.luau", output = "init.luau" }, - { path = ".luaurc", output = ".luaurc" }, - { path = "LICENSE", output = "LICENSE" }, - { path = "README.md", output = "README.md" }, + { path = "jecs.luau", output = "init.luau" }, + { path = ".luaurc", output = "$" }, + { path = "LICENSE", output = "$" }, + { path = "README.md", output = "$" }, + { path = "CHANGELOG.md", output = "$" }, }) print(`🟧 Modifying README.md to include repo info...`) @@ -184,28 +189,29 @@ fs.writeFile("README.md", new_readme_contents) print(`✅ Modified README.md.`) local pesde_manifest: pesde_manifest = { - name = "mark_marks/jecs_pesde", - description = "A minimal copy of jecs published on the official pesde registry", - version = jecs_version, - authors = { "jecs authors" }, - repository = "https://git.devmarked.win/marked/jecs-pesde", - license = manifest.package.license, - includes = { - "init.luau", - "pesde.toml", - "README.md", - "LICENSE", - ".luaurc", - }, + name = "mark_marks/jecs_pesde", + description = "A minimal copy of jecs published on the official pesde registry", + version = jecs_version, + authors = { "jecs authors" }, + repository = "https://git.devmarked.win/marked/jecs-pesde", + license = manifest.package.license, + includes = { + "init.luau", + "pesde.toml", + "README.md", + "CHANGELOG.md", + "LICENSE", + ".luaurc", + }, - target = { - environment = "luau", - lib = "init.luau", - }, + target = { + environment = "luau", + lib = "init.luau", + }, - indices = { - default = "https://github.com/daimond113/pesde-index", - }, + indices = { + default = "https://github.com/pesde-pkg/index", + }, } print("🟧 Writing pesde manifest...") diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 0000000..b3180c6 --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,204 @@ +# Jecs Changelog + +All notable changes to this project will be documented in this file. + +The format is based on [Keep a Changelog][kac], and this project adheres to +[Semantic Versioning][semver]. + +[kac]: https://keepachangelog.com/en/1.1.0/ +[semver]: https://semver.org/spec/v2.0.0.html + +## [Unreleased] + +- `[world]`: + - 16% faster `world:get` +- `[typescript]` + + - Fixed Entity type to default to `undefined | unknown` instead of just `undefined` + +- `[query]`: + - Fixed bug where `world:clear` did not invoke `jecs.OnRemove` hooks + - Changed `query.__iter` to drain on iteration + - It will initialize once wherever you left iteration off at last time + - Changed `query:iter` to restart the iterator + - Removed `query:drain` and `query:next` + - If you want to get individual results outside of a for-loop, you need to call `query:iter` to initialize the iterator and then call the iterator function manually + ```lua + local it = world:query(A, B, C):iter() + local entity, a, b, c = it() + entity, a, b, c = it() -- get next results + ``` +- `[world` + - Fixed a bug with `world:clear` not invoking `jecs.OnRemove` hooks +- `[typescript]`: + - Changed pair to accept generics + - Improved handling of Tags + +## [0.3.2] - 2024-10-01 + +- `[world]`: + - Changed `world:cleanup` to traverse a header type for graph edges. (Edit) + - Fixed a regression that occurred when you call `world:set` following a `world:remove` using the same component + - Remove explicit error in JECS_DEBUG for `world:target` when not applying an index parameter +- `[typescript]` : + - Fixed `world.set` with NoInfer + +## [0.3.1] - 2024-10-01 + +- `[world]`: + - Added an index parameter to `world:target` + - Added a way to change the components limit via `_G.JECS_HI_COMPONENT_ID` + - Set it to whatever number you want but try to make it as close to the number of components you will use as possible + - Make sure to set this before importing jecs or else it will not work + - Added debug mode, enable via setting `_G.JECS_DEBUG` to true + - Make sure to set this before importing jecs or else it will not work + - Added `world:cleanup` which is called to cleanup empty archetypes manually + - Changed `world:delete` to delete archetypes that are dependent on the passed entity + - Changed `world:delete` to delete entity's children before the entity to prevent cycles +- `[query]`: + - Fixed the iterator to not drain by default +- `[typescript]` + - Fixed entry point of the package.json file to be `src` rather than `src/init` + - Fixed `query.next` returning a query object whereas it would be expected to return a tuple containing the entity and the corresponding component values + - Exported `query.archetypes` + - Changed `pair` to return a number instead of an entity + - Preventing direct usage of a pair as an entity while still allowing it to be used as a component + - Exported built-in components `ChildOf` and `Name` + - Exported `world.parent` + +## [0.2.10] - 2024-09-07 + +- `[world]`: + - Improved performance for hooks + - Changed `world:set` to be idempotent when setting tags +- `[traits]`: + - Added cleanup condition `jecs.OnDelete` for when the entity or component is deleted + - Added cleanup action `jecs.Remove` which removes instances of the specified (component) id from all entities + - This is the default cleanup action + - Added component trait `jecs.Tag` which allows for zero-cost components used as tags + - Setting data to a component with this trait will do nothing +- `[luau]`: + - Exported `world:contains()` + - Exported `query:drain()` + - Exported `Query` + - Improved types for the hook `OnAdd`, `OnSet`, `OnRemove` + - Changed functions to accept any ID including pairs in type parameters + - Applies to `world:add()`, `world:set()`, `world:remove()`, `world:get()`, `world:has()` and `world:query()` + - New exported type `Id = Entity | Pair` + - Changed `world:contains()` to return a `boolean` instead of an entity which may or may not exist + - Fixed `world:has()` to take the correct parameters + +## [0.2.2] - 2024-07-07 + +### Added + +- Added `query:replace(function(...T) return ...U end)` for replacing components in place + - Method is fast pathed to replace the data to the components for each corresponding entity + +### Changed + +- Iterator now goes backwards instead to prevent common cases of iterator invalidation + +## [0.2.1] - 2024-07-06 + +### Added + +- Added `jecs.Component` built-in component which will be added to ids created with `world:component()`. + - Used to find every component id with `query(jecs.Component) + +## [0.2.0] - 2024-07-03 + +### Added + +- Added `world:parent(entity)` and `jecs.ChildOf` respectively as first class citizen for building parent-child relationships. + - Give a parent to an entity with `world:add($source, pair(ChildOf, $target))` + - Use `world:parent(entity)` to find the target of the relationship +- Added user-facing Luau types + +### Changed + +- Improved iteration speeds 20-40% by manually indexing rather than using `next()` :scream: + +## [0.1.1] - 2024-05-19 + +### Added + +- Added `world:clear(entity)` for removing the components to the corresponding entity +- Added Typescript Types + +## [0.1.0] - 2024-05-13 + +### Changed + +- Optimized iterator + +## [0.1.0-rc.6] - 2024-05-13 + +### Added + +- Added a `jecs.Wildcard` term + - it lets you query any partially matched pairs + +## [0.1.0-rc.5] - 2024-05-10 + +### Added + +- Added Entity relationships for creating logical connections between entities +- Added `world:__iter method` which allows for iteration over the whole world to get every entity + - used for reconciling whole worlds such as via replication, saving/loading, etc +- Added `world:add(entity, component)` which adds a component to the entity + - it is an idempotent function, so calling it twice and in any order should be fine + +### Fixed + +- Fixed component overriding when in disorder + - Previously setting the components in different order results in it overriding component data because it incorrectly mapped the index of the column. So it took the index from the source archetype rather than the destination archetype + +## [0.0.0-prototype.rc.3] - 2024-05-01 + +### Added + +- Added observers +- Added an arm to query `query:without()` for chaining invariants. + +### Changed + +- Separates ranges for components and entity IDs. + + - IDs created with `world:component()` will promote array lookups rather than map lookups in the `componentIndex` which is a significant boost + +- No longer caches the column pointers directly and instead the column indices which stay persistent even when data is reallocated during swap-removals + - This was an issue with the iterator being invalidated when you move an entity to a different archetype. + +### Fixedhttps://github.com/Ukendio/jecs/releases/tag/v0.0.0-prototype.rc.3 + +- Fixed a bug where changing an existing component would be slow because it was always appending changing the row of the entity record + - The fix dramatically improves times where it is basically down to just the speed of setting a field in a table + +## [0.0.0-prototype.rc.2] - 2024-04-26 + +### Changed + +- Optimized the creation of the query + - It will now finds the smallest archetype map to iterate over +- Optimized the query iterator + + - It will now populates iterator with columns for faster indexing + +- Renamed the insertion method from world:add to world:set to better reflect what it does. + +## [0.0.0-prototype.rc.2] - 2024-04-23 + +- Initial release + +[unreleased]: https://github.com/ukendio/jecs/compare/v0.0.0.0-prototype.rc.2...HEAD +[0.2.2]: https://github.com/ukendio/jecs/releases/tag/v0.2.2 +[0.2.1]: https://github.com/ukendio/jecs/releases/tag/v0.2.1 +[0.2.0]: https://github.com/ukendio/jecs/releases/tag/v0.2.0 +[0.1.1]: https://github.com/ukendio/jecs/releases/tag/v0.1.1 +[0.1.0]: https://github.com/ukendio/jecs/releases/tag/v0.1.0 +[0.1.0-rc.6]: https://github.com/ukendio/jecs/releases/tag/v0.1.0-rc.6 +[0.1.0-rc.5]: https://github.com/ukendio/jecs/releases/tag/v0.1.0-rc.5 +[0.0.0-prototype-rc.3]: https://github.com/ukendio/jecs/releases/tag/v0.0.0-prototype.rc.3 +[0.0.0-prototype.rc.2]: https://github.com/ukendio/jecs/releases/tag/v0.0.0-prototype.rc.2 +[0.0.0-prototype-rc.1]: https://github.com/ukendio/jecs/releases/tag/v0.0.0-prototype.rc.1 diff --git a/README.md b/README.md index 166a5db..07140a1 100644 --- a/README.md +++ b/README.md @@ -1,68 +1,67 @@ An automatically synced minimal copy of the [jecs](https://github.com/Ukendio/jecs) repo published to pesde. - -

- -

- -[![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg?style=for-the-badge)](LICENSE) [![Wally](https://img.shields.io/github/v/tag/ukendio/jecs?&style=for-the-badge)](https://wally.run/package/ukendio/jecs) - -Just a stupidly fast Entity Component System - -* [Entity Relationships](https://ajmmertens.medium.com/building-games-in-ecs-with-entity-relationships-657275ba2c6c) as first class citizens -* Iterate 800,000 entities at 60 frames per second -* Type-safe [Luau](https://luau-lang.org/) API -* Zero-dependency package -* Optimized for column-major operations -* Cache friendly [archetype/SoA](https://ajmmertens.medium.com/building-an-ecs-2-archetypes-and-vectorization-fe21690805f9) storage -* Rigorously [unit tested](https://github.com/Ukendio/jecs/actions/workflows/ci.yaml) for stability - -### Example - -```lua -local world = jecs.World.new() -local pair = jecs.pair - --- These components and functions are actually already builtin --- but have been illustrated for demonstration purposes -local ChildOf = world:component() -local Name = world:component() - -local function parent(entity) - return world:target(entity, ChildOf) -end -local function getName(entity) - return world:get(entity, Name) -end - -local alice = world:entity() -world:set(alice, Name, "alice") - -local bob = world:entity() -world:add(bob, pair(ChildOf, alice)) -world:set(bob, Name, "bob") - -local sara = world:entity() -world:add(sara, pair(ChildOf, alice)) -world:set(sara, Name, "sara") - -print(getName(parent(sara))) - -for e in world:query(pair(ChildOf, alice)) do - print(getName(e), "is the child of alice") -end - --- Output --- "alice" --- bob is the child of alice --- sara is the child of alice -``` - -21,000 entities 125 archetypes 4 random components queried. -![Queries](image-3.png) -Can be found under /benches/visual/query.luau - -Inserting 8 components to an entity and updating them over 50 times. -![Insertions](image-4.png) -Can be found under /benches/visual/insertions.luau +

+ +

+ +[![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg?style=for-the-badge)](LICENSE) [![Wally](https://img.shields.io/github/v/tag/ukendio/jecs?&style=for-the-badge)](https://wally.run/package/ukendio/jecs) + +Just a stupidly fast Entity Component System + +- [Entity Relationships](https://ajmmertens.medium.com/building-games-in-ecs-with-entity-relationships-657275ba2c6c) as first class citizens +- Iterate 800,000 entities at 60 frames per second +- Type-safe [Luau](https://luau-lang.org/) API +- Zero-dependency package +- Optimized for column-major operations +- Cache friendly [archetype/SoA](https://ajmmertens.medium.com/building-an-ecs-2-archetypes-and-vectorization-fe21690805f9) storage +- Rigorously [unit tested](https://github.com/Ukendio/jecs/actions/workflows/ci.yaml) for stability + +### Example + +```lua +local world = jecs.World.new() +local pair = jecs.pair + +-- These components and functions are actually already builtin +-- but have been illustrated for demonstration purposes +local ChildOf = world:component() +local Name = world:component() + +local function parent(entity) + return world:target(entity, ChildOf) +end +local function getName(entity) + return world:get(entity, Name) +end + +local alice = world:entity() +world:set(alice, Name, "alice") + +local bob = world:entity() +world:add(bob, pair(ChildOf, alice)) +world:set(bob, Name, "bob") + +local sara = world:entity() +world:add(sara, pair(ChildOf, alice)) +world:set(sara, Name, "sara") + +print(getName(parent(sara))) + +for e in world:query(pair(ChildOf, alice)) do + print(getName(e), "is the child of alice") +end + +-- Output +-- "alice" +-- bob is the child of alice +-- sara is the child of alice +``` + +21,000 entities 125 archetypes 4 random components queried. +![Queries](assets/image-3.png) +Can be found under /benches/visual/query.luau + +Inserting 8 components to an entity and updating them over 50 times. +![Insertions](assets/image-4.png) +Can be found under /benches/visual/insertions.luau diff --git a/init.luau b/init.luau index d9b4434..940e4b7 100644 --- a/init.luau +++ b/init.luau @@ -16,9 +16,9 @@ type Map = { [K]: V } type GraphEdge = { from: Archetype, to: Archetype?, + id: number, prev: GraphEdge?, next: GraphEdge?, - id: number, } type GraphEdges = Map @@ -31,14 +31,14 @@ type GraphNode = { export type Archetype = { id: number, - node: GraphNode, types: Ty, type: string, entities: { number }, columns: { Column }, records: { ArchetypeRecord }, -} -type Record = { +} & GraphNode + +export type Record = { archetype: Archetype, row: number, dense: i24, @@ -78,30 +78,32 @@ type EntityIndex = { local HI_COMPONENT_ID = _G.__JECS_HI_COMPONENT_ID or 256 -- stylua: ignore start -local EcsOnAdd = HI_COMPONENT_ID + 1 -local EcsOnRemove = HI_COMPONENT_ID + 2 -local EcsOnSet = HI_COMPONENT_ID + 3 -local EcsWildcard = HI_COMPONENT_ID + 4 -local EcsChildOf = HI_COMPONENT_ID + 5 -local EcsComponent = HI_COMPONENT_ID + 6 -local EcsOnDelete = HI_COMPONENT_ID + 7 -local EcsOnDeleteTarget = HI_COMPONENT_ID + 8 -local EcsDelete = HI_COMPONENT_ID + 9 -local EcsRemove = HI_COMPONENT_ID + 10 -local EcsName = HI_COMPONENT_ID + 11 -local EcsRest = HI_COMPONENT_ID + 12 +local EcsOnAdd = HI_COMPONENT_ID + 1 +local EcsOnRemove = HI_COMPONENT_ID + 2 +local EcsOnSet = HI_COMPONENT_ID + 3 +local EcsWildcard = HI_COMPONENT_ID + 4 +local EcsChildOf = HI_COMPONENT_ID + 5 +local EcsComponent = HI_COMPONENT_ID + 6 +local EcsOnDelete = HI_COMPONENT_ID + 7 +local EcsOnDeleteTarget = HI_COMPONENT_ID + 8 +local EcsDelete = HI_COMPONENT_ID + 9 +local EcsRemove = HI_COMPONENT_ID + 10 +local EcsName = HI_COMPONENT_ID + 11 +local EcsOnArchetypeCreate = HI_COMPONENT_ID + 12 +local EcsOnArchetypeDelete = HI_COMPONENT_ID + 13 +local EcsRest = HI_COMPONENT_ID + 14 -local ECS_PAIR_FLAG = 0x8 -local ECS_ID_FLAGS_MASK = 0x10 -local ECS_ENTITY_MASK = bit32.lshift(1, 24) -local ECS_GENERATION_MASK = bit32.lshift(1, 16) +local ECS_PAIR_FLAG = 0x8 +local ECS_ID_FLAGS_MASK = 0x10 +local ECS_ENTITY_MASK = bit32.lshift(1, 24) +local ECS_GENERATION_MASK = bit32.lshift(1, 16) -local ECS_ID_DELETE = 0b0000_0001 -local ECS_ID_IS_TAG = 0b0000_0010 -local ECS_ID_HAS_ON_ADD = 0b0000_0100 -local ECS_ID_HAS_ON_SET = 0b0000_1000 -local ECS_ID_HAS_ON_REMOVE = 0b0001_0000 -local ECS_ID_MASK = 0b0000_0000 +local ECS_ID_DELETE = 0b0000_0001 +local ECS_ID_IS_TAG = 0b0000_0010 +local ECS_ID_HAS_ON_ADD = 0b0000_0100 +local ECS_ID_HAS_ON_SET = 0b0000_1000 +local ECS_ID_HAS_ON_REMOVE = 0b0001_0000 +local ECS_ID_MASK = 0b0000_0000 -- stylua: ignore end local NULL_ARRAY = table.freeze({}) :: Column @@ -168,7 +170,7 @@ local function _STRIP_GENERATION(e: i53): i24 end local function ECS_PAIR(pred: i53, obj: i53): i53 - return ECS_COMBINE(ECS_ENTITY_T_LO(obj), ECS_ENTITY_T_LO(pred)) + FLAGS_ADD(--[[isPair]] true) :: i53 + return ECS_COMBINE(ECS_ENTITY_T_LO(pred), ECS_ENTITY_T_LO(obj)) + FLAGS_ADD(--[[isPair]] true) :: i53 end local function entity_index_try_get_any(entity_index: EntityIndex, entity: number): Record? @@ -220,7 +222,7 @@ local function entity_index_is_alive(entity_index: EntityIndex, entity: number) return entity_index_try_get(entity_index, entity) ~= nil end -local function entity_index_new_id(entity_index: EntityIndex, data): i53 +local function entity_index_new_id(entity_index: EntityIndex): i53 local dense_array = entity_index.dense_array local alive_count = entity_index.alive_count if alive_count ~= #dense_array then @@ -242,12 +244,42 @@ end -- ECS_PAIR_FIRST, gets the relationship target / obj / HIGH bits local function ecs_pair_first(world, e) - return entity_index_get_alive(world.entity_index, ECS_ENTITY_T_HI(e)) + return entity_index_get_alive(world.entity_index, ECS_ENTITY_T_LO(e)) end -- ECS_PAIR_SECOND gets the relationship / pred / LOW bits local function ecs_pair_second(world, e) - return entity_index_get_alive(world.entity_index, ECS_ENTITY_T_LO(e)) + return entity_index_get_alive(world.entity_index, ECS_ENTITY_T_HI(e)) +end + +local function query_match(query, archetype: Archetype) + local records = archetype.records + local with = query.filter_with + + for _, id in with do + if not records[id] then + return false + end + end + + local without = query.filter_without + if without then + for _, id in without do + if records[id] then + return false + end + end + end + + return true +end + +local function find_observers(world: World, event, component): { Observer }? + local cache = world.observerable[event] + if not cache then + return nil + end + return cache[component] :: any end local function archetype_move(entity_index: EntityIndex, to: Archetype, dst_row: i24, from: Archetype, src_row: i24) @@ -257,7 +289,7 @@ local function archetype_move(entity_index: EntityIndex, to: Archetype, dst_row: local src_entities = from.entities local last = #src_entities - local types = from.types + local id_types = from.types local records = to.records for i, column in src_columns do @@ -266,12 +298,13 @@ local function archetype_move(entity_index: EntityIndex, to: Archetype, dst_row: end -- Retrieves the new column index from the source archetype's record from each component -- We have to do this because the columns are tightly packed and indexes may not correspond to each other. - local tr = records[types[i]] + local tr = records[id_types[i]] -- Sometimes target column may not exist, e.g. when you remove a component. if tr then dst_columns[tr.column][dst_row] = column[src_row] end + -- If the entity is the last row in the archetype then swapping it would be meaningless. if src_row ~= last then -- Swap rempves columns to ensure there are no holes in the archetype. @@ -330,51 +363,43 @@ local function hash(arr: { number }): string return table.concat(arr, "_") end -local world_get: (world: World, entityId: i53, a: i53, b: i53?, c: i53?, d: i53?, e: i53?) -> ...any -do - -- Keeping the function as small as possible to enable inlining - local records: { ArchetypeRecord } - local columns: { { any } } - local row: number +local function fetch(id, records: { ArchetypeRecord }, columns: { Column }, row: number): any + local tr = records[id] - local function fetch(id): any - local tr = records[id] - - if not tr then - return nil - end - - return columns[tr.column][row] + if not tr then + return nil end - function world_get(world: World, entity: i53, a: i53, b: i53?, c: i53?, d: i53?, e: i53?): ...any - local record = entity_index_try_get_fast(world.entity_index, entity) - if not record then - return nil - end + return columns[tr.column][row] +end - local archetype = record.archetype - if not archetype then - return nil - end +local function world_get(world: World, entity: i53, a: i53, b: i53?, c: i53?, d: i53?, e: i53?): ...any + local record = entity_index_try_get_fast(world.entity_index, entity) + if not record then + return nil + end - records = archetype.records - columns = archetype.columns - row = record.row + local archetype = record.archetype + if not archetype then + return nil + end - local va = fetch(a) + local records = archetype.records + local columns = archetype.columns + local row = record.row - if not b then - return va - elseif not c then - return va, fetch(b) - elseif not d then - return va, fetch(b), fetch(c) - elseif not e then - return va, fetch(b), fetch(c), fetch(d) - else - error("args exceeded") - end + local va = fetch(a, records, columns, row) + + if not b then + return va + elseif not c then + return va, fetch(b, records, columns, row) + elseif not d then + return va, fetch(b, records, columns, row), fetch(c, records, columns, row) + elseif not e then + return va, fetch(b, records, columns, row), fetch(c, records, columns, row), fetch(d, records, columns, row) + else + error("args exceeded") end end @@ -483,7 +508,8 @@ local function id_record_ensure(world: World, id: number): IdRecord if not idr then local flags = ECS_ID_MASK local relation = id - if ECS_IS_PAIR(id) then + local is_pair = ECS_IS_PAIR(id) + if is_pair then relation = ecs_pair_first(world, id) end @@ -500,6 +526,10 @@ local function id_record_ensure(world: World, id: number): IdRecord local is_tag = not world_has_one_inline(world, relation, EcsComponent) + if is_tag and is_pair then + is_tag = not world_has_one_inline(world, ecs_pair_second(world, id), EcsComponent) + end + flags = bit32.bor( flags, if on_add then ECS_ID_HAS_ON_ADD else 0, @@ -543,15 +573,29 @@ local function archetype_append_to_records( end end -local function archetype_create(world: World, types: { i24 }, ty, prev: i53?): Archetype +local function archetype_create(world: World, id_types: { i24 }, ty, prev: i53?): Archetype local archetype_id = (world.nextArchetypeId :: number) + 1 world.nextArchetypeId = archetype_id - local length = #types + local length = #id_types local columns = (table.create(length) :: any) :: { Column } local records: { ArchetypeRecord } = {} - for i, componentId in types do + + local archetype: Archetype = { + columns = columns, + entities = {}, + id = archetype_id, + records = records, + type = ty, + types = id_types, + + add = {}, + remove = {}, + refs = {} :: GraphEdge, + } + + for i, componentId in id_types do local idr = id_record_ensure(world, componentId) archetype_append_to_records(idr, archetype_id, records, componentId, i) @@ -567,6 +611,7 @@ local function archetype_create(world: World, types: { i24 }, ty, prev: i53?): A local idr_t = id_record_ensure(world, t) archetype_append_to_records(idr_t, archetype_id, records, t, i) end + if bit32.band(idr.flags, ECS_ID_IS_TAG) == 0 then columns[i] = {} else @@ -574,15 +619,17 @@ local function archetype_create(world: World, types: { i24 }, ty, prev: i53?): A end end - local archetype: Archetype = { - columns = columns, - node = { add = {}, remove = {}, refs = {} :: GraphEdge }, - entities = {}, - id = archetype_id, - records = records, - type = ty, - types = types, - } + for _, id in id_types do + local observer_list = find_observers(world, EcsOnArchetypeCreate, id) + if not observer_list then + continue + end + for _, observer in observer_list do + if query_match(observer.query, archetype) then + observer.callback(archetype) + end + end + end world.archetypeIndex[ty] = archetype world.archetypes[archetype_id] = archetype @@ -598,22 +645,22 @@ local function world_parent(world: World, entity: i53) return world_target(world, entity, EcsChildOf, 0) end -local function archetype_ensure(world: World, types): Archetype - if #types < 1 then +local function archetype_ensure(world: World, id_types): Archetype + if #id_types < 1 then return world.ROOT_ARCHETYPE end - local ty = hash(types) + local ty = hash(id_types) local archetype = world.archetypeIndex[ty] if archetype then return archetype end - return archetype_create(world, types, ty) + return archetype_create(world, id_types, ty) end -local function find_insert(types: { i53 }, toAdd: i53): number - for i, id in types do +local function find_insert(id_types: { i53 }, toAdd: i53): number + for i, id in id_types do if id == toAdd then return -1 end @@ -621,17 +668,17 @@ local function find_insert(types: { i53 }, toAdd: i53): number return i end end - return #types + 1 + return #id_types + 1 end local function find_archetype_with(world: World, node: Archetype, id: i53): Archetype - local types = node.types + local id_types = node.types -- Component IDs are added incrementally, so inserting and sorting -- them each time would be expensive. Instead this insertion sort can find the insertion -- point in the types array. local dst = table.clone(node.types) :: { i53 } - local at = find_insert(types, id) + local at = find_insert(id_types, id) if at == -1 then -- If it finds a duplicate, it just means it is the same archetype so it can return it -- directly instead of needing to hash types for a lookup to the archetype. @@ -643,13 +690,13 @@ local function find_archetype_with(world: World, node: Archetype, id: i53): Arch end local function find_archetype_without(world: World, node: Archetype, id: i53): Archetype - local types = node.types - local at = table.find(types, id) + local id_types = node.types + local at = table.find(id_types, id) if at == nil then return node end - local dst = table.clone(types) + local dst = table.clone(id_types) table.remove(dst, at) return archetype_ensure(world, dst) @@ -671,11 +718,11 @@ local function archetype_ensure_edge(world, edges: GraphEdges, id): GraphEdge return edge end -local function init_edge_for_add(world, archetype, edge: GraphEdge, id, to) +local function init_edge_for_add(world, archetype: Archetype, edge: GraphEdge, id, to: Archetype) archetype_init_edge(archetype, edge, id, to) - archetype_ensure_edge(world, archetype.node.add, id) + archetype_ensure_edge(world, archetype.add, id) if archetype ~= to then - local to_refs = to.node.refs + local to_refs = to.refs local next_edge = to_refs.next to_refs.next = edge @@ -690,9 +737,9 @@ end local function init_edge_for_remove(world: World, archetype: Archetype, edge: GraphEdge, id: number, to: Archetype) archetype_init_edge(archetype, edge, id, to) - archetype_ensure_edge(world, archetype.node.remove, id) + archetype_ensure_edge(world, archetype.remove, id) if archetype ~= to then - local to_refs = to.node.refs + local to_refs = to.refs local prev_edge = to_refs.prev to_refs.prev = edge @@ -717,9 +764,9 @@ local function create_edge_for_remove(world: World, node: Archetype, edge: Graph return to end -local function archetype_traverse_add(world: World, id: i53, from: Archetype?): Archetype +local function archetype_traverse_add(world: World, id: i53, from: Archetype): Archetype from = from or world.ROOT_ARCHETYPE - local edge = archetype_ensure_edge(world, from.node.add, id) + local edge = archetype_ensure_edge(world, from.add, id) local to = edge.to if not to then @@ -732,7 +779,7 @@ end local function archetype_traverse_remove(world: World, id: i53, from: Archetype): Archetype from = from or world.ROOT_ARCHETYPE - local edge = archetype_ensure_edge(world, from.node.remove, id) + local edge = archetype_ensure_edge(world, from.remove, id) local to = edge.to if not to then @@ -891,7 +938,7 @@ end local function archetype_delete(world: World, archetype: Archetype, row: number, destruct: boolean?) local entityIndex = world.entity_index local columns = archetype.columns - local types = archetype.types + local id_types = archetype.types local entities = archetype.entities local column_count = #entities local last = #entities @@ -910,7 +957,7 @@ local function archetype_delete(world: World, archetype: Archetype, row: number, -- TODO: if last == 0 then deactivate table - for _, id in types do + for _, id in id_types do local on_remove: (entity: i53) -> () = world_get_one_inline(world, id, EcsOnRemove) if on_remove then on_remove(delete) @@ -918,9 +965,9 @@ local function archetype_delete(world: World, archetype: Archetype, row: number, end if row == last then - archetype_fast_delete_last(columns, column_count, types, delete) + archetype_fast_delete_last(columns, column_count, id_types, delete) else - archetype_fast_delete(columns, column_count, row, types, delete) + archetype_fast_delete(columns, column_count, row, id_types, delete) end end @@ -961,10 +1008,9 @@ local function archetype_remove_edge(edges: Map, id: i53, edge: end local function archetype_clear_edges(archetype: Archetype) - local node: GraphNode = archetype.node - local add: GraphEdges = node.add - local remove: GraphEdges = node.remove - local node_refs: GraphEdge = node.refs + local add: GraphEdges = archetype.add + local remove: GraphEdges = archetype.remove + local node_refs = archetype.refs for id, edge in add do archetype_disconnect_edge(edge) add[id] = nil :: any @@ -976,9 +1022,9 @@ local function archetype_clear_edges(archetype: Archetype) local cur = node_refs.next while cur do - local edge: GraphEdge = cur + local edge = cur :: GraphEdge local next_edge = edge.next - archetype_remove_edge(edge.from.node.add, edge.id, edge) + archetype_remove_edge(edge.from.add, edge.id, edge) cur = next_edge end @@ -986,7 +1032,7 @@ local function archetype_clear_edges(archetype: Archetype) while cur do local edge: GraphEdge = cur local next_edge = edge.prev - archetype_remove_edge(edge.from.node.remove, edge.id, edge) + archetype_remove_edge(edge.from.remove, edge.id, edge) cur = next_edge end @@ -1006,6 +1052,18 @@ local function archetype_destroy(world: World, archetype: Archetype) world.archetypeIndex[archetype.type] = nil :: any local records = archetype.records + for id in records do + local observer_list = find_observers(world, EcsOnArchetypeDelete, id) + if not observer_list then + continue + end + for _, observer in observer_list do + if query_match(observer.query, archetype) then + observer.callback(archetype) + end + end + end + for id in records do local idr = component_index[id] idr.cache[archetype_id] = nil :: any @@ -1058,29 +1116,36 @@ do local delete = entity local component_index = world.componentIndex - local archetypes = world.archetypes + local archetypes: Archetypes = world.archetypes local tgt = ECS_PAIR(EcsWildcard, delete) local idr_t = component_index[tgt] local idr = component_index[delete] if idr then - local children = {} - for archetype_id in idr.cache do - local idr_archetype = archetypes[archetype_id] - - for i, child in idr_archetype.entities do - table.insert(children, child) - end - end local flags = idr.flags if bit32.band(flags, ECS_ID_DELETE) ~= 0 then - for _, child in children do - -- Cascade deletion to children - world_delete(world, child) + for archetype_id in idr.cache do + local idr_archetype = archetypes[archetype_id] + + local entities = idr_archetype.entities + local n = #entities + for i = n, 1, -1 do + world_delete(world, entities[i]) + end end else - for _, child in children do - world_remove(world, child, delete) + for archetype_id in idr.cache do + local idr_archetype = archetypes[archetype_id] + local entities = idr_archetype.entities + local n = #entities + for i = n, 1, -1 do + world_remove(world, entities[i], delete) + end + end + + for archetype_id in idr.cache do + local idr_archetype = archetypes[archetype_id] + archetype_destroy(world, idr_archetype) end end end @@ -1091,6 +1156,7 @@ do local idr_t_archetype = archetypes[archetype_id] local idr_t_types = idr_t_archetype.types + local on_remove = idr_t.hooks.on_remove for _, child in idr_t_archetype.entities do table.insert(children, child) @@ -1112,8 +1178,18 @@ do end break else - for _, child in children do - world_remove(world, child, id) + local to = archetype_traverse_remove(world, id, idr_t_archetype) + if on_remove then + for _, child in children do + on_remove(child) + local r = entity_index_try_get_fast(entity_index, child) :: Record + entity_move(entity_index, child, r, to) + end + else + for _, child in children do + local r = entity_index_try_get_fast(entity_index, child) :: Record + entity_move(entity_index, child, r, to) + end end end end @@ -1167,7 +1243,16 @@ local EMPTY_QUERY = { setmetatable(EMPTY_QUERY, EMPTY_QUERY) -local function query_iter_init(query): () -> (number, ...any) +type QueryInner = { + compatible_archetypes: { Archetype }, + ids: { i53 }, + filter_with: { i53 }, + filter_without: { i53 }, + next: () -> (number, ...any), + world: World, +} + +local function query_iter_init(query: QueryInner): () -> (number, ...any) local world_query_iter_next local compatible_archetypes = query.compatible_archetypes @@ -1244,9 +1329,12 @@ local function query_iter_init(query): () -> (number, ...any) entities = archetype.entities i = #entities + if i == 0 then + continue + end entityId = entities[i] columns = archetype.columns - local records = archetype.records + records = archetype.records a = columns[records[A].column] end @@ -1267,9 +1355,12 @@ local function query_iter_init(query): () -> (number, ...any) entities = archetype.entities i = #entities + if i == 0 then + continue + end entityId = entities[i] columns = archetype.columns - local records = archetype.records + records = archetype.records a = columns[records[A].column] b = columns[records[B].column] end @@ -1291,9 +1382,12 @@ local function query_iter_init(query): () -> (number, ...any) entities = archetype.entities i = #entities + if i == 0 then + continue + end entityId = entities[i] columns = archetype.columns - local records = archetype.records + records = archetype.records a = columns[records[A].column] b = columns[records[B].column] c = columns[records[C].column] @@ -1316,9 +1410,12 @@ local function query_iter_init(query): () -> (number, ...any) entities = archetype.entities i = #entities + if i == 0 then + continue + end entityId = entities[i] columns = archetype.columns - local records = archetype.records + records = archetype.records a = columns[records[A].column] b = columns[records[B].column] c = columns[records[C].column] @@ -1343,9 +1440,12 @@ local function query_iter_init(query): () -> (number, ...any) entities = archetype.entities i = #entities + if i == 0 then + continue + end entityId = entities[i] columns = archetype.columns - local records = archetype.records + records = archetype.records if not F then a = columns[records[A].column] @@ -1393,7 +1493,6 @@ local function query_iter_init(query): () -> (number, ...any) return entityId, a[row], b[row], c[row], d[row], e[row], f[row], g[row], h[row] end - local records = archetype.records for j, id in ids do queryOutput[j] = columns[records[id].column][row] end @@ -1414,65 +1513,64 @@ local function query_iter(query): () -> (number, ...any) return query_next end -local function query_without(query: { compatible_archetypes: { Archetype } }, ...) +local function query_without(query: QueryInner, ...: i53) + local without = { ... } + query.filter_without = without local compatible_archetypes = query.compatible_archetypes - local N = select("#", ...) for i = #compatible_archetypes, 1, -1 do local archetype = compatible_archetypes[i] local records = archetype.records - local shouldRemove = false + local matches = true - for j = 1, N do - local id = select(j, ...) + for _, id in without do if records[id] then - shouldRemove = true + matches = false break end end - if shouldRemove then - local last = #compatible_archetypes - if last ~= i then - compatible_archetypes[i] = compatible_archetypes[last] - end - compatible_archetypes[last] = nil :: any + if matches then + continue end - end - if #compatible_archetypes == 0 then - return EMPTY_QUERY + local last = #compatible_archetypes + if last ~= i then + compatible_archetypes[i] = compatible_archetypes[last] + end + compatible_archetypes[last] = nil :: any end return query :: any end -local function query_with(query: { compatible_archetypes: { Archetype } }, ...) +local function query_with(query: QueryInner, ...: i53) local compatible_archetypes = query.compatible_archetypes - local N = select("#", ...) + local with = { ... } + query.filter_with = with + for i = #compatible_archetypes, 1, -1 do local archetype = compatible_archetypes[i] local records = archetype.records - local shouldRemove = false + local matches = true - for j = 1, N do - local id = select(j, ...) + for _, id in with do if not records[id] then - shouldRemove = true + matches = false break end end - if shouldRemove then - local last = #compatible_archetypes - if last ~= i then - compatible_archetypes[i] = compatible_archetypes[last] - end - compatible_archetypes[last] = nil :: any + if matches then + continue end + + local last = #compatible_archetypes + if last ~= i then + compatible_archetypes[i] = compatible_archetypes[last] + end + compatible_archetypes[last] = nil :: any end - if #compatible_archetypes == 0 then - return EMPTY_QUERY - end + return query :: any end @@ -1483,6 +1581,324 @@ local function query_archetypes(query) return query.compatible_archetypes end +local function query_cached(query: QueryInner) + local with = query.filter_with + local ids = query.ids + if with then + table.move(ids, 1, #ids, #with + 1, with) + else + query.filter_with = ids + end + + local compatible_archetypes = query.compatible_archetypes + local lastArchetype = 1 + + local A, B, C, D, E, F, G, H, I = unpack(ids) + local a: Column, b: Column, c: Column, d: Column + local e: Column, f: Column, g: Column, h: Column + + local world_query_iter_next + local columns: { Column } + local entities: { number } + local i: number + local archetype: Archetype + local records: { ArchetypeRecord } + local archetypes = query.compatible_archetypes + + local world = query.world :: World + -- Only need one observer for EcsArchetypeCreate and EcsArchetypeDelete respectively + -- because the event will be emitted for all components of that Archetype. + local observerable = world.observerable + local on_create_action = observerable[EcsOnArchetypeCreate] + if not on_create_action then + on_create_action = {} + observerable[EcsOnArchetypeCreate] = on_create_action + end + local query_cache_on_create = on_create_action[A] + if not query_cache_on_create then + query_cache_on_create = {} + on_create_action[A] = query_cache_on_create + end + + local on_delete_action = observerable[EcsOnArchetypeDelete] + if not on_delete_action then + on_delete_action = {} + observerable[EcsOnArchetypeDelete] = on_delete_action + end + local query_cache_on_delete = on_delete_action[A] + if not query_cache_on_delete then + query_cache_on_delete = {} + on_delete_action[A] = query_cache_on_delete + end + + local function on_create_callback(archetype) + table.insert(archetypes, archetype) + end + + local function on_delete_callback(archetype) + local i = table.find(archetypes, archetype) :: number + local n = #archetypes + archetypes[i] = archetypes[n] + archetypes[n] = nil + end + + local observer_for_create = { query = query, callback = on_create_callback } + local observer_for_delete = { query = query, callback = on_delete_callback } + + table.insert(query_cache_on_create, observer_for_create) + table.insert(query_cache_on_delete, observer_for_delete) + + local function cached_query_iter() + lastArchetype = 1 + archetype = compatible_archetypes[lastArchetype] + if not archetype then + return NOOP + end + entities = archetype.entities + i = #entities + records = archetype.records + columns = archetype.columns + if not B then + a = columns[records[A].column] + elseif not C then + a = columns[records[A].column] + b = columns[records[B].column] + elseif not D then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + elseif not E then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + elseif not F then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + elseif not G then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + f = columns[records[F].column] + elseif not H then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + f = columns[records[F].column] + g = columns[records[G].column] + elseif not I then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + f = columns[records[F].column] + g = columns[records[G].column] + h = columns[records[H].column] + end + + return world_query_iter_next + end + + if not B then + function world_query_iter_next(): any + local entityId = entities[i] + while entityId == nil do + lastArchetype += 1 + archetype = compatible_archetypes[lastArchetype] + if not archetype then + return nil + end + + entities = archetype.entities + i = #entities + if i == 0 then + continue + end + entityId = entities[i] + columns = archetype.columns + records = archetype.records + a = columns[records[A].column] + end + + local row = i + i -= 1 + + return entityId, a[row] + end + elseif not C then + function world_query_iter_next(): any + local entityId = entities[i] + while entityId == nil do + lastArchetype += 1 + archetype = compatible_archetypes[lastArchetype] + if not archetype then + return nil + end + + entities = archetype.entities + i = #entities + if i == 0 then + continue + end + entityId = entities[i] + columns = archetype.columns + records = archetype.records + a = columns[records[A].column] + b = columns[records[B].column] + end + + local row = i + i -= 1 + + return entityId, a[row], b[row] + end + elseif not D then + function world_query_iter_next(): any + local entityId = entities[i] + while entityId == nil do + lastArchetype += 1 + archetype = compatible_archetypes[lastArchetype] + if not archetype then + return nil + end + + entities = archetype.entities + i = #entities + if i == 0 then + continue + end + entityId = entities[i] + columns = archetype.columns + records = archetype.records + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + end + + local row = i + i -= 1 + + return entityId, a[row], b[row], c[row] + end + elseif not E then + function world_query_iter_next(): any + local entityId = entities[i] + while entityId == nil do + lastArchetype += 1 + archetype = compatible_archetypes[lastArchetype] + if not archetype then + return nil + end + + entities = archetype.entities + i = #entities + if i == 0 then + continue + end + entityId = entities[i] + columns = archetype.columns + records = archetype.records + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + end + + local row = i + i -= 1 + + return entityId, a[row], b[row], c[row], d[row] + end + else + local queryOutput = {} + function world_query_iter_next(): any + local entityId = entities[i] + while entityId == nil do + lastArchetype += 1 + archetype = compatible_archetypes[lastArchetype] + if not archetype then + return nil + end + + entities = archetype.entities + i = #entities + if i == 0 then + continue + end + entityId = entities[i] + columns = archetype.columns + records = archetype.records + + if not F then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + elseif not G then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + f = columns[records[F].column] + elseif not H then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + f = columns[records[F].column] + g = columns[records[G].column] + elseif not I then + a = columns[records[A].column] + b = columns[records[B].column] + c = columns[records[C].column] + d = columns[records[D].column] + e = columns[records[E].column] + f = columns[records[F].column] + g = columns[records[G].column] + h = columns[records[H].column] + end + end + + local row = i + i -= 1 + + if not F then + return entityId, a[row], b[row], c[row], d[row], e[row] + elseif not G then + return entityId, a[row], b[row], c[row], d[row], e[row], f[row] + elseif not H then + return entityId, a[row], b[row], c[row], d[row], e[row], f[row], g[row] + elseif not I then + return entityId, a[row], b[row], c[row], d[row], e[row], f[row], g[row], h[row] + end + + for j, id in ids do + queryOutput[j] = columns[records[id].column][row] + end + + return entityId, unpack(queryOutput) + end + end + + local cached_query = query :: any + cached_query.archetypes = query_archetypes + cached_query.__iter = cached_query_iter + cached_query.iter = cached_query_iter + setmetatable(cached_query, cached_query) + return cached_query +end + local Query = {} Query.__index = Query Query.__iter = query_iter @@ -1490,6 +1906,7 @@ Query.iter = query_iter_init Query.without = query_without Query.with = query_with Query.archetypes = query_archetypes +Query.cached = query_cached local function world_query(world: World, ...) local compatible_archetypes = {} @@ -1502,10 +1919,16 @@ local function world_query(world: World, ...) local idr: IdRecord? local componentIndex = world.componentIndex + local q = setmetatable({ + ids = ids, + compatible_archetypes = compatible_archetypes, + world = world, + }, Query) + for _, id in ids do local map = componentIndex[id] if not map then - return EMPTY_QUERY + return q end if idr == nil or map.size < idr.size then @@ -1514,7 +1937,7 @@ local function world_query(world: World, ...) end if not idr then - return EMPTY_QUERY + return q end for archetype_id in idr.cache do @@ -1542,16 +1965,45 @@ local function world_query(world: World, ...) compatible_archetypes[length] = compatibleArchetype end - if length == 0 then - return EMPTY_QUERY + return q +end + +local function world_each(world: World, id): () -> () + local idr = world.componentIndex[id] + if not idr then + return NOOP end - local q = setmetatable({ - compatible_archetypes = compatible_archetypes, - ids = ids, - }, Query) :: any + local idr_cache = idr.cache + local archetypes = world.archetypes + local archetype_id = next(idr_cache, nil) :: number + local archetype = archetypes[archetype_id] + if not archetype then + return NOOP + end - return q + local entities = archetype.entities + local row = #entities + + return function(): any + local entity = entities[row] + while not entity do + archetype_id = next(idr_cache, archetype_id) :: number + if not archetype_id then + return + end + archetype = archetypes[archetype_id] + entities = archetype.entities + row = #entities + entity = entities[row] + end + row -= 1 + return entity + end +end + +local function world_children(world, parent) + return world_each(world, ECS_PAIR(EcsChildOf, parent)) end local World = {} @@ -1571,15 +2023,19 @@ World.target = world_target World.parent = world_parent World.contains = world_contains World.cleanup = world_cleanup +World.each = world_each +World.children = world_children if _G.__JECS_DEBUG then - -- taken from https://github.com/centau/ecr/blob/main/src/ecr.luau - -- error but stack trace always starts at first callsite outside of this file + local function dbg_info(n: number): any + return debug.info(n, "s") + end local function throw(msg: string) local s = 1 + local root = dbg_info(1) repeat s += 1 - until debug.info(s, "s") ~= debug.info(1, "s") + until dbg_info(s) ~= root if warn then error(msg, s) else @@ -1594,15 +2050,18 @@ if _G.__JECS_DEBUG then throw(msg) end - local function get_name(world, id): string - local name: string | nil + local function get_name(world, id) + return world_get_one_inline(world, id, EcsName) + end + + local function bname(world: World, id): string + local name: string if ECS_IS_PAIR(id) then - name = `pair({get_name(world, ECS_ENTITY_T_HI(id))}, {get_name(world, ECS_ENTITY_T_LO(id))})` + local first = get_name(world, ecs_pair_first(world, id)) + local second = get_name(world, ecs_pair_second(world, id)) + name = `pair({first}, {second})` else - local _1 = world_get_one_inline(world, id, EcsName) - if _1 then - name = `${_1}` - end + return get_name(world, id) end if name then return name @@ -1626,14 +2085,14 @@ if _G.__JECS_DEBUG then World.set = function(world: World, entity: i53, id: i53, value: any): () local is_tag = ID_IS_TAG(world, id) if is_tag and value == nil then - local _1 = get_name(world, entity) - local _2 = get_name(world, id) + local _1 = bname(world, entity) + local _2 = bname(world, id) local why = "cannot set component value to nil" throw(why) return elseif value ~= nil and is_tag then - local _1 = get_name(world, entity) - local _2 = get_name(world, id) + local _1 = bname(world, entity) + local _2 = bname(world, id) local why = `cannot set a component value because {_2} is a tag` why ..= `\n[jecs] note: consider using "world:add({_1}, {_2})" instead` throw(why) @@ -1643,10 +2102,10 @@ if _G.__JECS_DEBUG then world_set(world, entity, id, value) end - World.add = function(world: World, entity: i53, id: i53, value: nil) + World.add = function(world: World, entity: i53, id: i53, value: any) if value ~= nil then - local _1 = get_name(world, entity) - local _2 = get_name(world, id) + local _1 = bname(world, entity) + local _2 = bname(world, id) throw("You provided a value when none was expected. " .. `Did you mean to use "world:add({_1}, {_2})"`) end @@ -1692,6 +2151,7 @@ function World.new() nextComponentId = 0 :: number, nextEntityId = 0 :: number, ROOT_ARCHETYPE = (nil :: any) :: Archetype, + observerable = {}, }, World) :: any self.ROOT_ARCHETYPE = archetype_create(self, {}, "") @@ -1731,37 +2191,53 @@ function World.new() return self end -export type Id = Entity | Pair, Entity> +export type Id = + | Entity + | Pair, Entity> + | Pair> + | Pair, Entity> -export type Pair = number & { - __relation: First, +export type Pair = number & { + __P: P, + __O: O, } --- type function _Pair(first, second) --- local thing = first:components()[2] +-- type function ecs_id_t(entity) +-- local ty = entity:components()[2] +-- local __T = ty:readproperty(types.singleton("__T")) +-- if not __T then +-- return ty:readproperty(types.singleton("__jecs_pair_value")) +-- end +-- return __T +-- end --- if thing:readproperty(types.singleton("__T")):is("nil") then +-- type function ecs_pair_t(first, second) +-- if ecs_id_t(first):is("nil") then -- return second -- else -- return first -- end -- end --- type TestPair = _Pair, Entity> - type Item = (self: Query) -> (Entity, T...) export type Entity = number & { __T: T } type Iter = (query: Query) -> () -> (Entity, T...) -type Query = typeof(setmetatable({}, { +export type Query = typeof(setmetatable({}, { __iter = (nil :: any) :: Iter, })) & { iter: Iter, - with: (self: Query, ...i53) -> Query, - without: (self: Query, ...i53) -> Query, + with: (self: Query, ...Id) -> Query, + without: (self: Query, ...Id) -> Query, archetypes: (self: Query) -> { Archetype }, + cached: (self: Query) -> Query, +} + +type Observer = { + callback: (archetype: Archetype) -> (), + query: QueryInner, } export type World = { @@ -1774,6 +2250,14 @@ export type World = { nextComponentId: number, nextEntityId: number, nextArchetypeId: number, + + observerable: { + [i53]: { + [i53]: { + { query: QueryInner, callback: (Archetype) -> () } + } + } + }, } & { --- Creates a new entity entity: (self: World) -> Entity, @@ -1811,47 +2295,23 @@ export type World = { --- Checks if the world contains the given entity contains: (self: World, entity: Entity) -> boolean, + each: (self: World, id: Id) -> () -> Entity, + + children: (self: World, id: Id) -> () -> Entity, + --- Searches the world for entities that match a given query - query: ((self: World, Id) -> Query) - & ((self: World, Id, Id) -> Query) - & ((self: World, Id, Id, Id) -> Query) - & ((self: World, Id, Id, Id, Id) -> Query) - & ((self: World, Id, Id, Id, Id, Id) -> Query) - & (( - self: World, - Id, - Id, - Id, - Id, - Id, - Id - ) -> Query) - & (( - self: World, - Id, - Id, - Id, - Id, - Id, - Id, - Id - ) -> Query) - & (( - self: World, - Id, - Id, - Id, - Id, - Id, - Id, - Id, - Id, - ...Id - ) -> Query), + query: ((World, Id) -> Query) + & ((World, Id, Id) -> Query) + & ((World, Id, Id, Id) -> Query) + & ((World, Id, Id, Id, Id) -> Query) + & ((World, Id, Id, Id, Id, Id) -> Query) + & ((World, Id, Id, Id, Id, Id, Id) -> Query) + & ((World, Id, Id, Id, Id, Id, Id, Id) -> Query) + & ((World, Id, Id, Id, Id, Id, Id, Id, Id, ...Id) -> Query) } return { - World = World :: { new: () -> World }, + World = World, OnAdd = EcsOnAdd :: Entity<(entity: Entity) -> ()>, OnRemove = EcsOnRemove :: Entity<(entity: Entity) -> ()>, @@ -1898,6 +2358,13 @@ return { entity_index_try_get = entity_index_try_get, entity_index_try_get_any = entity_index_try_get_any, + entity_index_try_get_fast = entity_index_try_get_fast, entity_index_is_alive = entity_index_is_alive, entity_index_new_id = entity_index_new_id, + + query_iter = query_iter, + query_iter_init = query_iter_init, + query_with = query_with, + query_without = query_without, + query_archetypes = query_archetypes, } diff --git a/pesde.toml b/pesde.toml index b9a5002..a7a3a55 100644 --- a/pesde.toml +++ b/pesde.toml @@ -4,16 +4,17 @@ includes = [ "init.luau", "pesde.toml", "README.md", + "CHANGELOG.md", "LICENSE", ".luaurc", ] license = "MIT" name = "mark_marks/jecs_pesde" repository = "https://git.devmarked.win/marked/jecs-pesde" -version = "0.4.0" +version = "0.5.2" [indices] -default = "https://github.com/daimond113/pesde-index" +default = "https://github.com/pesde-pkg/index" [target] environment = "luau" diff --git a/rokit.toml b/rokit.toml index fd63af4..9675eea 100644 --- a/rokit.toml +++ b/rokit.toml @@ -5,4 +5,4 @@ [tools] lune = "lune-org/lune@0.8.9" -pesde = "daimond113/pesde@0.5.0-rc.14" +pesde = "daimond113/pesde@0.5.3+registry.0.1.2" diff --git a/stylua.toml b/stylua.toml new file mode 100644 index 0000000..920827f --- /dev/null +++ b/stylua.toml @@ -0,0 +1,10 @@ +column_width = 120 +line_endings = "Unix" +indent_type = "Spaces" +indent_width = 4 +quote_style = "AutoPreferDouble" +call_parentheses = "Input" +collapse_simple_statement = "Never" + +[sort_requires] +enabled = true