This commit is contained in:
Tyrel Souza 2022-01-02 12:19:43 -05:00
parent ceb000f229
commit ea986af048
11 changed files with 38956 additions and 54 deletions

2
.gitignore vendored Normal file
View File

@ -0,0 +1,2 @@
venv/
__pycache__/

View File

@ -23,14 +23,36 @@ class EscapeAction(Action):
def perform(self, engine: Engine, entity: Entity) -> None: def perform(self, engine: Engine, entity: Entity) -> None:
raise SystemExit() raise SystemExit()
class ActionWithDirection(Action):
class MovementAction(Action):
def __init__(self, dx: int, dy: int): def __init__(self, dx: int, dy: int):
super().__init__() super().__init__()
self.dx = dx self.dx = dx
self.dy = dy self.dy = dy
def perform(self, engine: Engine, entity: Entity) -> None:
raise NotImplementedError()
class MeleeAction(ActionWithDirection):
def perform(self, engine: Engine, entity: Entity) -> None:
dest_x = entity.x + self.dx
dest_y = entity.y + self.dy
target = engine.game_map.get_blocking_entity_at_location(dest_x, dest_y)
if not target:
return # no entity to blockj
print(f"You kick {target.name}, much to its annoyance")
class BumpAction(ActionWithDirection):
def perform(self, engine: Engine, entity: Entity) -> None:
dest_x = entity.x + self.dx
dest_y = entity.y + self.dy
if engine.game_map.get_blocking_entity_at_location(dest_x, dest_y):
return MeleeAction(self.dx, self.dy).perform(engine, entity)
else:
return MovementAction(self.dx, self.dy).perform(engine, entity)
class MovementAction(ActionWithDirection):
def perform(self, engine: Engine, entity: Entity) -> None: def perform(self, engine: Engine, entity: Entity) -> None:
dest_x = entity.x + self.dx dest_x = entity.x + self.dx
dest_y = entity.y + self.dy dest_y = entity.y + self.dy
@ -39,5 +61,7 @@ class MovementAction(Action):
return # OOB return # OOB
if not engine.game_map.tiles["walkable"][dest_x, dest_y]: if not engine.game_map.tiles["walkable"][dest_x, dest_y]:
return # can't walk return # can't walk
if engine.game_map.get_blocking_entity_at_location(dest_x, dest_y):
return # Blocked
entity.move(self.dx, self.dy) entity.move(self.dx, self.dy)

View File

@ -1,4 +1,4 @@
from typing import Set, Iterable, Any from typing import Iterable, Any
from tcod.context import Context from tcod.context import Context
from tcod.console import Console from tcod.console import Console
@ -10,19 +10,23 @@ from input_handlers import EventHandler
class Engine: class Engine:
def __init__(self, entities: Set[Entity], event_handler: EventHandler, game_map: GameMap, player: Entity): def __init__(self, event_handler: EventHandler, game_map: GameMap, player: Entity):
self.entities = entities
self.event_handler = event_handler self.event_handler = event_handler
self.player = player self.player = player
self.game_map = game_map self.game_map = game_map
self.update_fov() self.update_fov()
def handle_enemy_turns(self) -> None:
for entity in self.game_map.entities - {self.player}:
print(f'the {entity.name} wonders when it will move')
def handle_events(self, events: Iterable[Any]) -> None: def handle_events(self, events: Iterable[Any]) -> None:
for event in events: for event in events:
action = self.event_handler.dispatch(event) action = self.event_handler.dispatch(event)
if action is None: if action is None:
continue continue
action.perform(self, self.player) action.perform(self, self.player)
self.handle_enemy_turns()
self.update_fov() self.update_fov()
def update_fov(self) -> None: def update_fov(self) -> None:
@ -35,9 +39,6 @@ class Engine:
def render(self, console: Console, context: Context) -> None: def render(self, console: Console, context: Context) -> None:
self.game_map.render(console) self.game_map.render(console)
for entity in self.entities:
if self.game_map.visible[entity.x, entity.y]:
console.print(entity.x, entity.y, entity.char, fg=entity.color)
context.present(console) context.present(console)
console.clear() console.clear()

View File

