mirror of
https://github.com/Xevion/Pac-Man.git
synced 2025-12-06 23:15:42 -06:00
Compare commits
2 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
33775166a7 | ||
|
|
f2732a7ff7 |
@@ -31,6 +31,7 @@ use bevy_ecs::schedule::common_conditions::resource_changed;
|
|||||||
use bevy_ecs::schedule::{Condition, IntoScheduleConfigs, Schedule, SystemSet};
|
use bevy_ecs::schedule::{Condition, IntoScheduleConfigs, Schedule, SystemSet};
|
||||||
use bevy_ecs::system::ResMut;
|
use bevy_ecs::system::ResMut;
|
||||||
use bevy_ecs::world::World;
|
use bevy_ecs::world::World;
|
||||||
|
use glam::UVec2;
|
||||||
use sdl2::event::EventType;
|
use sdl2::event::EventType;
|
||||||
use sdl2::image::LoadTexture;
|
use sdl2::image::LoadTexture;
|
||||||
use sdl2::render::{BlendMode, Canvas, ScaleMode, TextureCreator};
|
use sdl2::render::{BlendMode, Canvas, ScaleMode, TextureCreator};
|
||||||
@@ -42,7 +43,7 @@ use crate::{
|
|||||||
asset::{get_asset_bytes, Asset},
|
asset::{get_asset_bytes, Asset},
|
||||||
events::GameCommand,
|
events::GameCommand,
|
||||||
map::render::MapRenderer,
|
map::render::MapRenderer,
|
||||||
systems::debug::TtfAtlasResource,
|
systems::debug::{BatchedLinesResource, TtfAtlasResource},
|
||||||
systems::input::{Bindings, CursorPosition},
|
systems::input::{Bindings, CursorPosition},
|
||||||
texture::sprite::{AtlasMapper, SpriteAtlas},
|
texture::sprite::{AtlasMapper, SpriteAtlas},
|
||||||
};
|
};
|
||||||
@@ -126,7 +127,7 @@ impl Game {
|
|||||||
EventType::Display,
|
EventType::Display,
|
||||||
EventType::Window,
|
EventType::Window,
|
||||||
EventType::MouseWheel,
|
EventType::MouseWheel,
|
||||||
EventType::MouseMotion,
|
// EventType::MouseMotion,
|
||||||
EventType::MouseButtonDown,
|
EventType::MouseButtonDown,
|
||||||
EventType::MouseButtonUp,
|
EventType::MouseButtonUp,
|
||||||
EventType::MouseButtonDown,
|
EventType::MouseButtonDown,
|
||||||
@@ -299,6 +300,10 @@ impl Game {
|
|||||||
EventRegistry::register_event::<GameEvent>(&mut world);
|
EventRegistry::register_event::<GameEvent>(&mut world);
|
||||||
EventRegistry::register_event::<AudioEvent>(&mut world);
|
EventRegistry::register_event::<AudioEvent>(&mut world);
|
||||||
|
|
||||||
|
let scale =
|
||||||
|
(UVec2::from(canvas.output_size().unwrap()).as_vec2() / UVec2::from(canvas.logical_size()).as_vec2()).min_element();
|
||||||
|
|
||||||
|
world.insert_resource(BatchedLinesResource::new(&map, scale));
|
||||||
world.insert_resource(Self::create_ghost_animations(&atlas)?);
|
world.insert_resource(Self::create_ghost_animations(&atlas)?);
|
||||||
world.insert_resource(map);
|
world.insert_resource(map);
|
||||||
world.insert_resource(GlobalState { exit: false });
|
world.insert_resource(GlobalState { exit: false });
|
||||||
|
|||||||
@@ -12,6 +12,9 @@ use sdl2::pixels::Color;
|
|||||||
use sdl2::rect::{Point, Rect};
|
use sdl2::rect::{Point, Rect};
|
||||||
use sdl2::render::{Canvas, Texture};
|
use sdl2::render::{Canvas, Texture};
|
||||||
use sdl2::video::Window;
|
use sdl2::video::Window;
|
||||||
|
use smallvec::SmallVec;
|
||||||
|
use std::collections::{HashMap, HashSet};
|
||||||
|
use tracing::warn;
|
||||||
|
|
||||||
#[derive(Resource, Default, Debug, Copy, Clone)]
|
#[derive(Resource, Default, Debug, Copy, Clone)]
|
||||||
pub struct DebugState {
|
pub struct DebugState {
|
||||||
@@ -28,6 +31,118 @@ pub struct DebugTextureResource(pub Texture);
|
|||||||
/// Resource to hold the TTF text atlas
|
/// Resource to hold the TTF text atlas
|
||||||
pub struct TtfAtlasResource(pub TtfAtlas);
|
pub struct TtfAtlasResource(pub TtfAtlas);
|
||||||
|
|
||||||
|
/// Resource to hold pre-computed batched line segments
|
||||||
|
#[derive(Resource, Default, Debug, Clone)]
|
||||||
|
pub struct BatchedLinesResource {
|
||||||
|
horizontal_lines: Vec<(i32, i32, i32)>, // (y, x_start, x_end)
|
||||||
|
vertical_lines: Vec<(i32, i32, i32)>, // (x, y_start, y_end)
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BatchedLinesResource {
|
||||||
|
/// Computes and caches batched line segments for the map graph
|
||||||
|
pub fn new(map: &Map, scale: f32) -> Self {
|
||||||
|
let mut horizontal_segments: HashMap<i32, Vec<(i32, i32)>> = HashMap::new();
|
||||||
|
let mut vertical_segments: HashMap<i32, Vec<(i32, i32)>> = HashMap::new();
|
||||||
|
let mut processed_edges: HashSet<(u16, u16)> = HashSet::new();
|
||||||
|
|
||||||
|
// Process all edges and group them by axis
|
||||||
|
for (start_node_id, edge) in map.graph.edges() {
|
||||||
|
// Acquire a stable key for the edge (from < to)
|
||||||
|
let edge_key = (start_node_id.min(edge.target), start_node_id.max(edge.target));
|
||||||
|
|
||||||
|
// Skip if we've already processed this edge in the reverse direction
|
||||||
|
if processed_edges.contains(&edge_key) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
processed_edges.insert(edge_key);
|
||||||
|
|
||||||
|
let start_pos = map.graph.get_node(start_node_id).unwrap().position;
|
||||||
|
let end_pos = map.graph.get_node(edge.target).unwrap().position;
|
||||||
|
|
||||||
|
let start = transform_position_with_offset(start_pos, scale);
|
||||||
|
let end = transform_position_with_offset(end_pos, scale);
|
||||||
|
|
||||||
|
// Determine if this is a horizontal or vertical line
|
||||||
|
if (start.y - end.y).abs() < 2 {
|
||||||
|
// Horizontal line (allowing for slight vertical variance)
|
||||||
|
let y = start.y;
|
||||||
|
let x_min = start.x.min(end.x);
|
||||||
|
let x_max = start.x.max(end.x);
|
||||||
|
horizontal_segments.entry(y).or_default().push((x_min, x_max));
|
||||||
|
} else if (start.x - end.x).abs() < 2 {
|
||||||
|
// Vertical line (allowing for slight horizontal variance)
|
||||||
|
let x = start.x;
|
||||||
|
let y_min = start.y.min(end.y);
|
||||||
|
let y_max = start.y.max(end.y);
|
||||||
|
vertical_segments.entry(x).or_default().push((y_min, y_max));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Merges overlapping or adjacent segments into continuous lines
|
||||||
|
fn merge_segments(segments: Vec<(i32, i32)>) -> Vec<(i32, i32)> {
|
||||||
|
if segments.is_empty() {
|
||||||
|
return Vec::new();
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut merged = Vec::new();
|
||||||
|
let mut current_start = segments[0].0;
|
||||||
|
let mut current_end = segments[0].1;
|
||||||
|
|
||||||
|
for &(start, end) in segments.iter().skip(1) {
|
||||||
|
if start <= current_end + 1 {
|
||||||
|
// Adjacent or overlapping
|
||||||
|
current_end = current_end.max(end);
|
||||||
|
} else {
|
||||||
|
merged.push((current_start, current_end));
|
||||||
|
current_start = start;
|
||||||
|
current_end = end;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
merged.push((current_start, current_end));
|
||||||
|
merged
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert to flat vectors for fast iteration during rendering
|
||||||
|
let horizontal_lines = horizontal_segments
|
||||||
|
.into_iter()
|
||||||
|
.flat_map(|(y, mut segments)| {
|
||||||
|
segments.sort_unstable_by_key(|(start, _)| *start);
|
||||||
|
let merged = merge_segments(segments);
|
||||||
|
merged.into_iter().map(move |(x_start, x_end)| (y, x_start, x_end))
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
let vertical_lines = vertical_segments
|
||||||
|
.into_iter()
|
||||||
|
.flat_map(|(x, mut segments)| {
|
||||||
|
segments.sort_unstable_by_key(|(start, _)| *start);
|
||||||
|
let merged = merge_segments(segments);
|
||||||
|
merged.into_iter().map(move |(y_start, y_end)| (x, y_start, y_end))
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
Self {
|
||||||
|
horizontal_lines,
|
||||||
|
vertical_lines,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn render(&self, canvas: &mut Canvas<Window>) {
|
||||||
|
// Render horizontal lines
|
||||||
|
for &(y, x_start, x_end) in &self.horizontal_lines {
|
||||||
|
let points = [Point::new(x_start, y), Point::new(x_end, y)];
|
||||||
|
let _ = canvas.draw_lines(&points[..]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Render vertical lines
|
||||||
|
for &(x, y_start, y_end) in &self.vertical_lines {
|
||||||
|
let points = [Point::new(x, y_start), Point::new(x, y_end)];
|
||||||
|
let _ = canvas.draw_lines(&points[..]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Transforms a position from logical canvas coordinates to output canvas coordinates (with board offset)
|
/// Transforms a position from logical canvas coordinates to output canvas coordinates (with board offset)
|
||||||
fn transform_position_with_offset(pos: Vec2, scale: f32) -> IVec2 {
|
fn transform_position_with_offset(pos: Vec2, scale: f32) -> IVec2 {
|
||||||
((pos + BOARD_PIXEL_OFFSET.as_vec2()) * scale).as_ivec2()
|
((pos + BOARD_PIXEL_OFFSET.as_vec2()) * scale).as_ivec2()
|
||||||
@@ -91,6 +206,7 @@ pub fn debug_render_system(
|
|||||||
mut canvas: NonSendMut<&mut Canvas<Window>>,
|
mut canvas: NonSendMut<&mut Canvas<Window>>,
|
||||||
mut debug_texture: NonSendMut<DebugTextureResource>,
|
mut debug_texture: NonSendMut<DebugTextureResource>,
|
||||||
mut ttf_atlas: NonSendMut<TtfAtlasResource>,
|
mut ttf_atlas: NonSendMut<TtfAtlasResource>,
|
||||||
|
batched_lines: Res<BatchedLinesResource>,
|
||||||
debug_state: Res<DebugState>,
|
debug_state: Res<DebugState>,
|
||||||
timings: Res<SystemTimings>,
|
timings: Res<SystemTimings>,
|
||||||
map: Res<Map>,
|
map: Res<Map>,
|
||||||
@@ -131,55 +247,70 @@ pub fn debug_render_system(
|
|||||||
};
|
};
|
||||||
|
|
||||||
debug_canvas.set_draw_color(Color::GREEN);
|
debug_canvas.set_draw_color(Color::GREEN);
|
||||||
for (collider, position) in colliders.iter() {
|
{
|
||||||
|
let rects = colliders
|
||||||
|
.iter()
|
||||||
|
.map(|(collider, position)| {
|
||||||
let pos = position.get_pixel_position(&map.graph).unwrap();
|
let pos = position.get_pixel_position(&map.graph).unwrap();
|
||||||
|
|
||||||
// Transform position and size using common methods
|
// Transform position and size using common methods
|
||||||
let pos = (pos * scale).as_ivec2();
|
let pos = (pos * scale).as_ivec2();
|
||||||
let size = (collider.size * scale) as u32;
|
let size = (collider.size * scale) as u32;
|
||||||
|
|
||||||
let rect = Rect::from_center(Point::from((pos.x, pos.y)), size, size);
|
Rect::from_center(Point::from((pos.x, pos.y)), size, size)
|
||||||
debug_canvas.draw_rect(rect).unwrap();
|
})
|
||||||
|
.collect::<SmallVec<[Rect; 100]>>();
|
||||||
|
if rects.len() > rects.capacity() {
|
||||||
|
warn!(
|
||||||
|
capacity = rects.capacity(),
|
||||||
|
count = rects.len(),
|
||||||
|
"Collider rects capacity exceeded"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
debug_canvas.draw_rects(&rects).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
debug_canvas.set_draw_color(Color {
|
debug_canvas.set_draw_color(Color {
|
||||||
a: f32_to_u8(0.4),
|
a: f32_to_u8(0.6),
|
||||||
..Color::RED
|
..Color::RED
|
||||||
});
|
});
|
||||||
debug_canvas.set_blend_mode(sdl2::render::BlendMode::Blend);
|
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
|
// Use cached batched line segments
|
||||||
let start = transform_position_with_offset(start_node_model.position, scale);
|
batched_lines.render(debug_canvas);
|
||||||
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)))
|
let rects: Vec<_> = map
|
||||||
.unwrap();
|
.graph
|
||||||
|
.nodes()
|
||||||
|
.enumerate()
|
||||||
|
.filter_map(|(id, node)| {
|
||||||
|
let pos = transform_position_with_offset(node.position, scale);
|
||||||
|
let size = (2.0 * scale) as u32;
|
||||||
|
let rect = Rect::new(pos.x - (size as i32 / 2), pos.y - (size as i32 / 2), size, size);
|
||||||
|
|
||||||
|
// If the node is the one closest to the cursor, draw it immediately
|
||||||
|
if closest_node == Some(id) {
|
||||||
|
debug_canvas.set_draw_color(Color::YELLOW);
|
||||||
|
debug_canvas.fill_rect(rect).unwrap();
|
||||||
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (id, node) in map.graph.nodes().enumerate() {
|
Some(rect)
|
||||||
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
|
|
||||||
})
|
})
|
||||||
});
|
.collect();
|
||||||
|
|
||||||
// Transform position using common method
|
if rects.len() > rects.capacity() {
|
||||||
let pos = transform_position_with_offset(pos, scale);
|
warn!(
|
||||||
let size = (2.0 * scale) as u32;
|
capacity = rects.capacity(),
|
||||||
|
count = rects.len(),
|
||||||
|
"Node rects capacity exceeded"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
debug_canvas
|
// Draw the non-closest nodes all at once in blue
|
||||||
.fill_rect(Rect::new(pos.x - (size as i32 / 2), pos.y - (size as i32 / 2), size, size))
|
debug_canvas.set_draw_color(Color::BLUE);
|
||||||
.unwrap();
|
debug_canvas.fill_rects(&rects).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Render node ID if a node is highlighted
|
// Render node ID if a node is highlighted
|
||||||
|
|||||||
Reference in New Issue
Block a user