1
0
python-roguelike/actions.py

114 lines
3.3 KiB
Python
Raw Normal View History

2022-01-06 11:56:08 -05:00
from __future__ import annotations
2022-01-11 16:08:57 -05:00
from typing import overload, Optional, Tuple, TYPE_CHECKING
import color
2022-01-06 11:56:08 -05:00
if TYPE_CHECKING:
from engine import Engine
from entity import Actor, Entity
2022-01-06 11:56:08 -05:00
2022-01-07 14:18:47 -05:00
2022-01-06 11:09:43 -05:00
class Action:
def __init__(self, entity: Actor) -> None:
super().__init__()
self.entity = entity
@property
def engine(self) -> Engine:
"""Return the engine this action belongs to."""
return self.entity.gamemap.engine
2022-01-07 16:48:49 -05:00
@overload
def perform(self) -> None:
2022-01-06 11:56:08 -05:00
"""Perform this action with the objects needed to determine its scope.
`self.engine` is the scope this action is being performed in.
2022-01-06 11:56:08 -05:00
`self.entity` is the object performing the action.
2022-01-06 11:56:08 -05:00
This method must be overwritten by Action subclasses.
"""
2022-01-06 11:09:43 -05:00
class EscapeAction(Action):
def perform(self) -> None:
2022-01-06 11:56:08 -05:00
raise SystemExit()
2022-01-06 11:09:43 -05:00
2022-01-10 13:21:17 -05:00
class WaitAction(Action):
def perform(self) -> None:
pass
2022-01-07 15:52:53 -05:00
class ActionWithDirection(Action):
def __init__(self, entity: Actor, dx: int, dy: int):
super().__init__(entity)
2022-01-06 11:09:43 -05:00
self.dx = dx
self.dy = dy
2022-01-06 11:56:08 -05:00
@property
def dest_xy(self) -> Tuple[int, int]:
"""Returns this action's destination."""
return self.entity.x + self.dx, self.entity.y + self.dy
@property
def blocking_entity(self) -> Optional[Entity]:
"""Return the blocking entity at this action's destination."""
return self.engine.game_map.get_blocking_entity_at_location(*self.dest_xy)
@property
def target_actor(self) -> Optional[Actor]:
"""Return the actor at this action's destination."""
return self.engine.game_map.get_actor_at_location(*self.dest_xy)
2022-01-07 15:52:53 -05:00
class MeleeAction(ActionWithDirection):
def perform(self) -> None:
target = self.target_actor
2022-01-07 15:52:53 -05:00
if not target:
return # No entity to attack.
damage = self.entity.fighter.power - target.fighter.defense
attack_desc = f"{self.entity.name.capitalize()} attacks {target.name}"
2022-01-11 16:08:57 -05:00
if self.entity is self.engine.player:
attack_color = color.player_atk
else:
attack_color = color.enemy_atk
if damage > 0:
2022-01-11 16:08:57 -05:00
self.engine.message_log.add_message(
f"{attack_desc} for {damage} hit points.",
attack_color,
)
target.fighter.hp -= damage
else:
2022-01-11 16:08:57 -05:00
self.engine.message_log.add_message(
f"{attack_desc} but does no damage.",
attack_color,
)
2022-01-07 15:52:53 -05:00
class MovementAction(ActionWithDirection):
def perform(self) -> None:
dest_x, dest_y = self.dest_xy
2022-01-06 11:56:08 -05:00
if not self.engine.game_map.in_bounds(dest_x, dest_y):
2022-01-07 14:18:47 -05:00
return # Destination is out of bounds
if not self.engine.game_map.tiles["walkable"][dest_x, dest_y]:
2022-01-07 14:18:47 -05:00
return # Destination is blocked by a tile.
if self.engine.game_map.get_blocking_entity_at_location(dest_x, dest_y):
2022-01-07 15:52:53 -05:00
return # Destination is blocked by an entity.
2022-01-06 11:56:08 -05:00
self.entity.move(self.dx, self.dy)
2022-01-07 15:52:53 -05:00
class BumpAction(ActionWithDirection):
def perform(self) -> None:
if self.target_actor:
return MeleeAction(self.entity, self.dx, self.dy).perform()
2022-01-07 15:52:53 -05:00
else:
return MovementAction(self.entity, self.dx, self.dy).perform()