Mouseover and log scroll added

Part 7 finished
This commit is contained in:
Kayne Ruse 2025-03-27 17:04:20 +11:00
parent 7685d57286
commit d630693ef8
8 changed files with 147 additions and 59 deletions

View File

@ -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()

View File

@ -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

View File

@ -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()

View File

@ -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
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)

View File

@ -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

View File

@ -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

View File

@ -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)

View File

@ -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)
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)