Part 6 done, entities & player are dying correctly
This commit is contained in:
parent
f89c2bbdb8
commit
83f7723c08
@ -1,21 +1,25 @@
|
|||||||
|
from typing import Any
|
||||||
|
|
||||||
class BaseAction:
|
class BaseAction:
|
||||||
|
entity: Any
|
||||||
|
|
||||||
def __init__(self, entity):
|
def __init__(self, entity):
|
||||||
self.entity = entity
|
self.entity = entity
|
||||||
|
|
||||||
def apply(self) -> None:
|
def apply(self) -> bool:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
class QuitAction(BaseAction):
|
class QuitAction(BaseAction):
|
||||||
def __init__(self): #override the base __init__
|
def __init__(self): #override the base __init__
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def apply(self) -> None:
|
def apply(self) -> bool:
|
||||||
raise SystemExit()
|
raise SystemExit()
|
||||||
|
|
||||||
|
|
||||||
class WaitAction(BaseAction):
|
class WaitAction(BaseAction):
|
||||||
def apply(self) -> None:
|
def apply(self) -> bool:
|
||||||
pass
|
return True
|
||||||
|
|
||||||
|
|
||||||
class DirectionAction(BaseAction):
|
class DirectionAction(BaseAction):
|
||||||
@ -24,43 +28,52 @@ class DirectionAction(BaseAction):
|
|||||||
self.xdir = xdir
|
self.xdir = xdir
|
||||||
self.ydir = ydir
|
self.ydir = ydir
|
||||||
|
|
||||||
def apply(self) -> None:
|
def apply(self) -> bool:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
|
||||||
class MovementAction(DirectionAction):
|
class MovementAction(DirectionAction):
|
||||||
def apply(self) -> None:
|
def apply(self) -> bool:
|
||||||
dest_x = self.entity.x + self.xdir
|
dest_x = self.entity.x + self.xdir
|
||||||
dest_y = self.entity.y + self.ydir
|
dest_y = self.entity.y + self.ydir
|
||||||
|
|
||||||
#bounds and collision checks
|
#bounds and collision checks
|
||||||
if not self.entity.floor_map.in_bounds(dest_x, dest_y):
|
if not self.entity.floor_map.in_bounds(dest_x, dest_y):
|
||||||
return
|
return False
|
||||||
if not self.entity.floor_map.tiles["walkable"][dest_x, dest_y]:
|
if not self.entity.floor_map.tiles["walkable"][dest_x, dest_y]:
|
||||||
return
|
return False
|
||||||
if self.entity.floor_map.get_entity_at(dest_x, dest_y, unwalkable_only=True) is not None:
|
if self.entity.floor_map.get_entity_at(dest_x, dest_y, unwalkable_only=True) is not None:
|
||||||
return
|
return False
|
||||||
|
|
||||||
self.entity.set_pos(dest_x, dest_y)
|
self.entity.set_pos(dest_x, dest_y)
|
||||||
|
return True
|
||||||
|
|
||||||
class MeleeAction(DirectionAction):
|
class MeleeAction(DirectionAction):
|
||||||
def apply(self) -> None:
|
def apply(self) -> bool:
|
||||||
dest_x = self.entity.x + self.xdir
|
dest_x = self.entity.x + self.xdir
|
||||||
dest_y = self.entity.y + self.ydir
|
dest_y = self.entity.y + self.ydir
|
||||||
|
|
||||||
target = self.entity.floor_map.get_entity_at(dest_x, dest_y)
|
target = self.entity.floor_map.get_actor_at(dest_x, dest_y)
|
||||||
|
|
||||||
if not target:
|
if not target:
|
||||||
return
|
return False
|
||||||
|
|
||||||
print(f"You kicked the {target.name}, which was funny")
|
damage = self.entity.fighter.attack - target.fighter.defense
|
||||||
|
|
||||||
|
if damage > 0:
|
||||||
|
print(f"{self.entity.name} attacked {target.name} for {damage} damage")
|
||||||
|
target.fighter.current_hp -= damage
|
||||||
|
else:
|
||||||
|
print(f"{self.entity.name} attacked {target.name} but was ineffective")
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
class BumpAction(DirectionAction): #bad name, deal with it
|
class BumpAction(DirectionAction): #bad name, deal with it
|
||||||
def apply(self) -> None:
|
def apply(self) -> bool:
|
||||||
dest_x = self.entity.x + self.xdir
|
dest_x = self.entity.x + self.xdir
|
||||||
dest_y = self.entity.y + self.ydir
|
dest_y = self.entity.y + self.ydir
|
||||||
|
|
||||||
if self.entity.floor_map.get_entity_at(dest_x, dest_y):
|
if self.entity.floor_map.get_entity_at(dest_x, dest_y, unwalkable_only=True):
|
||||||
return MeleeAction(self.entity, self.xdir, self.ydir).apply()
|
return MeleeAction(self.entity, self.xdir, self.ydir).apply()
|
||||||
else:
|
else:
|
||||||
return MovementAction(self.entity, self.xdir, self.ydir).apply()
|
return MovementAction(self.entity, self.xdir, self.ydir).apply()
|
||||||
|
@ -13,6 +13,10 @@ class Fighter(BaseComponent):
|
|||||||
self.attack = attack
|
self.attack = attack
|
||||||
self.defense = defense
|
self.defense = defense
|
||||||
|
|
||||||
|
@property
|
||||||
|
def maximum_hp(self) -> int:
|
||||||
|
return self._maximum_hp
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def current_hp(self) -> int:
|
def current_hp(self) -> int:
|
||||||
return self._current_hp
|
return self._current_hp
|
||||||
@ -20,4 +24,21 @@ class Fighter(BaseComponent):
|
|||||||
@current_hp.setter
|
@current_hp.setter
|
||||||
def current_hp(self, value: int) -> None:
|
def current_hp(self, value: int) -> None:
|
||||||
self._current_hp = max(0, min(value, self._maximum_hp))
|
self._current_hp = max(0, min(value, self._maximum_hp))
|
||||||
|
if self.current_hp <= 0:
|
||||||
|
self.die_and_despawn()
|
||||||
|
|
||||||
|
|
||||||
|
def die_and_despawn(self) -> None:
|
||||||
|
if self.entity is self.entity.floor_map.engine.player and self.entity.ai:
|
||||||
|
from event_handler import GameOverEventHandler
|
||||||
|
self.entity.floor_map.engine.event_handler = GameOverEventHandler(self.entity.floor_map.engine)
|
||||||
|
print("You died")
|
||||||
|
|
||||||
|
else:
|
||||||
|
print(f"The {self.entity.name} died")
|
||||||
|
|
||||||
|
self.entity.char = "%"
|
||||||
|
self.entity.color = (191, 0, 0)
|
||||||
|
self.entity.walkable = True
|
||||||
|
self.entity.ai = None
|
||||||
|
self.entity.name = f"Dead {self.entity.name}"
|
||||||
|
@ -7,8 +7,8 @@ from floor_map import FloorMap #TODO: replace with "DungeonMap"
|
|||||||
|
|
||||||
class Engine:
|
class Engine:
|
||||||
def __init__(self, floor_map: FloorMap):
|
def __init__(self, floor_map: FloorMap):
|
||||||
from event_handler import EventHandler
|
from event_handler import InGameEventHandler
|
||||||
self.event_handler = EventHandler(self)
|
self.event_handler = InGameEventHandler(self)
|
||||||
self.floor_map = floor_map
|
self.floor_map = floor_map
|
||||||
self.floor_map.engine = self #references everywhere!
|
self.floor_map.engine = self #references everywhere!
|
||||||
|
|
||||||
@ -18,6 +18,13 @@ class Engine:
|
|||||||
#kick off the render
|
#kick off the render
|
||||||
self.update_fov()
|
self.update_fov()
|
||||||
|
|
||||||
|
def run_loop(self, context: Context, console: Console) -> None:
|
||||||
|
while True:
|
||||||
|
if self.event_handler.handle_events():
|
||||||
|
self.handle_entities()
|
||||||
|
|
||||||
|
self.handle_rendering(context, console)
|
||||||
|
|
||||||
def handle_entities(self) -> None:
|
def handle_entities(self) -> None:
|
||||||
self.update_fov() #knowing the FOV lets entities mess with it
|
self.update_fov() #knowing the FOV lets entities mess with it
|
||||||
|
|
||||||
@ -30,7 +37,11 @@ class Engine:
|
|||||||
#map and all entities within
|
#map and all entities within
|
||||||
self.floor_map.render(console)
|
self.floor_map.render(console)
|
||||||
|
|
||||||
#TODO: UI
|
#UI
|
||||||
|
console.print(
|
||||||
|
x=1, y=47,
|
||||||
|
string=f"HP: {self.player.fighter.current_hp}/{self.player.fighter.current_hp}",
|
||||||
|
)
|
||||||
|
|
||||||
#send to the screen
|
#send to the screen
|
||||||
context.present(console)
|
context.present(console)
|
||||||
|
@ -25,7 +25,7 @@ class Entity:
|
|||||||
self.walkable = walkable
|
self.walkable = walkable
|
||||||
self.floor_map = floor_map
|
self.floor_map = floor_map
|
||||||
|
|
||||||
def spawn(self: T, x: int, y: int, floor_map):
|
def spawn(self, x: int, y: int, floor_map):
|
||||||
clone = copy.deepcopy(self)
|
clone = copy.deepcopy(self)
|
||||||
clone.x = x
|
clone.x = x
|
||||||
clone.y = y
|
clone.y = y
|
||||||
|
@ -8,7 +8,7 @@ player = Actor(
|
|||||||
name = "Player",
|
name = "Player",
|
||||||
walkable = False,
|
walkable = False,
|
||||||
ai_class = BaseAI,
|
ai_class = BaseAI,
|
||||||
fighter = Fighter(hp = 10, attack = 2, defense = 2),
|
fighter = Fighter(hp = 10, attack = 2, defense = 0),
|
||||||
)
|
)
|
||||||
|
|
||||||
#gobbos
|
#gobbos
|
||||||
|
@ -2,46 +2,115 @@ from typing import Optional
|
|||||||
|
|
||||||
import tcod
|
import tcod
|
||||||
|
|
||||||
from actions import BaseAction, QuitAction, BumpAction
|
from actions import BaseAction, QuitAction, BumpAction, WaitAction
|
||||||
from engine import Engine
|
from engine import Engine
|
||||||
|
|
||||||
|
#input options
|
||||||
|
MOVE_KEYS = {
|
||||||
|
#arrow keys
|
||||||
|
tcod.event.KeySym.UP: (0, -1),
|
||||||
|
tcod.event.KeySym.DOWN: (0, 1),
|
||||||
|
tcod.event.KeySym.LEFT: (-1, 0),
|
||||||
|
tcod.event.KeySym.RIGHT: (1, 0),
|
||||||
|
|
||||||
|
tcod.event.KeySym.HOME: (-1, -1),
|
||||||
|
tcod.event.KeySym.END: (-1, 1),
|
||||||
|
tcod.event.KeySym.PAGEUP: (1, -1),
|
||||||
|
tcod.event.KeySym.PAGEDOWN: (1, 1),
|
||||||
|
|
||||||
|
#numpad keys
|
||||||
|
tcod.event.KeySym.KP_1: (-1, 1),
|
||||||
|
tcod.event.KeySym.KP_2: (0, 1),
|
||||||
|
tcod.event.KeySym.KP_3: (1, 1),
|
||||||
|
tcod.event.KeySym.KP_4: (-1, 0),
|
||||||
|
|
||||||
|
tcod.event.KeySym.KP_6: (1, 0),
|
||||||
|
tcod.event.KeySym.KP_7: (-1, -1),
|
||||||
|
tcod.event.KeySym.KP_8: (0, -1),
|
||||||
|
tcod.event.KeySym.KP_9: (1, -1),
|
||||||
|
|
||||||
|
#vi key mapping
|
||||||
|
tcod.event.KeySym.h: (-1, 0),
|
||||||
|
tcod.event.KeySym.j: (0, 1),
|
||||||
|
tcod.event.KeySym.k: (0, -1),
|
||||||
|
tcod.event.KeySym.l: (1, 0),
|
||||||
|
|
||||||
|
tcod.event.KeySym.y: (-1, -1),
|
||||||
|
tcod.event.KeySym.u: (1, -1),
|
||||||
|
tcod.event.KeySym.b: (-1, 1),
|
||||||
|
tcod.event.KeySym.n: (1, 1),
|
||||||
|
}
|
||||||
|
|
||||||
|
WAIT_KEYS = {
|
||||||
|
tcod.event.KeySym.PERIOD,
|
||||||
|
tcod.event.KeySym.KP_5,
|
||||||
|
tcod.event.KeySym.CLEAR,
|
||||||
|
}
|
||||||
|
|
||||||
#event handler is one part of the engine
|
#event handler is one part of the engine
|
||||||
class EventHandler(tcod.event.EventDispatch[BaseAction]):
|
class EventHandler(tcod.event.EventDispatch[BaseAction]):
|
||||||
def __init__(self, engine: Engine):
|
def __init__(self, engine: Engine):
|
||||||
super().__init__()
|
super().__init__()
|
||||||
self.engine = engine
|
self.engine = engine
|
||||||
|
|
||||||
def handle_events(self) -> None:
|
#callbacks
|
||||||
|
def ev_quit(self, event: tcod.event.Quit) -> Optional[BaseAction]:
|
||||||
|
return QuitAction()
|
||||||
|
|
||||||
|
def handle_events(self) -> bool:
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
|
||||||
|
class InGameEventHandler(EventHandler):
|
||||||
|
def handle_events(self) -> bool:
|
||||||
|
result = False
|
||||||
|
|
||||||
for event in tcod.event.wait():
|
for event in tcod.event.wait():
|
||||||
action = self.dispatch(event)
|
action = self.dispatch(event)
|
||||||
|
|
||||||
if action is None:
|
if action is None:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
action.apply() #entity references the engine
|
result |= action.apply() #entity references the engine
|
||||||
|
|
||||||
#callbacks
|
return result
|
||||||
def ev_quit(self, event: tcod.event.Quit) -> Optional[BaseAction]:
|
|
||||||
return QuitAction()
|
|
||||||
|
|
||||||
def ev_keydown(self, event: tcod.event.KeyDown) -> Optional[BaseAction]:
|
def ev_keydown(self, event: tcod.event.KeyDown) -> Optional[BaseAction]:
|
||||||
key = event.sym #SDL stuff, neat.
|
key = event.sym #SDL stuff, neat.
|
||||||
|
|
||||||
player = self.engine.player
|
player = self.engine.player
|
||||||
|
|
||||||
#parse player input
|
#player input
|
||||||
match key:
|
if key == tcod.event.KeySym.ESCAPE:
|
||||||
case tcod.event.KeySym.ESCAPE:
|
|
||||||
return QuitAction()
|
return QuitAction()
|
||||||
|
|
||||||
case tcod.event.KeySym.UP:
|
if key in MOVE_KEYS:
|
||||||
return BumpAction(player, xdir = 0, ydir = -1)
|
xdir, ydir = MOVE_KEYS[key]
|
||||||
case tcod.event.KeySym.DOWN:
|
return BumpAction(player, xdir = xdir, ydir = ydir)
|
||||||
return BumpAction(player, xdir = 0, ydir = 1)
|
|
||||||
case tcod.event.KeySym.LEFT:
|
if key in WAIT_KEYS:
|
||||||
return BumpAction(player, xdir = -1, ydir = 0)
|
return WaitAction(player)
|
||||||
case tcod.event.KeySym.RIGHT:
|
|
||||||
return BumpAction(player, xdir = 1, ydir = 0)
|
|
||||||
|
class GameOverEventHandler(EventHandler):
|
||||||
|
def handle_events(self) -> bool:
|
||||||
|
result = False
|
||||||
|
|
||||||
|
for event in tcod.event.wait():
|
||||||
|
action = self.dispatch(event)
|
||||||
|
|
||||||
|
if action is None:
|
||||||
|
continue
|
||||||
|
|
||||||
|
result |= action.apply()
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
def ev_keydown(self, event: tcod.event.KeyDown) -> Optional[BaseAction]:
|
||||||
|
key = event.sym #SDL stuff, neat.
|
||||||
|
|
||||||
|
#player input
|
||||||
|
if key == tcod.event.KeySym.ESCAPE:
|
||||||
|
return QuitAction()
|
||||||
|
|
||||||
case _:
|
|
||||||
return None
|
return None
|
@ -48,7 +48,7 @@ class FloorMap:
|
|||||||
|
|
||||||
def get_actor_at(self, x: int, y: int) -> Optional[Actor]:
|
def get_actor_at(self, x: int, y: int) -> Optional[Actor]:
|
||||||
for actor in self.actors:
|
for actor in self.actors:
|
||||||
if actor.x == x and actor.y == y:
|
if actor.x == x and actor.y == y and actor.is_alive():
|
||||||
return actor
|
return actor
|
||||||
|
|
||||||
return None
|
return None
|
||||||
@ -60,6 +60,18 @@ class FloorMap:
|
|||||||
default = tile_types.SHROUD
|
default = tile_types.SHROUD
|
||||||
)
|
)
|
||||||
|
|
||||||
for entity in self.entities:
|
#render the dead stuff below the alive stuff
|
||||||
|
alive = (entity for entity in self.entities if entity.is_alive())
|
||||||
|
dead = (entity for entity in self.entities if not entity.is_alive())
|
||||||
|
|
||||||
|
for entity in dead:
|
||||||
if self.visible[entity.x, entity.y]:
|
if self.visible[entity.x, entity.y]:
|
||||||
console.print(entity.x, entity.y, entity.char, fg=entity.color)
|
console.print(entity.x, entity.y, entity.char, fg=entity.color)
|
||||||
|
|
||||||
|
for entity in alive:
|
||||||
|
if self.visible[entity.x, entity.y]:
|
||||||
|
console.print(entity.x, entity.y, entity.char, fg=entity.color)
|
||||||
|
|
||||||
|
#print the player above everything else for clarity
|
||||||
|
if self.player:
|
||||||
|
console.print(self.player.x, self.player.y, self.player.char, fg=self.player.color) #TODO: I didn't realize the render order would be fixed in the tutorial
|
@ -18,7 +18,7 @@ def main() -> None:
|
|||||||
|
|
||||||
console = tcod.console.Console(
|
console = tcod.console.Console(
|
||||||
width = w,
|
width = w,
|
||||||
height = h,
|
height = h + 5,
|
||||||
order = "F"
|
order = "F"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -27,11 +27,8 @@ def main() -> None:
|
|||||||
floor_map = generate_floor_map(80, 45, 10, 10)
|
floor_map = generate_floor_map(80, 45, 10, 10)
|
||||||
)
|
)
|
||||||
|
|
||||||
# game loop
|
#game loop that never returns
|
||||||
while True:
|
engine.run_loop(context, console)
|
||||||
engine.event_handler.handle_events()
|
|
||||||
engine.handle_entities()
|
|
||||||
engine.handle_rendering(context, console)
|
|
||||||
|
|
||||||
# this seems odd to me
|
# this seems odd to me
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
Loading…
x
Reference in New Issue
Block a user