mirror of
https://github.com/Xevion/Pac-Man.git
synced 2025-12-07 01:15:48 -06:00
208 lines
7.7 KiB
Rust
208 lines
7.7 KiB
Rust
//! Debug rendering system
|
|
use std::cmp::Ordering;
|
|
|
|
use crate::constants::BOARD_PIXEL_OFFSET;
|
|
use crate::map::builder::Map;
|
|
use crate::systems::{Collider, CursorPosition, Position, SystemTimings};
|
|
use bevy_ecs::resource::Resource;
|
|
use bevy_ecs::system::{NonSendMut, Query, Res};
|
|
use glam::{IVec2, UVec2, Vec2};
|
|
use sdl2::pixels::Color;
|
|
use sdl2::rect::{Point, Rect};
|
|
use sdl2::render::{Canvas, Texture, TextureCreator};
|
|
use sdl2::ttf::Font;
|
|
use sdl2::video::{Window, WindowContext};
|
|
|
|
#[derive(Resource, Default, Debug, Copy, Clone)]
|
|
pub struct DebugState {
|
|
pub enabled: bool,
|
|
}
|
|
|
|
fn f32_to_u8(value: f32) -> u8 {
|
|
(value * 255.0) as u8
|
|
}
|
|
|
|
/// Resource to hold the debug texture for persistent rendering
|
|
pub struct DebugTextureResource(pub Texture<'static>);
|
|
|
|
/// Resource to hold the debug font
|
|
pub struct DebugFontResource(pub Font<'static, 'static>);
|
|
|
|
/// Transforms a position from logical canvas coordinates to output canvas coordinates (with board offset)
|
|
fn transform_position_with_offset(pos: Vec2, scale: f32) -> IVec2 {
|
|
((pos + BOARD_PIXEL_OFFSET.as_vec2()) * scale).as_ivec2()
|
|
}
|
|
|
|
/// Renders timing information in the top-left corner of the screen
|
|
fn render_timing_display(
|
|
canvas: &mut Canvas<Window>,
|
|
texture_creator: &mut TextureCreator<WindowContext>,
|
|
timings: &SystemTimings,
|
|
font: &Font,
|
|
) {
|
|
// Format timing information using the formatting module
|
|
let lines = timings.format_timing_display();
|
|
let line_height = 14; // Approximate line height for 12pt font
|
|
let padding = 10;
|
|
|
|
// Calculate background dimensions
|
|
let max_width = lines
|
|
.iter()
|
|
.filter(|l| !l.is_empty()) // Don't consider empty lines for width
|
|
.map(|line| font.size_of(line).unwrap().0)
|
|
.max()
|
|
.unwrap_or(0);
|
|
|
|
// Only draw background if there is text to display
|
|
if max_width > 0 {
|
|
let total_height = (lines.len() as u32) * line_height as u32;
|
|
let bg_padding = 5;
|
|
|
|
// Draw background
|
|
let bg_rect = Rect::new(
|
|
padding - bg_padding,
|
|
padding - bg_padding,
|
|
max_width + (bg_padding * 2) as u32,
|
|
total_height + bg_padding as u32,
|
|
);
|
|
canvas.set_blend_mode(sdl2::render::BlendMode::Blend);
|
|
canvas.set_draw_color(Color::RGBA(40, 40, 40, 180));
|
|
canvas.fill_rect(bg_rect).unwrap();
|
|
}
|
|
|
|
for (i, line) in lines.iter().enumerate() {
|
|
if line.is_empty() {
|
|
continue;
|
|
}
|
|
|
|
// Render each line
|
|
let surface = font.render(line).blended(Color::RGBA(255, 255, 255, 200)).unwrap();
|
|
let texture = texture_creator.create_texture_from_surface(&surface).unwrap();
|
|
|
|
// Position each line below the previous one
|
|
let y_pos = padding + (i * line_height) as i32;
|
|
let dest = Rect::new(padding, y_pos, texture.query().width, texture.query().height);
|
|
canvas.copy(&texture, None, dest).unwrap();
|
|
}
|
|
}
|
|
|
|
#[allow(clippy::too_many_arguments)]
|
|
pub fn debug_render_system(
|
|
mut canvas: NonSendMut<&mut Canvas<Window>>,
|
|
mut debug_texture: NonSendMut<DebugTextureResource>,
|
|
debug_font: NonSendMut<DebugFontResource>,
|
|
debug_state: Res<DebugState>,
|
|
timings: Res<SystemTimings>,
|
|
map: Res<Map>,
|
|
colliders: Query<(&Collider, &Position)>,
|
|
cursor: Res<CursorPosition>,
|
|
) {
|
|
if !debug_state.enabled {
|
|
return;
|
|
}
|
|
let scale =
|
|
(UVec2::from(canvas.output_size().unwrap()).as_vec2() / UVec2::from(canvas.logical_size()).as_vec2()).min_element();
|
|
|
|
// Get texture creator before entering the closure to avoid borrowing conflicts
|
|
let mut texture_creator = canvas.texture_creator();
|
|
let font = &debug_font.0;
|
|
|
|
let cursor_world_pos = match *cursor {
|
|
CursorPosition::None => None,
|
|
CursorPosition::Some { position, .. } => Some(position - BOARD_PIXEL_OFFSET.as_vec2()),
|
|
};
|
|
|
|
// Draw debug info on the high-resolution debug texture
|
|
canvas
|
|
.with_texture_canvas(&mut debug_texture.0, |debug_canvas| {
|
|
// Clear the debug canvas
|
|
debug_canvas.set_draw_color(Color::RGBA(0, 0, 0, 0));
|
|
debug_canvas.clear();
|
|
|
|
// Find the closest node to the cursor
|
|
let closest_node = if let Some(cursor_world_pos) = cursor_world_pos {
|
|
map.graph
|
|
.nodes()
|
|
.map(|node| node.position.distance(cursor_world_pos))
|
|
.enumerate()
|
|
.min_by(|(_, a), (_, b)| a.partial_cmp(b).unwrap_or(Ordering::Less))
|
|
.map(|(id, _)| id)
|
|
} else {
|
|
None
|
|
};
|
|
|
|
debug_canvas.set_draw_color(Color::GREEN);
|
|
for (collider, position) in colliders.iter() {
|
|
let pos = position.get_pixel_position(&map.graph).unwrap();
|
|
|
|
// Transform position and size using common methods
|
|
let pos = (pos * scale).as_ivec2();
|
|
let size = (collider.size * scale) as u32;
|
|
|
|
let rect = Rect::from_center(Point::from((pos.x, pos.y)), size, size);
|
|
debug_canvas.draw_rect(rect).unwrap();
|
|
}
|
|
|
|
debug_canvas.set_draw_color(Color {
|
|
a: f32_to_u8(0.4),
|
|
..Color::RED
|
|
});
|
|
debug_canvas.set_blend_mode(sdl2::render::BlendMode::Blend);
|
|
for (start_node, end_node) in map.graph.edges() {
|
|
let start_node_model = map.graph.get_node(start_node).unwrap();
|
|
let end_node = map.graph.get_node(end_node.target).unwrap().position;
|
|
|
|
// Transform positions using common method
|
|
let start = transform_position_with_offset(start_node_model.position, scale);
|
|
let end = transform_position_with_offset(end_node, scale);
|
|
|
|
debug_canvas
|
|
.draw_line(Point::from((start.x, start.y)), Point::from((end.x, end.y)))
|
|
.unwrap();
|
|
}
|
|
|
|
for (id, node) in map.graph.nodes().enumerate() {
|
|
let pos = node.position;
|
|
|
|
// Set color based on whether the node is the closest to the cursor
|
|
debug_canvas.set_draw_color(Color {
|
|
a: f32_to_u8(if Some(id) == closest_node { 0.75 } else { 0.6 }),
|
|
..(if Some(id) == closest_node {
|
|
Color::YELLOW
|
|
} else {
|
|
Color::BLUE
|
|
})
|
|
});
|
|
|
|
// Transform position using common method
|
|
let pos = transform_position_with_offset(pos, scale);
|
|
let size = (2.0 * scale) as u32;
|
|
|
|
debug_canvas
|
|
.fill_rect(Rect::new(pos.x - (size as i32 / 2), pos.y - (size as i32 / 2), size, size))
|
|
.unwrap();
|
|
}
|
|
|
|
// Render node ID if a node is highlighted
|
|
if let Some(closest_node_id) = closest_node {
|
|
let node = map.graph.get_node(closest_node_id).unwrap();
|
|
let pos = transform_position_with_offset(node.position, scale);
|
|
|
|
let surface = font
|
|
.render(&closest_node_id.to_string())
|
|
.blended(Color {
|
|
a: f32_to_u8(0.4),
|
|
..Color::WHITE
|
|
})
|
|
.unwrap();
|
|
let texture = texture_creator.create_texture_from_surface(&surface).unwrap();
|
|
let dest = Rect::new(pos.x + 10, pos.y - 5, texture.query().width, texture.query().height);
|
|
debug_canvas.copy(&texture, None, dest).unwrap();
|
|
}
|
|
|
|
// Render timing information in the top-left corner
|
|
render_timing_display(debug_canvas, &mut texture_creator, &timings, font);
|
|
})
|
|
.unwrap();
|
|
}
|