@ -1,14 +1,41 @@
from typing import Tuple from __future__ import annotations
import copy
from typing import Tuple, TypeVar, TYPE_CHECKING
if TYPE_CHECKING:
from game_map import GameMap
T = TypeVar("T", bound="Entity")
class Entity: class Entity:
""" """
A generic object to represent players, enemies, items, etc. A generic object to represent players, enemies, items, etc.
""" """
def __init__(self, x: int, y: int, char: str, color: Tuple[int, int, int]):
def __init__(self,
x: int = 0,
y: int = 0,
char: str = "?",
color: Tuple[int, int, int] = (255,255,255),
name: str = "<Unnamed>",
blocks_movement: bool = False,
):
self.x = x self.x = x
self.y = y self.y = y
self.char = char self.char = char
self.color = color self.color = color
self.name = name
self.blocks_movement = blocks_movement
def spawn(self: T, gamemap: GameMap, x: int, y: int) -> T:
"""Spawns a copy of this instance at the given location"""
clone = copy.deepcopy(self)
clone.x = x
clone.y = y
gamemap.entities.add(clone)
return clone
def move(self, dx: int, dy: int) -> None: def move(self, dx: int, dy: int) -> None:
self.x += dx self.x += dx

7
entity_factories.py Normal file
View File

@ -0,0 +1,7 @@
from entity import Entity
player = Entity(char="@", color=(255, 255, 255), name="Player", blocks_movement=True)
orc = Entity(char="o", color=(63, 127, 63), name="Orc", blocks_movement=True)
troll = Entity(char="T", color=(0, 127, 0), name="Troll", blocks_movement=True)

View File

@ -1,17 +1,30 @@
from __future__ import annotations
from typing import Iterable, TYPE_CHECKING, Optional
import numpy as np # type: ignore import numpy as np # type: ignore
from tcod.console import Console from tcod.console import Console
import tile_types import tile_types
if TYPE_CHECKING:
from entity import Entity
class GameMap: class GameMap:
def __init__(self, width: int, height: int): def __init__(self, width: int, height: int, entities: Iterable[Entity] = ()):
self.width, self.height = width, height self.width, self.height = width, height
self.entities = set(entities)
self.tiles = np.full((width, height), fill_value=tile_types.wall, order="F") self.tiles = np.full((width, height), fill_value=tile_types.wall, order="F")
self.visible = np.full((width, height), fill_value=False, order="F") self.visible = np.full((width, height), fill_value=False, order="F")
self.explored = np.full((width, height), fill_value=False, order="F") self.explored = np.full((width, height), fill_value=False, order="F")
def get_blocking_entity_at_location(self, location_x: int, location_y: int) -> Optiona9[Entity]:
for entity in self.entities:
if entity.blocks_movement and entity.x == location_x and entity.y == location_y:
return entity
return None
def in_bounds(self, x: int, y: int) -> bool: def in_bounds(self, x: int, y: int) -> bool:
return 0 <= x < self.width and 0 <= y < self.height return 0 <= x < self.width and 0 <= y < self.height
@ -19,5 +32,9 @@ class GameMap:
console.tiles_rgb[0 : self.width, 0 : self.height] = np.select( console.tiles_rgb[0 : self.width, 0 : self.height] = np.select(
condlist=[self.visible, self.explored], condlist=[self.visible, self.explored],
choicelist=[self.tiles["light"], self.tiles["dark"]], choicelist=[self.tiles["light"], self.tiles["dark"]],
default=tile_types.SHROUD default=tile_types.SHROUD,
) )
for entity in self.entities:
if self.visible[entity.x, entity.y]:
console.print(entity.x, entity.y, entity.char, fg=entity.color)

View File

@ -2,7 +2,7 @@ from typing import Optional
import tcod.event import tcod.event
from actions import Action, EscapeAction, MovementAction from actions import Action, EscapeAction, BumpAction
class EventHandler(tcod.event.EventDispatch[Action]): class EventHandler(tcod.event.EventDispatch[Action]):
@ -15,13 +15,13 @@ class EventHandler(tcod.event.EventDispatch[Action]):
key = event.sym key = event.sym
if key == tcod.event.K_UP: if key == tcod.event.K_UP:
action = MovementAction(dx=0, dy=-1) action = BumpAction(dx=0, dy=-1)
elif key == tcod.event.K_DOWN: elif key == tcod.event.K_DOWN:
action = MovementAction(dx=0, dy=1) action = BumpAction(dx=0, dy=1)
elif key == tcod.event.K_LEFT: elif key == tcod.event.K_LEFT:
action = MovementAction(dx=-1, dy=0) action = BumpAction(dx=-1, dy=0)
elif key == tcod.event.K_RIGHT: elif key == tcod.event.K_RIGHT:
action = MovementAction(dx=1, dy=0) action = BumpAction(dx=1, dy=0)
elif key == tcod.event.K_ESCAPE: elif key == tcod.event.K_ESCAPE:
action = EscapeAction() action = EscapeAction()

