113 lines
2.8 KiB
Python
113 lines
2.8 KiB
Python
from __future__ import annotations
|
|
from typing import TYPE_CHECKING
|
|
|
|
import colors
|
|
|
|
from floor_map import FloorMap
|
|
|
|
if TYPE_CHECKING:
|
|
from engine import Engine
|
|
from entity import Entity
|
|
|
|
class BaseAction:
|
|
entity: Entity #the entity to which this action applies
|
|
|
|
def __init__(self, entity):
|
|
self.entity = entity
|
|
|
|
def apply(self) -> bool:
|
|
"""return True if the game state should be progressed"""
|
|
raise NotImplementedError()
|
|
|
|
|
|
class QuitAction(BaseAction):
|
|
def __init__(self):
|
|
"""override the base __init__, as no entity is needed"""
|
|
pass
|
|
|
|
def apply(self) -> bool:
|
|
raise SystemExit()
|
|
|
|
|
|
class WaitAction(BaseAction):
|
|
def apply(self) -> bool:
|
|
return True
|
|
|
|
|
|
class MovementAction(BaseAction):
|
|
"""Move an Entity within the map"""
|
|
def __init__(self, entity, xdir: int, ydir: int):
|
|
super().__init__(entity)
|
|
self.xdir = xdir
|
|
self.ydir = ydir
|
|
|
|
def apply(self) -> bool:
|
|
dest_x = self.entity.x + self.xdir
|
|
dest_y = self.entity.y + self.ydir
|
|
|
|
floor_map: FloorMap = self.entity.floor_map
|
|
|
|
#bounds and collision checks
|
|
if not floor_map.in_bounds(dest_x, dest_y):
|
|
return False
|
|
if not floor_map.tiles["walkable"][dest_x, dest_y]:
|
|
return False
|
|
if floor_map.get_entity_at(dest_x, dest_y, unwalkable_only=True) is not None:
|
|
return False
|
|
|
|
self.entity.set_pos(dest_x, dest_y)
|
|
return True
|
|
|
|
class MeleeAction(BaseAction):
|
|
"""Melee attack from the Entity towards a target"""
|
|
def __init__(self, entity, xdir: int, ydir: int):
|
|
super().__init__(entity)
|
|
self.xdir = xdir
|
|
self.ydir = ydir
|
|
|
|
def apply(self) -> bool:
|
|
dest_x = self.entity.x + self.xdir
|
|
dest_y = self.entity.y + self.ydir
|
|
|
|
target = self.entity.floor_map.get_entity_at(dest_x, dest_y, unwalkable_only=True)
|
|
|
|
if not target or not target.stats:
|
|
return False
|
|
|
|
#TODO: better combat system
|
|
|
|
#calculate damage
|
|
damage = self.entity.stats.attack - target.stats.defense
|
|
|
|
#calculate message output
|
|
engine: Engine = self.entity.floor_map.engine
|
|
msg_text = f"{self.entity.name} attacked {target.name}"
|
|
|
|
if damage > 0:
|
|
msg_text += f" for {damage} damage"
|
|
else:
|
|
msg_text += f" but was ineffective"
|
|
|
|
engine.message_log.add_message(text = msg_text)
|
|
|
|
#actually applying the change here, so the player's death event is at the bottom of the message log
|
|
target.stats.current_hp -= damage
|
|
|
|
return True
|
|
|
|
class BumpAction(BaseAction):
|
|
"""Move an Entity within the map, or attack a target if one is found"""
|
|
def __init__(self, entity, xdir: int, ydir: int):
|
|
super().__init__(entity)
|
|
self.xdir = xdir
|
|
self.ydir = ydir
|
|
|
|
def apply(self) -> bool:
|
|
dest_x = self.entity.x + self.xdir
|
|
dest_y = self.entity.y + self.ydir
|
|
|
|
if self.entity.floor_map.get_entity_at(dest_x, dest_y, unwalkable_only=True):
|
|
return MeleeAction(self.entity, self.xdir, self.ydir).apply()
|
|
else:
|
|
return MovementAction(self.entity, self.xdir, self.ydir).apply()
|