From 40a1de4b8b70bf7bb75cb8bade6b898a5f40a138 Mon Sep 17 00:00:00 2001 From: Silas Bartha Date: Sat, 22 Mar 2025 12:57:47 -0400 Subject: [PATCH 1/3] Sat Mar 22 12:57:47 PM EDT 2025 --- .cargo/config.toml | 2 -- .gitignore | 1 + Cargo.toml | 14 ++++++++++++-- 3 files changed, 13 insertions(+), 4 deletions(-) delete mode 100644 .cargo/config.toml diff --git a/.cargo/config.toml b/.cargo/config.toml deleted file mode 100644 index 62d8a86..0000000 --- a/.cargo/config.toml +++ /dev/null @@ -1,2 +0,0 @@ -[profile.dev.package."*"] -opt-level = 2 diff --git a/.gitignore b/.gitignore index 96ef6c0..35e6e2b 100644 --- a/.gitignore +++ b/.gitignore @@ -1,2 +1,3 @@ /target Cargo.lock +.cargo diff --git a/Cargo.toml b/Cargo.toml index 27cd1ad..84f4844 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,11 +6,16 @@ license = "0BSD OR MIT OR Apache-2.0" description = "A plugin for the Bevy game engine which enables rendering to a terminal using unicode braille characters." repository = "https://git.soaos.dev/soaos/bevy_terminal_display" +[profile.dev] +opt-level = 1 + +[profile.dev.package."*"] +opt-level = 2 + [dependencies] crossbeam-channel = "0.5" downcast-rs = "2.0" once_cell = "1.19" -bevy_headless_render = "0.2" ratatui = "0.29" color-eyre = "0.6" leafwing-input-manager = "0.16" @@ -26,5 +31,10 @@ features = ["bevy_render"] version = "0.28" features = ["serde"] +# SOAOS DEPS + [dependencies.bevy_dither_post_process] -git = "https://git.soaos.dev/soaos/bevy_dither_post_process" +version = "0.3" + +[dependencies.bevy_headless_render] +version = "0.2" \ No newline at end of file -- 2.39.5 From 9bcb260cca8d48b97aeec224d7249e6dc6fc9639 Mon Sep 17 00:00:00 2001 From: Silas Bartha Date: Thu, 27 Mar 2025 17:19:11 -0400 Subject: [PATCH 2/3] WIP - widget focus --- src/lib.rs | 17 ++++------------- src/widgets/commands.rs | 23 +++++++++++++++++++++++ src/widgets/mod.rs | 6 ++++++ src/widgets/resources.rs | 4 ++++ 4 files changed, 37 insertions(+), 13 deletions(-) create mode 100644 src/widgets/commands.rs create mode 100644 src/widgets/resources.rs diff --git a/src/lib.rs b/src/lib.rs index cd6519f..fd7686c 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -3,7 +3,7 @@ //! Bevy plugin which allows a camera to render to a terminal window. use std::{ - fs::OpenOptions, io::{stdout, Write}, path::PathBuf, sync::{Arc, Mutex} + fs::OpenOptions, io::{stdout, Write}, path::PathBuf, }; use bevy::{ @@ -18,7 +18,6 @@ use bevy_headless_render::HeadlessRenderPlugin; use color_eyre::config::HookBuilder; pub use crossterm; use crossterm::{event::{DisableMouseCapture, PopKeyboardEnhancementFlags}, terminal::{disable_raw_mode, LeaveAlternateScreen}, ExecutableCommand}; -use once_cell::sync::Lazy; pub use ratatui; /// Functions and types related to capture and display of world to terminal @@ -30,8 +29,6 @@ pub mod input; /// Functions and types related to constructing and rendering TUI widgets pub mod widgets; -static LOG_PATH: Lazy>> = Lazy::new(|| Arc::new(Mutex::new(PathBuf::default()))); - /// Plugin providing terminal display functionality pub struct TerminalDisplayPlugin { /// Path to redirect tracing logs to. Defaults to "debug.log" @@ -48,19 +45,12 @@ impl Default for TerminalDisplayPlugin { impl Plugin for TerminalDisplayPlugin { fn build(&self, app: &mut App) { - *LOG_PATH - .lock() - .expect("Failed to get lock on log path mutex") = self.log_path.clone(); + let log_path = self.log_path.clone(); let log_file = OpenOptions::new() .write(true) .create(true) .truncate(true) - .open( - LOG_PATH - .lock() - .expect("Failed to get lock on log path mutex") - .clone(), - ) + .open(log_path) .unwrap(); let file_layer = tracing_subscriber::fmt::Layer::new() .with_writer(log_file) @@ -100,6 +90,7 @@ impl Plugin for TerminalDisplayPlugin { ) .insert_resource(display::resources::Terminal::default()) .insert_resource(input::resources::EventQueue::default()) + .init_resource::() .add_event::(); } } diff --git a/src/widgets/commands.rs b/src/widgets/commands.rs new file mode 100644 index 0000000..e5ee614 --- /dev/null +++ b/src/widgets/commands.rs @@ -0,0 +1,23 @@ +use bevy::prelude::*; + +use super::resources::FocusedWidget; + +struct FocusWidgetCommand(Entity); + +impl Command for FocusWidgetCommand { + fn apply(self, world: &mut World) { + **world.resource_mut::() = Some(self.0); + } +} + +/// Command interface for manipulating terminal widget resources +pub trait TerminalWidgetCommands { + /// Gives focus to the terminal widget on the provided entity. + fn focus_widget(&mut self, widget: Entity); +} + +impl<'w, 's> TerminalWidgetCommands for Commands<'w,'s> { + fn focus_widget(&mut self, widget: Entity) { + self.queue(FocusWidgetCommand(widget)); + } +} \ No newline at end of file diff --git a/src/widgets/mod.rs b/src/widgets/mod.rs index 7751b2d..4435af8 100644 --- a/src/widgets/mod.rs +++ b/src/widgets/mod.rs @@ -7,9 +7,15 @@ use crate::input::events::TerminalInputEvent; /// Components for this module pub mod components; +/// Resources for this module +pub mod resources; + /// Systems for this module pub(crate) mod systems; +/// Commands for this module +pub mod commands; + /// Trait which defines an interface for terminal widgets pub trait TerminalWidget: DowncastSync { /// Called every frame to render the widget diff --git a/src/widgets/resources.rs b/src/widgets/resources.rs new file mode 100644 index 0000000..afecb7f --- /dev/null +++ b/src/widgets/resources.rs @@ -0,0 +1,4 @@ +use bevy::prelude::*; + +#[derive(Resource, Default, Deref, DerefMut, Debug)] +pub struct FocusedWidget(pub Option); -- 2.39.5 From 8514c007d5b64f6d4c447c14bfa53e72dc859f5e Mon Sep 17 00:00:00 2001 From: Silas Bartha Date: Thu, 27 Mar 2025 19:06:53 -0400 Subject: [PATCH 3/3] Widget focus --- src/widgets/resources.rs | 3 +++ src/widgets/systems.rs | 15 ++++++++++----- 2 files changed, 13 insertions(+), 5 deletions(-) diff --git a/src/widgets/resources.rs b/src/widgets/resources.rs index afecb7f..84f4b8d 100644 --- a/src/widgets/resources.rs +++ b/src/widgets/resources.rs @@ -1,4 +1,7 @@ use bevy::prelude::*; +/// Terminal widget entity currently focused and handling input +/// Can be manipulated directly or you can request an entity be focused through +/// the `focus_widget` command. #[derive(Resource, Default, Deref, DerefMut, Debug)] pub struct FocusedWidget(pub Option); diff --git a/src/widgets/systems.rs b/src/widgets/systems.rs index 0638b01..c911b0b 100644 --- a/src/widgets/systems.rs +++ b/src/widgets/systems.rs @@ -2,17 +2,22 @@ use bevy::prelude::*; use crate::input::events::TerminalInputEvent; -use super::components::Widget; +use super::{components::Widget, resources::FocusedWidget}; -/// Invokes every enabled widget's `handle_events` methods for each incoming input event +/// Invokes focused widget's `handle_events` methods for each incoming input event pub fn widget_input_handling( mut widgets: Query<&mut Widget>, mut event_reader: EventReader, mut commands: Commands, + focused_widget: Res, ) { - for event in event_reader.read() { - for mut widget in widgets.iter_mut().filter(|widget| widget.enabled) { - widget.widget.handle_events(event, &mut commands); + if let Some(entity) = **focused_widget { + if let Ok(mut widget) = widgets.get_mut(entity) { + if widget.enabled == true { + for event in event_reader.read() { + widget.widget.handle_events(event, &mut commands); + } + } } } } -- 2.39.5