diff --git a/index.html b/index.html
new file mode 100644
index 0000000..d6bb146
--- /dev/null
+++ b/index.html
@@ -0,0 +1,50 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
data:image/s3,"s3://crabby-images/e69dd/e69ddeff5c5763f5b085ac87c0ad3c4799be48ed" alt=""
+
data:image/s3,"s3://crabby-images/06442/06442d6d852c4add74c0de42fee97da6e961d401" alt=""
+
data:image/s3,"s3://crabby-images/2b466/2b466078389e641a9262b2e74463bdf0cb985d05" alt=""
+
data:image/s3,"s3://crabby-images/ad95c/ad95c19eedbd0e21f5006009a35e61504c1a4b7e" alt=""
+
data:image/s3,"s3://crabby-images/64603/6460355e71b82adf514bf103b4c831a38dc21240" alt=""
+
data:image/s3,"s3://crabby-images/9c9d4/9c9d40bc9a918a3da305d2a12211c3cdd3548735" alt=""
+
data:image/s3,"s3://crabby-images/e4742/e47423ca8d5fc0cd1eae46b6aeb37ac9144d70c8" alt=""
+
data:image/s3,"s3://crabby-images/7b9b3/7b9b3641127943c1f68c4403b6597aea693e887c" alt=""
+
data:image/s3,"s3://crabby-images/1e25e/1e25e77d92ffd813f9f0f0476176d7d292c70b7f" alt=""
+
data:image/s3,"s3://crabby-images/f6dea/f6deacb996c378373387af3e2ccc9e07808937e1" alt=""
+
data:image/s3,"s3://crabby-images/9d2ca/9d2cabca0bbf5f22698beaf92b2bd97133250812" alt=""
+
data:image/s3,"s3://crabby-images/26940/26940bf53d3953932ca74f94bf61867fff6af186" alt=""
+
data:image/s3,"s3://crabby-images/7cfda/7cfdae432bcf72079f105f23da4b0029f3cc8aa7" alt=""
+
data:image/s3,"s3://crabby-images/a09dc/a09dc1bfdc7e5b8d4f687863d29c9b0a64207687" alt=""
+
data:image/s3,"s3://crabby-images/a6414/a64140a67eeebe1997e2c916e2e801f7c3a20d12" alt=""
+
data:image/s3,"s3://crabby-images/893b2/893b21cc137e415709d4110a87c86bf0f94aab2b" alt=""
+
+
+
+
diff --git a/nodes.js b/nodes.js
new file mode 100644
index 0000000..155ca65
--- /dev/null
+++ b/nodes.js
@@ -0,0 +1,65 @@
+// CURTAIN {{{
+// CURTAIN }}}
+// ACT1SCENE1 {{{
+
+class ECS {
+ constructor() {
+ this.maxEntity = 0;
+ this.components = {};
+ }
+
+ addEntity(...components) {
+ const entity = ++this.maxEntity;
+ for (const c of components) {
+ this.attach(entity, c);
+ }
+ return entity;
+ }
+
+ attach(entity, component) {
+ (this.components[component.constructor.name] ||= {})[entity] = component;
+ }
+}
+
+class ComponentBounds {
+ constructor(x, y, width, height) {
+ this.x = x;
+ this.y = y;
+ this.width = width;
+ this.height = height;
+ }
+}
+
+class ComponentWhiteBox {}
+
+class SystemRender {
+ constructor(world, canvas) {
+ this.world = world;
+ this.ctx = canvas.getContext('2d');
+ }
+
+ render() {
+ const { width, height } = this.ctx.canvas;
+ this.ctx.clearRect(0, 0, width, height);
+ for (const entity in this.world.components.ComponentWhiteBox || []) {
+ const { x, y, width, height } = this.world.components.ComponentBounds[entity];
+ this.ctx.fillStyle = 'white';
+ this.ctx.fillRect(x, y, width, height);
+ }
+ }
+}
+
+let world;
+
+window.onload = function() {
+ world = new ECS();
+ const canvas = document.getElementById('game');
+ const render = new SystemRender(world, canvas);
+ world.addEntity(
+ new ComponentBounds(100, 100, 80, 50),
+ new ComponentWhiteBox(),
+ );
+ render.render();
+};
+
+// ACT1SCENE1 }}}