diff options
author | Federico Igne <git@federicoigne.com> | 2023-03-05 21:25:43 +0000 |
---|---|---|
committer | Federico Igne <git@federicoigne.com> | 2023-03-05 21:25:43 +0000 |
commit | 5ab1e5c0bbed7d0958b4ba211e3085e97bb91ffd (patch) | |
tree | a2584aaf4be8fc16ce3ac8130286e6f5286207e1 | |
parent | 15245aff28022c167b5384f164ad3bf6f5cf60b4 (diff) | |
download | raccoon-5ab1e5c0bbed7d0958b4ba211e3085e97bb91ffd.tar.gz raccoon-5ab1e5c0bbed7d0958b4ba211e3085e97bb91ffd.zip |
refactor(entities): shorten component getters/checkers for entities
-rw-r--r-- | raccoon.lua | 189 |
1 files changed, 98 insertions, 91 deletions
diff --git a/raccoon.lua b/raccoon.lua index 0ada342..c0fec2f 100644 --- a/raccoon.lua +++ b/raccoon.lua | |||
@@ -104,41 +104,51 @@ end | |||
104 | -- Entities keep track of components using a table indexed by component | 104 | -- Entities keep track of components using a table indexed by component |
105 | -- class. | 105 | -- class. |
106 | -- @usage | 106 | -- @usage |
107 | -- Given an entity `e`, a component `c` of class `C` | 107 | -- Given an entity `e`, a component `c1` of class `C` |
108 | -- ``` | 108 | -- ``` |
109 | -- e:with_component(c) | 109 | -- e:with(c1) |
110 | -- e.has_component[C] -- true | 110 | -- e:has_c() == e:components[c1.__index] == true |
111 | -- e.get_component[C] -- c | 111 | -- e:get_c() == e:components[c1.__index] == c1 |
112 | -- ``` | 112 | -- ``` |
113 | -- | ||
114 | -- @return this entity. | ||
113 | function Entity:init() | 115 | function Entity:init() |
114 | self.id = Entity.get_id() | 116 | self.id = Entity.get_id() |
115 | self.has_component = {} | 117 | self.components = {} |
116 | self.get_component = {} | ||
117 | return self | 118 | return self |
118 | end | 119 | end |
119 | --- Add component to this entity. | 120 | --- Add component to this entity. |
120 | -- @param c the component to be added to this entity. | 121 | -- @param t the component(s) to be added to this entity. |
121 | -- @return this entity. | 122 | -- @return this entity. |
122 | -- | 123 | -- |
123 | -- @note the modified entity is returned to allow chains of multiple calls to this method. | 124 | -- @note the modified entity is returned to allow chains of multiple calls to this method. |
125 | -- | ||
124 | -- @usage | 126 | -- @usage |
125 | -- ``` | 127 | -- ``` |
126 | -- entity | 128 | -- entity:with(c1):with(c2):with(c3) |
127 | -- :with_component(c1) | 129 | -- entity:with({c1,c2,c3}) |
128 | -- :with_component(c2) | ||
129 | -- :with_component(c3) | ||
130 | -- ``` | 130 | -- ``` |
131 | function Entity:with_component(c) | 131 | function Entity:with(t) |
132 | self.has_component[c.__index] = true | 132 | if t.meta then |
133 | self.get_component[c.__index] = c | 133 | self.components[t.__index] = t |
134 | else | ||
135 | for _,c in ipairs(t) do | ||
136 | self.components[c.__index] = c | ||
137 | end | ||
138 | end | ||
134 | return self | 139 | return self |
135 | end | 140 | end |
136 | --- Remove component from this entity. | 141 | --- Remove component from this entity. |
137 | -- @param c the component to be removed from this entity. | 142 | -- @param t the component(s) to be removed from this entity. |
138 | -- @return this entity. | 143 | -- @return this entity. |
139 | function Entity:remove_component(c) | 144 | function Entity:without(t) |
140 | self.has_component[c.__index] = false | 145 | if t.meta then |
141 | self.get_component[c.__index] = nil -- possibly triggers GC | 146 | self.components[t.__index] = nil -- possibly triggers GC |
147 | else | ||
148 | for _,c in ipairs(t) do | ||
149 | self.components[c.__index] = nil | ||
150 | end | ||
151 | end | ||
142 | return self | 152 | return self |
143 | end | 153 | end |
144 | 154 | ||
@@ -147,12 +157,20 @@ end | |||
147 | -- component. Components are what make entities interesting and | 157 | -- component. Components are what make entities interesting and |
148 | -- different from each other. | 158 | -- different from each other. |
149 | -- | 159 | -- |
150 | -- @note at the moment this is simply a wrapper to create a new class. | 160 | -- The new component keeps track of its name in a `meta` table inside of it. |
151 | -- Component-specific functions can be added in the wrapper, if | 161 | -- @note for `t` table, you can use `t.meta` as a boolean check for whether `t` |
152 | -- necessary. | 162 | -- is a component. |
163 | -- | ||
164 | -- Given a component "Comp", functions `has_comp` and `get_comp` are added to `Entity`. | ||
153 | local Component = util.new_class() | 165 | local Component = util.new_class() |
154 | function Component:init() | 166 | function Component:init(name) |
155 | return util.new_class() | 167 | local component = util.new_class() |
168 | component.meta = { name = name or "" } -- NOTE use meta as "component check" | ||
169 | if name then | ||
170 | Entity["has_"..name:lower()] = function(self) return self.components[_G[name].__index] end | ||
171 | Entity["get_"..name:lower()] = function(self) return self.components[_G[name].__index] end | ||
172 | end | ||
173 | return component | ||
156 | end | 174 | end |
157 | 175 | ||
158 | --- Systems. | 176 | --- Systems. |
@@ -188,7 +206,7 @@ function System:init(...) | |||
188 | -- @todo is there a better way? | 206 | -- @todo is there a better way? |
189 | local selected = true | 207 | local selected = true |
190 | for _, component in ipairs(self.components) do | 208 | for _, component in ipairs(self.components) do |
191 | if not entity.has_component[component] then | 209 | if not Entity["has_"..component.meta.name:lower()](entity) then |
192 | selected = false | 210 | selected = false |
193 | break | 211 | break |
194 | end | 212 | end |
@@ -199,7 +217,7 @@ function System:init(...) | |||
199 | if cmps then | 217 | if cmps then |
200 | -- Remove components from entity | 218 | -- Remove components from entity |
201 | for _,c in ipairs(cmps) do | 219 | for _,c in ipairs(cmps) do |
202 | entity:remove_component(c) | 220 | entity:without(c) |
203 | end | 221 | end |
204 | else | 222 | else |
205 | -- Remove entity | 223 | -- Remove entity |
@@ -212,7 +230,7 @@ function System:init(...) | |||
212 | return cls | 230 | return cls |
213 | end | 231 | end |
214 | 232 | ||
215 | local Pos = Component() | 233 | Pos = Component("Pos") |
216 | function Pos:init(x,y) | 234 | function Pos:init(x,y) |
217 | self.x = x or 0 | 235 | self.x = x or 0 |
218 | self.y = y or 0 | 236 | self.y = y or 0 |
@@ -235,7 +253,7 @@ function Pos:to_screen(lpos) | |||
235 | return self.x - lpos.x, self.y - lpos.y | 253 | return self.x - lpos.x, self.y - lpos.y |
236 | end | 254 | end |
237 | 255 | ||
238 | local Movement = Component() | 256 | Movement = Component("Movement") |
239 | function Movement:init(accel, max) | 257 | function Movement:init(accel, max) |
240 | -- direction facing | 258 | -- direction facing |
241 | self.dir = true -- true: right, false: left | 259 | self.dir = true -- true: right, false: left |
@@ -250,19 +268,19 @@ function Movement:is_moving() | |||
250 | return self.cur.x ~= 0 or self.cur.y ~= 0 | 268 | return self.cur.x ~= 0 or self.cur.y ~= 0 |
251 | end | 269 | end |
252 | 270 | ||
253 | local Pixel = Component() | 271 | Pixel = Component("Pixel") |
254 | function Pixel:init(c) | 272 | function Pixel:init(c) |
255 | self.color = c | 273 | self.color = c |
256 | return self | 274 | return self |
257 | end | 275 | end |
258 | 276 | ||
259 | local Sprite = Component() | 277 | Sprite = Component("Sprite") |
260 | function Sprite:init(i) | 278 | function Sprite:init(i) |
261 | self.i = i | 279 | self.i = i |
262 | return self | 280 | return self |
263 | end | 281 | end |
264 | 282 | ||
265 | local Jump = Component() | 283 | Jump = Component("Jump") |
266 | function Jump:init(pwr) | 284 | function Jump:init(pwr) |
267 | self.jumping = false | 285 | self.jumping = false |
268 | self.cur = 0 | 286 | self.cur = 0 |
@@ -270,7 +288,7 @@ function Jump:init(pwr) | |||
270 | return self | 288 | return self |
271 | end | 289 | end |
272 | 290 | ||
273 | local Input = Component() | 291 | Input = Component("Input") |
274 | function Input:init(up, down, left, right, a, b) | 292 | function Input:init(up, down, left, right, a, b) |
275 | self.btns = { up or 0, down or 1, left or 2, right or 3, a or 4, b or 5 } | 293 | self.btns = { up or 0, down or 1, left or 2, right or 3, a or 4, b or 5 } |
276 | self.raw = { false, false, false, false, false, false } | 294 | self.raw = { false, false, false, false, false, false } |
@@ -278,14 +296,14 @@ function Input:init(up, down, left, right, a, b) | |||
278 | return self | 296 | return self |
279 | end | 297 | end |
280 | 298 | ||
281 | local Level = Component() | 299 | Level = Component("Level") |
282 | function Level:init(id, screens) | 300 | function Level:init(id, screens) |
283 | self.id = id | 301 | self.id = id |
284 | self.screens = screens | 302 | self.screens = screens |
285 | return self | 303 | return self |
286 | end | 304 | end |
287 | 305 | ||
288 | local Entities = Component() | 306 | Entities = Component("Entities") |
289 | function Entities:init(list) | 307 | function Entities:init(list) |
290 | for i,v in pairs(list) do | 308 | for i,v in pairs(list) do |
291 | self[i] = v | 309 | self[i] = v |
@@ -300,7 +318,7 @@ function Action:init(a,b) | |||
300 | return self | 318 | return self |
301 | end | 319 | end |
302 | 320 | ||
303 | local CircleEffect = Component() | 321 | CircleEffect = Component("CircleEffect") |
304 | function CircleEffect:init(r2, r1, time, color) | 322 | function CircleEffect:init(r2, r1, time, color) |
305 | self.r2 = r2 | 323 | self.r2 = r2 |
306 | self.r1 = r1 or 1 | 324 | self.r1 = r1 or 1 |
@@ -312,7 +330,7 @@ function CircleEffect:draw(x,y) | |||
312 | circb(x,y,self.r1,self.color) | 330 | circb(x,y,self.r1,self.color) |
313 | end | 331 | end |
314 | 332 | ||
315 | local PosAnim = Component() | 333 | PosAnim = Component("PosAnim") |
316 | function PosAnim:init(f,d) | 334 | function PosAnim:init(f,d) |
317 | self.t = 0 | 335 | self.t = 0 |
318 | self.d = d | 336 | self.d = d |
@@ -320,7 +338,7 @@ function PosAnim:init(f,d) | |||
320 | return self | 338 | return self |
321 | end | 339 | end |
322 | 340 | ||
323 | local Particles = Component() | 341 | Particles = Component("Particles") |
324 | function Particles:init(d) | 342 | function Particles:init(d) |
325 | self.t = 0 | 343 | self.t = 0 |
326 | self.d = d | 344 | self.d = d |
@@ -332,21 +350,21 @@ function Particles:spawn(pos, entities) | |||
332 | local offset = math.random(7) -- TODO change to pos.width once implemented | 350 | local offset = math.random(7) -- TODO change to pos.width once implemented |
333 | local speed = 0.2 + math.random() | 351 | local speed = 0.2 + math.random() |
334 | table.insert(entities, | 352 | table.insert(entities, |
335 | Entity():with_component(Pixel(12)) | 353 | Entity():with(Pixel(12)) |
336 | :with_component(Pos(pos.x + offset,pos.y)) | 354 | :with(Pos(pos.x + offset,pos.y)) |
337 | :with_component(PosAnim(function(t,p) p.y = p.y - speed end, 10))) | 355 | :with(PosAnim(function(t,p) p.y = p.y - speed end, 10))) |
338 | end | 356 | end |
339 | end | 357 | end |
340 | 358 | ||
341 | local ForceField = Component() | ||
342 | function ForceField:init(force,range,instant) | 359 | function ForceField:init(force,range,instant) |
360 | ForceField = Component("ForceField") | ||
343 | self.force = force or 1 | 361 | self.force = force or 1 |
344 | self.range = range or 1 | 362 | self.range = range or 1 |
345 | self.instant = instant | 363 | self.instant = instant |
346 | return self | 364 | return self |
347 | end | 365 | end |
348 | 366 | ||
349 | local Metadata = Component() | 367 | Metadata = Component("Metadata") |
350 | function Metadata:init(name) | 368 | function Metadata:init(name) |
351 | self.name = name or "Unnamed" | 369 | self.name = name or "Unnamed" |
352 | return self | 370 | return self |
@@ -354,12 +372,11 @@ end | |||
354 | 372 | ||
355 | local ForceFieldSystem = System(Pos,ForceField) | 373 | local ForceFieldSystem = System(Pos,ForceField) |
356 | function ForceFieldSystem:exec(entity,enemies) | 374 | function ForceFieldSystem:exec(entity,enemies) |
357 | local pos = entity.get_component[Pos] | 375 | local pos = entity:get_pos() |
358 | local ff = entity.get_component[ForceField] | 376 | local ff = entity:get_forcefield() |
359 | for _,e in pairs(enemies) do | 377 | for _,e in pairs(enemies) do |
360 | if e.has_component[Pos] and e.has_component[Movement] then | 378 | if e:has_pos() and ff.is_target(e) then |
361 | local pos1 = e.get_component[Pos] | 379 | local pos1 = e:get_pos() |
362 | local mov = e.get_component[Movement] | ||
363 | local dist,vec = util.distance(pos,pos1) | 380 | local dist,vec = util.distance(pos,pos1) |
364 | if dist < ff.range then | 381 | if dist < ff.range then |
365 | if e.has_component[Metadata] then | 382 | if e.has_component[Metadata] then |
@@ -380,7 +397,7 @@ end | |||
380 | 397 | ||
381 | local InputSystem = System(Input) | 398 | local InputSystem = System(Input) |
382 | function InputSystem:exec(entity) | 399 | function InputSystem:exec(entity) |
383 | local input = entity.get_component[Input] | 400 | local input = entity:get_input() |
384 | for i = 1,6 do | 401 | for i = 1,6 do |
385 | input.raw[i] = btn(input.btns[i]) | 402 | input.raw[i] = btn(input.btns[i]) |
386 | input.rawp[i] = btnp(input.btns[i]) | 403 | input.rawp[i] = btnp(input.btns[i]) |
@@ -520,25 +537,25 @@ function DrawingSystem:exec(entity,level) | |||
520 | end | 537 | end |
521 | function DrawingSystem:draw(game) | 538 | function DrawingSystem:draw(game) |
522 | local level = game.levels[game.level] | 539 | local level = game.levels[game.level] |
523 | local info = level.get_component[Level] | 540 | local info = level:get_level() |
524 | local pos = level.get_component[Pos] | 541 | local pos = level:get_pos() |
525 | map(pos.x//8,pos.y//8,31,18,-(pos.x%8),-(pos.y%8),-1) | 542 | map(pos.x//8,pos.y//8,31,18,-(pos.x%8),-(pos.y%8),-1) |
526 | DrawingSystem:run(game.entities,pos) | 543 | DrawingSystem:run(game.entities,pos) |
527 | end | 544 | end |
528 | 545 | ||
529 | local LevelSystem = System(Level) | 546 | local LevelSystem = System(Level) |
530 | function LevelSystem:exec(level, game) | 547 | function LevelSystem:exec(level, game) |
531 | local info = level.get_component[Level] | 548 | local info = level:get_level() |
532 | local lpos = level.get_component[Pos] | 549 | local lpos = level:get_pos() |
533 | local entities = level.get_component[Entities] | 550 | local entities = level:get_entities() |
534 | local ppos = game.player.get_component[Pos] | 551 | local ppos = game.player:get_pos() |
535 | lpos.x = math.min( | 552 | lpos.x = math.min( |
536 | math.max(0, ppos.x - util.screen.width // 2), | 553 | math.max(0, ppos.x - util.screen.width // 2), |
537 | (info.screens - 1) * 240 | 554 | (info.screens - 1) * 240 |
538 | ) | 555 | ) |
539 | for i,e in pairs(entities) do | 556 | for i,e in pairs(entities) do |
540 | if e.has_component[Pos] then | 557 | if e:has_pos() then |
541 | local pos = e.get_component[Pos] | 558 | local pos = e:get_pos() |
542 | if pos.x >= lpos.x and pos.x < lpos.x + 240 and | 559 | if pos.x >= lpos.x and pos.x < lpos.x + 240 and |
543 | pos.y >= lpos.y and pos.y < lpos.y + 134 then | 560 | pos.y >= lpos.y and pos.y < lpos.y + 134 then |
544 | table.insert(game.entities, e) | 561 | table.insert(game.entities, e) |
@@ -550,9 +567,9 @@ end | |||
550 | 567 | ||
551 | local ActionSystem = System(Pos,Input,Action) | 568 | local ActionSystem = System(Pos,Input,Action) |
552 | function ActionSystem:exec(entity,game) | 569 | function ActionSystem:exec(entity,game) |
553 | local pos = entity.get_component[Pos] | 570 | local pos = entity:get_pos() |
554 | local input = entity.get_component[Input] | 571 | local input = entity:get_input() |
555 | local action = entity.get_component[Action] | 572 | local action = entity:get_action() |
556 | if input.rawp[5] and action.a then | 573 | if input.rawp[5] and action.a then |
557 | action.a(entity,game) | 574 | action.a(entity,game) |
558 | end | 575 | end |
@@ -563,8 +580,8 @@ end | |||
563 | 580 | ||
564 | local EffectSystem = System() | 581 | local EffectSystem = System() |
565 | function EffectSystem:exec(entity) | 582 | function EffectSystem:exec(entity) |
566 | if entity.has_component[CircleEffect] then | 583 | if entity:has_circleeffect() then |
567 | local effect = entity.get_component[CircleEffect] | 584 | local effect = entity:get_circleeffect() |
568 | effect.r1 = effect.r1 + effect.step | 585 | effect.r1 = effect.r1 + effect.step |
569 | if effect.r1 > effect.r2 then | 586 | if effect.r1 > effect.r2 then |
570 | return true | 587 | return true |
@@ -574,8 +591,8 @@ end | |||
574 | 591 | ||
575 | local PosAnimSystem = System(Pos, PosAnim) | 592 | local PosAnimSystem = System(Pos, PosAnim) |
576 | function PosAnimSystem:exec(entity) | 593 | function PosAnimSystem:exec(entity) |
577 | local pos = entity.get_component[Pos] | 594 | local pos = entity:get_pos() |
578 | local anim = entity.get_component[PosAnim] | 595 | local anim = entity:get_posanim() |
579 | anim.f(anim.t, pos) | 596 | anim.f(anim.t, pos) |
580 | anim.t = anim.t + 1 | 597 | anim.t = anim.t + 1 |
581 | if anim.d and anim.t > anim.d then | 598 | if anim.d and anim.t > anim.d then |
@@ -585,8 +602,8 @@ end | |||
585 | 602 | ||
586 | local ParticlesSystem = System(Pos, Particles) | 603 | local ParticlesSystem = System(Pos, Particles) |
587 | function ParticlesSystem:exec(entity, game) | 604 | function ParticlesSystem:exec(entity, game) |
588 | local pos = entity.get_component[Pos] | 605 | local pos = entity:get_pos() |
589 | local parts = entity.get_component[Particles] | 606 | local parts = entity:get_particles() |
590 | parts:spawn(pos,game.entities) | 607 | parts:spawn(pos,game.entities) |
591 | parts.t = parts.t + 1 | 608 | parts.t = parts.t + 1 |
592 | if parts.d and parts.t > parts.d then | 609 | if parts.d and parts.t > parts.d then |
@@ -594,34 +611,24 @@ function ParticlesSystem:exec(entity, game) | |||
594 | end | 611 | end |
595 | end | 612 | end |
596 | 613 | ||
597 | local player = Entity() | 614 | local player = Entity():with({ |
598 | :with_component(Metadata("Giulia")) | 615 | Metadata("Giulia"), Sprite(256), |
599 | :with_component(Pos(100,10)) | 616 | Pos(100,10), Health(100), Status(), |
600 | :with_component(Movement()) | 617 | Movement(), Jump(10), |
601 | :with_component(Jump(10)) | 618 | Input(0,1,2,3,4,5), Action(Action.forcefield,nil) |
602 | :with_component(Sprite(256)) | 619 | }) |
603 | :with_component(Input(0,1,2,3,4,5)) | ||
604 | :with_component(Action( | ||
605 | function(entity,game) | ||
606 | --local lpos = game.levels[game.level].get_component[Pos] | ||
607 | local pos = entity.get_component[Pos]--:to_screen(lpos) | ||
608 | table.insert( | ||
609 | game.entities, | ||
610 | Entity():with_component(Pos(pos.x,pos.y)):with_component(ForceField(5,10,true)):with_component(CircleEffect(20))) | ||
611 | end, | ||
612 | nil)) | ||
613 | 620 | ||
614 | local level1_entities = { | 621 | local level1_entities = { |
615 | -- Turists | 622 | -- Turists |
616 | Entity():with_component(Metadata("Tourist1")):with_component(Pos(16*8,11*8)):with_component(Sprite(272)):with_component(Movement()), | 623 | Entity():with({Metadata("Tourist1"), Sprite(272), Pos(16*8,11*8), Health(5), Status(), Movement(), MovementAI(), Target(player:get_pos())}), |
617 | Entity():with_component(Metadata("Tourist2")):with_component(Pos(47*8, 8*8)):with_component(Sprite(272)):with_component(Movement()), | 624 | Entity():with({Metadata("Tourist2"), Sprite(272), Pos(47*8, 8*8), Health(5), Status(), Movement(), MovementAI()}), |
618 | Entity():with_component(Metadata("Tourist3")):with_component(Pos(85*8, 7*8)):with_component(Sprite(272)):with_component(Movement()), | 625 | Entity():with({Metadata("Tourist3"), Sprite(272), Pos(85*8, 7*8), Health(5), Status(), Movement(), MovementAI()}), |
619 | -- Waypoints | 626 | -- Waypoints |
620 | Entity():with_component(Pos(15*8,11*8)):with_component(Sprite(288)) | 627 | Entity():with(Pos(15*8,11*8)):with(Sprite(288)) |
621 | :with_component(PosAnim(function(t,p) p.y = p.y - math.sin(t/6)/3 end)) | 628 | :with(PosAnim(function(t,p) p.y = p.y - math.sin(t/6)/3 end)) |
622 | :with_component(Particles()), | 629 | :with(Particles()), |
623 | Entity():with_component(Pos(48*8, 8*8)):with_component(Sprite(288)) | 630 | Entity():with(Pos(48*8, 8*8)):with(Sprite(288)) |
624 | :with_component(PosAnim(function(t,p) p.y = p.y - math.sin(t/6)/3 end)), | 631 | :with(PosAnim(function(t,p) p.y = p.y - math.sin(t/6)/3 end)), |
625 | } | 632 | } |
626 | 633 | ||
627 | -- TODO turn this into something more structured, containing everything related | 634 | -- TODO turn this into something more structured, containing everything related |
@@ -632,9 +639,9 @@ local game = { | |||
632 | level = 1, | 639 | level = 1, |
633 | levels = { | 640 | levels = { |
634 | Entity() | 641 | Entity() |
635 | :with_component(Level(0,3)) | 642 | :with(Level(0,3)) |
636 | :with_component(Pos(0,0)) | 643 | :with(Pos(0,0)) |
637 | :with_component(Entities(level1_entities)) | 644 | :with(Entities(level1_entities)) |
638 | } | 645 | } |
639 | } | 646 | } |
640 | 647 | ||