aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFederico Igne <git@federicoigne.com>2023-03-05 21:25:43 +0000
committerFederico Igne <git@federicoigne.com>2023-03-05 21:25:43 +0000
commit5ab1e5c0bbed7d0958b4ba211e3085e97bb91ffd (patch)
treea2584aaf4be8fc16ce3ac8130286e6f5286207e1
parent15245aff28022c167b5384f164ad3bf6f5cf60b4 (diff)
downloadraccoon-5ab1e5c0bbed7d0958b4ba211e3085e97bb91ffd.tar.gz
raccoon-5ab1e5c0bbed7d0958b4ba211e3085e97bb91ffd.zip
refactor(entities): shorten component getters/checkers for entities
-rw-r--r--raccoon.lua189
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.
113function Entity:init() 115function 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
118end 119end
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-- ```
131function Entity:with_component(c) 131function 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
135end 140end
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.
139function Entity:remove_component(c) 144function 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
143end 153end
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`.
153local Component = util.new_class() 165local Component = util.new_class()
154function Component:init() 166function 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
156end 174end
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
213end 231end
214 232
215local Pos = Component() 233Pos = Component("Pos")
216function Pos:init(x,y) 234function 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
236end 254end
237 255
238local Movement = Component() 256Movement = Component("Movement")
239function Movement:init(accel, max) 257function 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
251end 269end
252 270
253local Pixel = Component() 271Pixel = Component("Pixel")
254function Pixel:init(c) 272function Pixel:init(c)
255 self.color = c 273 self.color = c
256 return self 274 return self
257end 275end
258 276
259local Sprite = Component() 277Sprite = Component("Sprite")
260function Sprite:init(i) 278function Sprite:init(i)
261 self.i = i 279 self.i = i
262 return self 280 return self
263end 281end
264 282
265local Jump = Component() 283Jump = Component("Jump")
266function Jump:init(pwr) 284function 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
271end 289end
272 290
273local Input = Component() 291Input = Component("Input")
274function Input:init(up, down, left, right, a, b) 292function 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
279end 297end
280 298
281local Level = Component() 299Level = Component("Level")
282function Level:init(id, screens) 300function 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
286end 304end
287 305
288local Entities = Component() 306Entities = Component("Entities")
289function Entities:init(list) 307function 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
301end 319end
302 320
303local CircleEffect = Component() 321CircleEffect = Component("CircleEffect")
304function CircleEffect:init(r2, r1, time, color) 322function 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)
313end 331end
314 332
315local PosAnim = Component() 333PosAnim = Component("PosAnim")
316function PosAnim:init(f,d) 334function 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
321end 339end
322 340
323local Particles = Component() 341Particles = Component("Particles")
324function Particles:init(d) 342function 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
339end 357end
340 358
341local ForceField = Component()
342function ForceField:init(force,range,instant) 359function ForceField:init(force,range,instant)
360ForceField = 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
347end 365end
348 366
349local Metadata = Component() 367Metadata = Component("Metadata")
350function Metadata:init(name) 368function 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
355local ForceFieldSystem = System(Pos,ForceField) 373local ForceFieldSystem = System(Pos,ForceField)
356function ForceFieldSystem:exec(entity,enemies) 374function 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
381local InputSystem = System(Input) 398local InputSystem = System(Input)
382function InputSystem:exec(entity) 399function 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)
520end 537end
521function DrawingSystem:draw(game) 538function 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)
527end 544end
528 545
529local LevelSystem = System(Level) 546local LevelSystem = System(Level)
530function LevelSystem:exec(level, game) 547function 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
551local ActionSystem = System(Pos,Input,Action) 568local ActionSystem = System(Pos,Input,Action)
552function ActionSystem:exec(entity,game) 569function 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
564local EffectSystem = System() 581local EffectSystem = System()
565function EffectSystem:exec(entity) 582function 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
575local PosAnimSystem = System(Pos, PosAnim) 592local PosAnimSystem = System(Pos, PosAnim)
576function PosAnimSystem:exec(entity) 593function 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
586local ParticlesSystem = System(Pos, Particles) 603local ParticlesSystem = System(Pos, Particles)
587function ParticlesSystem:exec(entity, game) 604function 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
595end 612end
596 613
597local player = Entity() 614local 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
614local level1_entities = { 621local 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