diff --git a/source/components/base_ai.py b/source/components/base_ai.py index c599c59..a6a0098 100644 --- a/source/components/base_ai.py +++ b/source/components/base_ai.py @@ -50,9 +50,9 @@ class AttackOnSight(BaseAI): if self.entity.floor_map.visible[self.entity.x, self.entity.y]: if distance <= 1: return MeleeAction(self.entity, xdir, ydir).apply() - + self.path = self.get_path_to(target.x, target.y) - + if self.path: dest_x, dest_y = self.path.pop(0) return MovementAction( @@ -60,5 +60,5 @@ class AttackOnSight(BaseAI): xdir = dest_x - self.entity.x, ydir = dest_y - self.entity.y, ).apply() - + return WaitAction(self.entity).apply() diff --git a/source/components/fighter.py b/source/components/fighter.py index cdb80ea..c84f696 100644 --- a/source/components/fighter.py +++ b/source/components/fighter.py @@ -13,7 +13,7 @@ class Fighter(BaseComponent): self._current_hp = hp self.attack = attack self.defense = defense - + @property def maximum_hp(self) -> int: return self._maximum_hp @@ -28,7 +28,7 @@ class Fighter(BaseComponent): if self.current_hp <= 0: self.die_and_despawn() - + def die_and_despawn(self) -> None: engine = self.entity.floor_map.engine diff --git a/source/engine.py b/source/engine.py index 6a977a7..cb86bb2 100644 --- a/source/engine.py +++ b/source/engine.py @@ -3,31 +3,37 @@ from tcod.console import Console from tcod.map import compute_fov from message_log import Message, MessageLog -from render_functions import render_hp_bar +from render_functions import render_hp_bar, render_names_at_location -import entity_types from floor_map import FloorMap #TODO: replace with "DungeonMap" class Engine: - def __init__(self, floor_map: FloorMap, intro_msg: Message = None): - from event_handler import InGameEventHandler - self.event_handler = InGameEventHandler(self) + def __init__(self, floor_map: FloorMap, intro_msg: Message = None, ui_width: int = None, ui_height: int = None): + #events + from event_handler import InGameHandler + self.event_handler = InGameHandler(self) + self.mouse_location = (0, 0) + + #map self.floor_map = floor_map self.floor_map.engine = self #references everywhere! + #messages self.message_log = MessageLog() if intro_msg: self.message_log.push_message(intro_msg) #grab the player object self.player = self.floor_map.player + self.ui_width = floor_map.width if ui_width is None else ui_width + self.ui_height = 0 if ui_height is None else ui_height #kick off the render self.update_fov() def run_loop(self, context: Context, console: Console) -> None: while True: - if self.event_handler.handle_events(): + if self.event_handler.handle_events(context): self.handle_entities() self.handle_rendering(context, console) @@ -47,18 +53,28 @@ class Engine: #UI render_hp_bar( console = console, + x = 0, + y = self.floor_map.height, current_value = self.player.fighter.current_hp, max_value = self.player.fighter.maximum_hp, - total_width = 20 + total_width = self.ui_width // 2, + ) + render_names_at_location( + console = console, + x = 1, + y = self.floor_map.height + 2, + engine = self, ) self.message_log.render( console=console, - x=21, - y=45 - 5, - width = 40, - height = 5 + x=self.ui_width // 2, + y=self.floor_map.height, + width = self.ui_width // 2, + height = self.ui_height, ) + self.event_handler.render(console) + #send to the screen context.present(console) console.clear() diff --git a/source/event_handler.py b/source/event_handler.py index 2d61959..e7b2dba 100644 --- a/source/event_handler.py +++ b/source/event_handler.py @@ -47,25 +47,34 @@ WAIT_KEYS = { tcod.event.KeySym.CLEAR, } +CURSOR_SCROLL_KEYS = { + tcod.event.KeySym.UP: -1, + tcod.event.KeySym.DOWN: 1, + tcod.event.KeySym.PAGEUP: -10, + tcod.event.KeySym.PAGEDOWN: 10, + + tcod.event.KeySym.KP_2: 1, + tcod.event.KeySym.KP_8: -1, +} + #event handler is one part of the engine class EventHandler(tcod.event.EventDispatch[BaseAction]): def __init__(self, engine: Engine): super().__init__() self.engine = engine + def render(self, console: tcod.console.Console) -> None: + pass #no-op + #callbacks def ev_quit(self, event: tcod.event.Quit) -> Optional[BaseAction]: return QuitAction() - def handle_events(self) -> bool: - raise NotImplementedError() - - -class InGameEventHandler(EventHandler): - def handle_events(self) -> bool: + def handle_events(self, context: tcod.context.Context) -> bool: result = False for event in tcod.event.wait(): + context.convert_event(event) action = self.dispatch(event) if action is None: @@ -75,6 +84,8 @@ class InGameEventHandler(EventHandler): return result + +class InGameHandler(EventHandler): def ev_keydown(self, event: tcod.event.KeyDown) -> Optional[BaseAction]: key = event.sym #SDL stuff, neat. @@ -87,30 +98,67 @@ class InGameEventHandler(EventHandler): if key in MOVE_KEYS: xdir, ydir = MOVE_KEYS[key] return BumpAction(player, xdir = xdir, ydir = ydir) - + if key in WAIT_KEYS: return WaitAction(player) + if key == tcod.event.KeySym.v: + self.engine.event_handler = LogHistoryViewer(self.engine) -class GameOverEventHandler(EventHandler): - def handle_events(self) -> bool: - result = False + def ev_mousemotion(self, event: tcod.event.MouseMotion) -> None: + if self.engine.floor_map.in_bounds(event.tile.x, event.tile.y): + self.engine.mouse_location = event.tile.x, event.tile.y - for event in tcod.event.wait(): - action = self.dispatch(event) - - if action is None: - continue - - result |= action.apply() - - return result +class GameOverHandler(EventHandler): def ev_keydown(self, event: tcod.event.KeyDown) -> Optional[BaseAction]: key = event.sym #SDL stuff, neat. #player input if key == tcod.event.KeySym.ESCAPE: return QuitAction() - - return None \ No newline at end of file + + return None + + +class LogHistoryViewer(EventHandler): + def __init__(self, engine: Engine): + super().__init__(engine) + self.log_length = len(engine.message_log.messages) + self.cursor = self.log_length - 1 + + def render(self, console: tcod.console.Console) -> None: + super().render(console) + + log_console = tcod.console.Console(console.width - 6, console.height - 6) + + #custom... + log_console.draw_frame(0, 0, log_console.width, log_console.height) + log_console.print_box( + 0, 0, log_console.width, 1, "Message History", alignment=tcod.constants.CENTER + ) + self.engine.message_log.render_messages( + log_console, + 1, 1, + log_console.width - 2, log_console.height - 2, + self.engine.message_log.messages[:self.cursor + 1] + ) + log_console.blit(console, 3, 3) #into the middle + + def ev_keydown(self, event: tcod.event.KeyDown) -> Optional[BaseAction]: + if event.sym in CURSOR_SCROLL_KEYS: + adjust = CURSOR_SCROLL_KEYS[event.sym] + if adjust < 0 and self.cursor == 0: + pass #do nothing + elif adjust > 0 and self.cursor == self.log_length - 1: + pass #do nothing + else: + self.cursor = max(0, min(self.log_length - 1, self.cursor + adjust)) #TODO: nicer scroll down + + elif event.sym == tcod.event.KeySym.HOME: + self.cursor = 0 + elif event.sym == tcod.event.KeySym.END: + self.cursor = self.log_length - 1 + else: + #return to the game + self.engine.event_handler = InGameHandler(self.engine) \ No newline at end of file diff --git a/source/floor_map.py b/source/floor_map.py index c336641..89cfecf 100644 --- a/source/floor_map.py +++ b/source/floor_map.py @@ -71,7 +71,7 @@ class FloorMap: for entity in alive: if self.visible[entity.x, entity.y]: console.print(entity.x, entity.y, entity.char, fg=entity.color) - + #print the player above everything else for clarity if self.player: console.print(self.player.x, self.player.y, self.player.char, fg=self.player.color) #TODO: I didn't realize the render order would be fixed in the tutorial \ No newline at end of file diff --git a/source/main.py b/source/main.py index 2b46ef0..cc573d3 100755 --- a/source/main.py +++ b/source/main.py @@ -7,29 +7,31 @@ from message_log import Message import colors def main() -> None: + #screen dimensions depend partially on the tileset + tileset = tcod.tileset.load_tilesheet("assets/dejavu10x10_gs_tc.png", 32, 8, tcod.tileset.CHARMAP_TCOD) + + #how big is the map's dimensions + map_width = 80 + map_height = 40 + + ui_height = 5 + #tcod stuff context = tcod.context.new( - columns = 80, - rows = 45, - tileset = tcod.tileset.load_tilesheet("assets/dejavu10x10_gs_tc.png", 32, 8, tcod.tileset.CHARMAP_TCOD), + columns = map_width, + rows = map_height, + tileset = tileset, title = "Stepwise Roguelike", vsync = True ) - w, h = context.recommended_console_size(min_columns=10, min_rows=10) - - print (w,h) - - console = tcod.console.Console( - width = w, - height = h + 5, - order = "F" - ) + console = context.new_console(map_width, map_height + ui_height, order="F") engine = Engine( #is created externally, because - floor_map = generate_floor_map(80, 45, 10, 10), - intro_msg = Message("Welcome to the Cave of Gobbos!", colors.welcome_text) + floor_map = generate_floor_map(map_width, map_height, room_width_max=12, room_height_max=12), + intro_msg = Message("Welcome to the Cave of Gobbos!", colors.welcome_text), + ui_height = ui_height, ) #game loop that never returns diff --git a/source/message_log.py b/source/message_log.py index 3aefa0f..392ea45 100644 --- a/source/message_log.py +++ b/source/message_log.py @@ -28,7 +28,7 @@ class MessageLog: self.messages[-1].count += 1 else: self.messages.append(Message(text, fg)) - + def push_message(self, msg: Message) -> None: self.messages.append(msg) diff --git a/source/render_functions.py b/source/render_functions.py index 352fbd8..aaf7b69 100644 --- a/source/render_functions.py +++ b/source/render_functions.py @@ -1,15 +1,37 @@ from __future__ import annotations -import colors +from typing import Any from tcod.console import Console -def render_hp_bar(console: Console, current_value: int, max_value: int, total_width: int) -> None: +import colors +from floor_map import FloorMap + +#utils +def get_names_at(x: int, y: int, floor_map: FloorMap) -> str: + if not floor_map.in_bounds(x, y) or not floor_map.visible[x, y]: + return "" + + names = ", ".join( + entity.name for entity in floor_map.entities if entity.x == x and entity.y == y + ) + + return names + +#direct rendering functions +def render_hp_bar(console: Console, x: int, y: int, current_value: int, max_value: int, total_width: int) -> None: bar_width = int(float(current_value) / max_value * total_width) - console.draw_rect(x=0, y=45, width=total_width, height=1, ch=1, bg=colors.bar_empty) + console.draw_rect(x=x, y=y, width=total_width, height=1, ch=1, bg=colors.bar_empty) if bar_width > 0: - console.draw_rect(x=0, y=45, width=bar_width, height=1, ch=1, bg=colors.bar_filled) - - console.print(x=1, y=45, string=f"HP: {current_value}/{max_value}", fg=colors.bar_text) \ No newline at end of file + console.draw_rect(x=x, y=y, width=bar_width, height=1, ch=1, bg=colors.bar_filled) + + console.print(x=x + 1, y=y, string=f"HP: {current_value}/{max_value}", fg=colors.bar_text) + +def render_names_at_location(console: Console, x: int, y: int, engine: Any) -> None: + mouse_x, mouse_y = engine.mouse_location + + names: str = get_names_at(mouse_x, mouse_y, engine.floor_map) + + console.print(x=x, y=y, string=names)