Widget Focus #3
@ -1,2 +0,0 @@
|
|||||||
[profile.dev.package."*"]
|
|
||||||
opt-level = 2
|
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,2 +1,3 @@
|
|||||||
/target
|
/target
|
||||||
Cargo.lock
|
Cargo.lock
|
||||||
|
.cargo
|
||||||
|
14
Cargo.toml
14
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."
|
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"
|
repository = "https://git.soaos.dev/soaos/bevy_terminal_display"
|
||||||
|
|
||||||
|
[profile.dev]
|
||||||
|
opt-level = 1
|
||||||
|
|
||||||
|
[profile.dev.package."*"]
|
||||||
|
opt-level = 2
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
crossbeam-channel = "0.5"
|
crossbeam-channel = "0.5"
|
||||||
downcast-rs = "2.0"
|
downcast-rs = "2.0"
|
||||||
once_cell = "1.19"
|
once_cell = "1.19"
|
||||||
bevy_headless_render = "0.2"
|
|
||||||
ratatui = "0.29"
|
ratatui = "0.29"
|
||||||
color-eyre = "0.6"
|
color-eyre = "0.6"
|
||||||
leafwing-input-manager = "0.16"
|
leafwing-input-manager = "0.16"
|
||||||
@ -26,5 +31,10 @@ features = ["bevy_render"]
|
|||||||
version = "0.28"
|
version = "0.28"
|
||||||
features = ["serde"]
|
features = ["serde"]
|
||||||
|
|
||||||
|
# SOAOS DEPS
|
||||||
|
|
||||||
[dependencies.bevy_dither_post_process]
|
[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"
|
17
src/lib.rs
17
src/lib.rs
@ -3,7 +3,7 @@
|
|||||||
//! Bevy plugin which allows a camera to render to a terminal window.
|
//! Bevy plugin which allows a camera to render to a terminal window.
|
||||||
|
|
||||||
use std::{
|
use std::{
|
||||||
fs::OpenOptions, io::{stdout, Write}, path::PathBuf, sync::{Arc, Mutex}
|
fs::OpenOptions, io::{stdout, Write}, path::PathBuf,
|
||||||
};
|
};
|
||||||
|
|
||||||
use bevy::{
|
use bevy::{
|
||||||
@ -18,7 +18,6 @@ use bevy_headless_render::HeadlessRenderPlugin;
|
|||||||
use color_eyre::config::HookBuilder;
|
use color_eyre::config::HookBuilder;
|
||||||
pub use crossterm;
|
pub use crossterm;
|
||||||
use crossterm::{event::{DisableMouseCapture, PopKeyboardEnhancementFlags}, terminal::{disable_raw_mode, LeaveAlternateScreen}, ExecutableCommand};
|
use crossterm::{event::{DisableMouseCapture, PopKeyboardEnhancementFlags}, terminal::{disable_raw_mode, LeaveAlternateScreen}, ExecutableCommand};
|
||||||
use once_cell::sync::Lazy;
|
|
||||||
pub use ratatui;
|
pub use ratatui;
|
||||||
|
|
||||||
/// Functions and types related to capture and display of world to terminal
|
/// 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
|
/// Functions and types related to constructing and rendering TUI widgets
|
||||||
pub mod widgets;
|
pub mod widgets;
|
||||||
|
|
||||||
static LOG_PATH: Lazy<Arc<Mutex<PathBuf>>> = Lazy::new(|| Arc::new(Mutex::new(PathBuf::default())));
|
|
||||||
|
|
||||||
/// Plugin providing terminal display functionality
|
/// Plugin providing terminal display functionality
|
||||||
pub struct TerminalDisplayPlugin {
|
pub struct TerminalDisplayPlugin {
|
||||||
/// Path to redirect tracing logs to. Defaults to "debug.log"
|
/// Path to redirect tracing logs to. Defaults to "debug.log"
|
||||||
@ -48,19 +45,12 @@ impl Default for TerminalDisplayPlugin {
|
|||||||
|
|
||||||
impl Plugin for TerminalDisplayPlugin {
|
impl Plugin for TerminalDisplayPlugin {
|
||||||
fn build(&self, app: &mut App) {
|
fn build(&self, app: &mut App) {
|
||||||
*LOG_PATH
|
let log_path = self.log_path.clone();
|
||||||
.lock()
|
|
||||||
.expect("Failed to get lock on log path mutex") = self.log_path.clone();
|
|
||||||
let log_file = OpenOptions::new()
|
let log_file = OpenOptions::new()
|
||||||
.write(true)
|
.write(true)
|
||||||
.create(true)
|
.create(true)
|
||||||
.truncate(true)
|
.truncate(true)
|
||||||
.open(
|
.open(log_path)
|
||||||
LOG_PATH
|
|
||||||
.lock()
|
|
||||||
.expect("Failed to get lock on log path mutex")
|
|
||||||
.clone(),
|
|
||||||
)
|
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let file_layer = tracing_subscriber::fmt::Layer::new()
|
let file_layer = tracing_subscriber::fmt::Layer::new()
|
||||||
.with_writer(log_file)
|
.with_writer(log_file)
|
||||||
@ -100,6 +90,7 @@ impl Plugin for TerminalDisplayPlugin {
|
|||||||
)
|
)
|
||||||
.insert_resource(display::resources::Terminal::default())
|
.insert_resource(display::resources::Terminal::default())
|
||||||
.insert_resource(input::resources::EventQueue::default())
|
.insert_resource(input::resources::EventQueue::default())
|
||||||
|
.init_resource::<widgets::resources::FocusedWidget>()
|
||||||
.add_event::<input::events::TerminalInputEvent>();
|
.add_event::<input::events::TerminalInputEvent>();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
23
src/widgets/commands.rs
Normal file
23
src/widgets/commands.rs
Normal file
@ -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::<FocusedWidget>() = 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));
|
||||||
|
}
|
||||||
|
}
|
@ -7,9 +7,15 @@ use crate::input::events::TerminalInputEvent;
|
|||||||
/// Components for this module
|
/// Components for this module
|
||||||
pub mod components;
|
pub mod components;
|
||||||
|
|
||||||
|
/// Resources for this module
|
||||||
|
pub mod resources;
|
||||||
|
|
||||||
/// Systems for this module
|
/// Systems for this module
|
||||||
pub(crate) mod systems;
|
pub(crate) mod systems;
|
||||||
|
|
||||||
|
/// Commands for this module
|
||||||
|
pub mod commands;
|
||||||
|
|
||||||
/// Trait which defines an interface for terminal widgets
|
/// Trait which defines an interface for terminal widgets
|
||||||
pub trait TerminalWidget: DowncastSync {
|
pub trait TerminalWidget: DowncastSync {
|
||||||
/// Called every frame to render the widget
|
/// Called every frame to render the widget
|
||||||
|
7
src/widgets/resources.rs
Normal file
7
src/widgets/resources.rs
Normal file
@ -0,0 +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<Entity>);
|
@ -2,20 +2,25 @@ use bevy::prelude::*;
|
|||||||
|
|
||||||
use crate::input::events::TerminalInputEvent;
|
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(
|
pub fn widget_input_handling(
|
||||||
mut widgets: Query<&mut Widget>,
|
mut widgets: Query<&mut Widget>,
|
||||||
mut event_reader: EventReader<TerminalInputEvent>,
|
mut event_reader: EventReader<TerminalInputEvent>,
|
||||||
mut commands: Commands,
|
mut commands: Commands,
|
||||||
|
focused_widget: Res<FocusedWidget>,
|
||||||
) {
|
) {
|
||||||
|
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() {
|
for event in event_reader.read() {
|
||||||
for mut widget in widgets.iter_mut().filter(|widget| widget.enabled) {
|
|
||||||
widget.widget.handle_events(event, &mut commands);
|
widget.widget.handle_events(event, &mut commands);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn update_widgets(mut widgets: Query<&mut Widget>, time: Res<Time>, mut commands: Commands) {
|
pub fn update_widgets(mut widgets: Query<&mut Widget>, time: Res<Time>, mut commands: Commands) {
|
||||||
for mut widget in widgets.iter_mut() {
|
for mut widget in widgets.iter_mut() {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user