2022-01-07 15:52:53 -05:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import copy
|
2022-01-10 13:21:17 -05:00
|
|
|
from typing import Optional, Tuple, Type, TypeVar, TYPE_CHECKING
|
2022-01-07 15:52:53 -05:00
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
2022-01-10 13:21:17 -05:00
|
|
|
from components.ai import BaseAI
|
|
|
|
from components.fighter import Fighter
|
2022-01-07 15:52:53 -05:00
|
|
|
from game_map import GameMap
|
|
|
|
|
|
|
|
T = TypeVar("T", bound="Entity")
|
2022-01-06 11:19:56 -05:00
|
|
|
|
|
|
|
|
|
|
|
class Entity:
|
|
|
|
"""
|
|
|
|
A generic object to represent players, enemies, items, etc.
|
|
|
|
"""
|
2022-01-07 15:52:53 -05:00
|
|
|
|
2022-01-07 16:25:07 -05:00
|
|
|
gamemap: GameMap
|
|
|
|
|
2022-01-07 15:52:53 -05:00
|
|
|
def __init__(
|
|
|
|
self,
|
2022-01-07 16:25:07 -05:00
|
|
|
gamemap: Optional[GameMap] = None,
|
2022-01-07 15:52:53 -05:00
|
|
|
x: int = 0,
|
|
|
|
y: int = 0,
|
|
|
|
char: str = "?",
|
|
|
|
color: Tuple[int, int, int] = (255, 255, 255),
|
|
|
|
name: str = "<Unnamed>",
|
|
|
|
blocks_movement: bool = False,
|
|
|
|
):
|
2022-01-06 11:19:56 -05:00
|
|
|
self.x = x
|
|
|
|
self.y = y
|
|
|
|
self.char = char
|
|
|
|
self.color = color
|
2022-01-07 15:52:53 -05:00
|
|
|
self.name = name
|
|
|
|
self.blocks_movement = blocks_movement
|
2022-01-07 16:25:07 -05:00
|
|
|
if gamemap:
|
|
|
|
# If gamemap isn't provided now, it will be later.
|
|
|
|
self.gamemap = gamemap
|
|
|
|
gamemap.entities.add(self)
|
2022-01-07 15:52:53 -05:00
|
|
|
|
|
|
|
def spawn(self: T, gamemap: GameMap, x: int, y: int) -> T:
|
|
|
|
"""Spawn a copy of this instance at the given location."""
|
|
|
|
clone = copy.deepcopy(self)
|
|
|
|
clone.x = x
|
|
|
|
clone.y = y
|
2022-01-07 16:25:07 -05:00
|
|
|
clone.gamemap = gamemap
|
2022-01-07 15:52:53 -05:00
|
|
|
gamemap.entities.add(clone)
|
|
|
|
return clone
|
2022-01-06 11:19:56 -05:00
|
|
|
|
2022-01-07 16:25:07 -05:00
|
|
|
def place(self, x: int, y: int, gamemap: Optional[GameMap] = None) -> None:
|
|
|
|
"""Place this entity at a new location. Handles moving across GameMaps."""
|
|
|
|
self.x = x
|
|
|
|
self.y = y
|
|
|
|
if gamemap:
|
|
|
|
if hasattr(self, "gamemap"): # Possibly uninitialized
|
|
|
|
self.gamemap.entities.remove(self)
|
|
|
|
|
|
|
|
self.gamemap = gamemap
|
|
|
|
gamemap.entities.add(self)
|
|
|
|
|
2022-01-06 11:19:56 -05:00
|
|
|
def move(self, dx: int, dy: int):
|
|
|
|
# Move the entity by a given amount
|
|
|
|
self.x += dx
|
|
|
|
self.y += dy
|
2022-01-10 13:21:17 -05:00
|
|
|
|
|
|
|
|
|
|
|
class Actor(Entity):
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
*,
|
|
|
|
x: int = 0,
|
|
|
|
y: int = 0,
|
|
|
|
char: str = "?",
|
|
|
|
color: Tuple[int, int, int] = (255, 255, 255),
|
|
|
|
name: str = "<Unamed>",
|
|
|
|
ai_cls: Type[BaseAI],
|
|
|
|
fighter: Fighter
|
|
|
|
):
|
|
|
|
super().__init__(
|
|
|
|
x=x,
|
|
|
|
y=y,
|
|
|
|
char=char,
|
|
|
|
color=color,
|
|
|
|
name=name,
|
|
|
|
blocks_movement=True
|
|
|
|
)
|
|
|
|
|
|
|
|
self.ai = Optional[BaseAI] = ai_cls(self)
|
|
|
|
|
|
|
|
self.fighter = fighter
|
|
|
|
self.fighter.entity = self
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_alive(self) -> bool:
|
|
|
|
"""Returns True as long as this actor can perform actions."""
|
|
|
|
return bool(self.ai)
|