2022-01-07 14:35:47 -05:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-01-10 13:21:17 -05:00
|
|
|
from typing import Iterable, Iterator, Optional, TYPE_CHECKING
|
2022-01-07 14:35:47 -05:00
|
|
|
|
2022-01-06 13:27:15 -05:00
|
|
|
import numpy as np # type: ignore
|
2022-01-06 11:41:34 -05:00
|
|
|
from tcod.console import Console
|
|
|
|
|
2022-01-12 16:24:09 -05:00
|
|
|
from entity import Actor, Item
|
2022-01-06 11:41:34 -05:00
|
|
|
import tile_types
|
|
|
|
|
2022-01-07 14:35:47 -05:00
|
|
|
if TYPE_CHECKING:
|
2022-01-07 16:25:07 -05:00
|
|
|
from engine import Engine
|
2022-01-07 14:35:47 -05:00
|
|
|
from entity import Entity
|
|
|
|
|
2022-01-06 13:27:15 -05:00
|
|
|
|
2022-01-06 11:41:34 -05:00
|
|
|
class GameMap:
|
2022-01-07 16:25:07 -05:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
engine: Engine,
|
|
|
|
width: int,
|
|
|
|
height: int,
|
|
|
|
entities: Iterable[Entity] = ()
|
|
|
|
):
|
|
|
|
self.engine = engine
|
2022-01-06 13:27:15 -05:00
|
|
|
self.width, self.height = width, height
|
2022-01-07 15:52:53 -05:00
|
|
|
self.entities = set(entities)
|
2022-01-06 13:27:15 -05:00
|
|
|
self.tiles = np.full((width, height), fill_value=tile_types.wall, order="F")
|
2022-01-06 11:41:34 -05:00
|
|
|
|
2022-01-07 16:25:07 -05:00
|
|
|
self.visible = np.full(
|
|
|
|
(width, height),
|
|
|
|
fill_value=False,
|
|
|
|
order="F"
|
|
|
|
) # Tiles the player can currently see
|
|
|
|
self.explored = np.full(
|
|
|
|
(width, height),
|
|
|
|
fill_value=False,
|
|
|
|
order="F"
|
|
|
|
) # Tiles the player has seen before
|
2022-01-07 14:18:47 -05:00
|
|
|
|
2022-01-20 16:33:19 -05:00
|
|
|
self.downstairs_location = (0, 0)
|
|
|
|
|
2022-01-12 13:45:52 -05:00
|
|
|
@property
|
|
|
|
def gamemap(self) -> GameMap:
|
|
|
|
return self
|
|
|
|
|
2022-01-10 13:21:17 -05:00
|
|
|
@property
|
|
|
|
def actors(self) -> Iterator[Actor]:
|
|
|
|
"""Iterate over this map's living actors."""
|
|
|
|
yield from (
|
|
|
|
entity
|
|
|
|
for entity in self.entities
|
|
|
|
if isinstance(entity, Actor) and entity.is_alive
|
|
|
|
)
|
|
|
|
|
2022-01-12 16:24:09 -05:00
|
|
|
@property
|
|
|
|
def items(self) -> Iterator[Item]:
|
|
|
|
yield from (entity for entity in self.entities if isinstance(entity, Item))
|
|
|
|
|
2022-01-07 16:25:07 -05:00
|
|
|
def get_blocking_entity_at_location(
|
|
|
|
self,
|
|
|
|
location_x: int,
|
|
|
|
location_y: int
|
|
|
|
) -> Optional[Entity]:
|
2022-01-07 15:52:53 -05:00
|
|
|
for entity in self.entities:
|
2022-01-07 16:25:07 -05:00
|
|
|
if (
|
|
|
|
entity.blocks_movement
|
|
|
|
and entity.x == location_x
|
|
|
|
and entity.y == location_y
|
|
|
|
):
|
2022-01-07 15:52:53 -05:00
|
|
|
return entity
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2022-01-10 13:21:17 -05:00
|
|
|
def get_actor_at_location(self, x: int, y: int) -> Optional[Actor]:
|
|
|
|
for actor in self.actors:
|
|
|
|
if actor.x == x and actor.y == y:
|
|
|
|
return actor
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2022-01-06 11:41:34 -05:00
|
|
|
def in_bounds(self, x: int, y: int) -> bool:
|
2022-01-06 13:27:15 -05:00
|
|
|
"""Return True if x and y are inside the bounds of the map."""
|
2022-01-06 11:41:34 -05:00
|
|
|
return 0 <= x < self.width and 0 <= y < self.height
|
|
|
|
|
|
|
|
def render(self, console: Console):
|
2022-01-07 14:18:47 -05:00
|
|
|
"""
|
|
|
|
Renders the map.
|
|
|
|
|
|
|
|
If a tile is in the "visible" array, then draw it with the "light" colors.
|
|
|
|
If it isn't, but it's in the "explored" array, then draw it with the "dark" colors.
|
|
|
|
Otherwise, the default is "SHROUD".
|
|
|
|
:param console:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
console.tiles_rgb[0: self.width, 0: self.height] = np.select(
|
|
|
|
condlist=[self.visible, self.explored],
|
|
|
|
choicelist=[self.tiles["light"], self.tiles["dark"]],
|
2022-01-07 16:25:07 -05:00
|
|
|
default=tile_types.SHROUD,
|
2022-01-07 14:18:47 -05:00
|
|
|
)
|
2022-01-07 14:35:47 -05:00
|
|
|
|
2022-01-10 14:09:31 -05:00
|
|
|
entities_sorted_for_rendering = sorted(
|
|
|
|
self.entities, key=lambda x: x.render_order.value
|
|
|
|
)
|
|
|
|
|
|
|
|
for entity in entities_sorted_for_rendering:
|
2022-01-07 14:35:47 -05:00
|
|
|
# Only print entities that are in the FOV
|
|
|
|
if self.visible[entity.x, entity.y]:
|
2022-01-10 14:09:31 -05:00
|
|
|
console.print(
|
|
|
|
x=entity.x,
|
|
|
|
y=entity.y,
|
|
|
|
string=entity.char,
|
|
|
|
fg=entity.color
|
|
|
|
)
|
2022-01-20 16:33:19 -05:00
|
|
|
|
|
|
|
|
|
|
|
class GameWorld:
|
|
|
|
"""
|
|
|
|
Holds the settings for the GameMap, and generates new maps when moving down the stairs.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
*,
|
|
|
|
engine: Engine,
|
|
|
|
map_width: int,
|
|
|
|
map_height: int,
|
|
|
|
max_rooms: int,
|
|
|
|
room_min_size: int,
|
|
|
|
room_max_size: int,
|
|
|
|
max_monsters_per_room: int,
|
|
|
|
max_items_per_room: int,
|
|
|
|
current_floor: int = 0
|
|
|
|
):
|
|
|
|
self.engine = engine
|
|
|
|
|
|
|
|
self.map_width = map_width
|
|
|
|
self.map_height = map_height
|
|
|
|
|
|
|
|
self.max_rooms = max_rooms
|
|
|
|
|
|
|
|
self.room_min_size = room_min_size
|
|
|
|
self.room_max_size = room_max_size
|
|
|
|
|
|
|
|
self.max_monsters_per_room = max_monsters_per_room
|
|
|
|
self.max_items_per_room = max_items_per_room
|
|
|
|
|
|
|
|
self.current_floor = current_floor
|
|
|
|
|
|
|
|
def generate_floor(self) -> None:
|
|
|
|
from procgen import generate_dungeon
|
|
|
|
|
|
|
|
self.current_floor += 1
|
|
|
|
|
|
|
|
self.engine.game_map = generate_dungeon(
|
|
|
|
max_rooms=self.max_rooms,
|
|
|
|
room_min_size=self.room_min_size,
|
|
|
|
room_max_size=self.room_max_size,
|
|
|
|
map_width=self.map_width,
|
|
|
|
map_height=self.map_height,
|
|
|
|
max_monsters_per_room=self.max_monsters_per_room,
|
|
|
|
max_items_per_room=self.max_items_per_room,
|
|
|
|
engine=self.engine,
|
|
|
|
)
|