16
main.py
View File

@ -1,7 +1,8 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
import tcod import tcod
import copy
import entity_factories
from entity import Entity
from engine import Engine from engine import Engine
from input_handlers import EventHandler from input_handlers import EventHandler
from procgen import generate_dungeon from procgen import generate_dungeon
@ -18,15 +19,15 @@ def main():
room_min_size = 6 room_min_size = 6
max_rooms = 30 max_rooms = 30
max_monsters_per_room = 2
tileset = tcod.tileset.load_tilesheet( tileset = tcod.tileset.load_tilesheet(
"dejavu10x10_gs_tc.png", 32, 8, tcod.tileset.CHARMAP_TCOD "dejavu10x10_gs_tc.png", 32, 8, tcod.tileset.CHARMAP_TCOD
) )
event_handler = EventHandler() event_handler = EventHandler()
player = Entity(screen_width // 2, screen_height // 2, "@", (255, 255, 255)) player = copy.deepcopy(entity_factories.player)
npc = Entity(screen_width // 2 - 5, screen_height // 2, "@", (255, 255, 0))
entities = {npc, player}
game_map = generate_dungeon( game_map = generate_dungeon(
max_rooms=max_rooms, max_rooms=max_rooms,
@ -34,17 +35,18 @@ def main():
room_max_size=room_max_size, room_max_size=room_max_size,
map_width=map_width, map_width=map_width,
map_height=map_height, map_height=map_height,
player=player max_monsters_per_room=max_monsters_per_room,
player=player,
) )
engine = Engine(entities=entities, event_handler=event_handler, game_map=game_map, player=player) engine = Engine(event_handler=event_handler, game_map=game_map, player=player)
with tcod.context.new_terminal( with tcod.context.new_terminal(
screen_width, screen_width,
screen_height, screen_height,
tileset=tileset, tileset=tileset,
title="Yet Another Roguelike Tutorial", title="Yet Another Roguelike Tutorial",
vsync=True vsync=True,
) as context: ) as context:
root_console = tcod.Console(screen_width, screen_height, order="F") root_console = tcod.Console(screen_width, screen_height, order="F")
while True: while True:

View File

@ -6,6 +6,7 @@ from typing import Tuple, Iterator, TYPE_CHECKING, List
import tcod.los import tcod.los
import tile_types import tile_types
import entity_factories
from game_map import GameMap from game_map import GameMap
if TYPE_CHECKING: if TYPE_CHECKING:
@ -54,6 +55,19 @@ def tunnel_between(
for x, y in tcod.los.bresenham((corner_x, corner_y), (x2, y2)).tolist(): for x, y in tcod.los.bresenham((corner_x, corner_y), (x2, y2)).tolist():
yield x, y yield x, y
def place_entities(
room: RectangularRoom, dungeon: GameMap, maximum_monsters: int
) -> None:
number_of_monsters = random.randint(0, maximum_monsters)
for i in range(number_of_monsters):
x = random.randint(room.x1 + 1, room.x2 -1)
y = random.randint(room.y1 + 1, room.y2 -1)
if not any(entity.x == x and entity.y == y for entity in dungeon.entities):
if random.random() <= 0.8:
entity_factories.orc.spawn(dungeon, x,y)
else:
entity_factories.troll.spawn(dungeon, x,y)
def generate_dungeon( def generate_dungeon(
max_rooms: int, max_rooms: int,
@ -61,9 +75,10 @@ def generate_dungeon(
room_max_size: int, room_max_size: int,
map_width: int, map_width: int,
map_height: int, map_height: int,
player: Entity max_monsters_per_room: int,
player: Entity,
) -> GameMap: ) -> GameMap:
dungeon = GameMap(map_width, map_height) dungeon = GameMap(map_width, map_height, entities=[player])
rooms: List[RectangularRoom] = [] rooms: List[RectangularRoom] = []
for room in range(max_rooms): for room in range(max_rooms):
@ -86,6 +101,8 @@ def generate_dungeon(
for x, y in tunnel_between(rooms[-1].center, new_room.center): for x, y in tunnel_between(rooms[-1].center, new_room.center):
dungeon.tiles[x, y] = tile_types.floor dungeon.tiles[x, y] = tile_types.floor
place_entities(new_room, dungeon, max_monsters_per_room)
rooms.append(new_room) rooms.append(new_room)
return dungeon return dungeon

38805
tags Normal file

File diff suppressed because it is too large Load Diff