Stepwise/source/actions.py

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()