from __future__ import annotations from typing import List, Optional, TYPE_CHECKING import colors from floor_map import FloorMap from inventory import Inventory 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 perform(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 perform(self) -> bool: raise SystemExit() class WaitAction(BaseAction): def perform(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 perform(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_all_entities_at(dest_x, dest_y, unwalkable_only=True): 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 perform(self) -> bool: dest_x = self.entity.x + self.xdir dest_y = self.entity.y + self.ydir targets = self.entity.floor_map.get_all_entities_at(dest_x, dest_y, unwalkable_only=True) if not targets: return False target = targets.pop() 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 perform(self) -> bool: dest_x = self.entity.x + self.xdir dest_y = self.entity.y + self.ydir if self.entity.floor_map.get_all_entities_at(dest_x, dest_y, unwalkable_only=True): return MeleeAction(self.entity, self.xdir, self.ydir).perform() else: return MovementAction(self.entity, self.xdir, self.ydir).perform() class PickupAction(BaseAction): """Pickup an item at the entity's location""" def perform(self) -> bool: x = self.entity.x y = self.entity.y floor_map: FloorMap = self.entity.floor_map engine: Engine = floor_map.engine item_stack: List[Entity] = floor_map.get_all_entities_at(x, y, items_only=True) if len(item_stack) == 0: return False elif len(item_stack) == 1: floor_map.entities.remove(item_stack[0]) self.entity.inventory.insert(item_stack[0]) engine.message_log.add_message(f"you picked up a(n) {item_stack[0].name}", color=colors.terminal_light) else: from event_handlers import OptionSelector #circular imports are a pain #build an options list options: List[str] = [] for item in item_stack: options.append(item.name) engine.event_handler = OptionSelector( engine=floor_map.engine, parent_handler=engine.event_handler, title="Pick Up Item", options=options, callback=lambda x: self.pickup_callback(engine, floor_map, self.entity, item_stack[x]) ) return True #utils def pickup_callback(self, engine: Engine, floor_map: FloorMap, entity: Entity, item: Entity) -> None: floor_map.entities.remove(item) entity.inventory.insert(item) engine.message_log.add_message(f"you picked up a(n) {item.name}", color=colors.terminal_light) class DropAction(BaseAction): """Drop an item from an entity's inventory at the entity's location""" index: int def __init__(self, entity: Entity, index: int): """override the base __init__""" super().__init__(entity) self.index = index def perform(self) -> bool: x = self.entity.x y = self.entity.y inventory: Inventory = self.entity.inventory floor_map: FloorMap = self.entity.floor_map engine: Engine = floor_map.engine item: Entity = inventory.withdraw(self.index) item.x = x item.y = y floor_map.entities.add(item) engine.message_log.add_message(f"you dropped a(n) {item.name}", color=colors.terminal_light) return True