Compare commits

...

23 Commits

Author SHA1 Message Date
57a76e4e46 Merge pull request 'widget-focus' (#4) from widget-focus into master
All checks were successful
Build / Build (push) Successful in 55m46s
Reviewed-on: #4
2025-04-02 18:36:10 -04:00
c356479a11 Merge branch 'master' into widget-focus
All checks were successful
Build / Build (push) Successful in 57m39s
2025-04-01 21:15:05 -04:00
faa365ecc5 update readme
Some checks failed
Build / Build (push) Has been cancelled
2025-04-01 21:08:28 -04:00
f818ae23e1 Apply clippy suggestions 2025-04-01 21:04:56 -04:00
4d4e9a652b Merge pull request 'Widget Focus' (#3) from 11-widget-focus into master
All checks were successful
Build / Build (push) Successful in 1h0m15s
Reviewed-on: #3
2025-03-28 19:33:22 -04:00
8514c007d5 Widget focus
All checks were successful
Build / Build (push) Successful in 59m0s
2025-03-27 19:06:53 -04:00
9bcb260cca WIP - widget focus 2025-03-27 17:19:11 -04:00
40a1de4b8b Sat Mar 22 12:57:47 PM EDT 2025 2025-03-22 12:57:47 -04:00
81caa78aa3 Merge pull request 'Add build action' (#2) from 1-add-build-action into master
All checks were successful
Build / Build (push) Successful in 4m4s
Reviewed-on: #2
2025-03-22 01:09:08 -04:00
2cacc2f138
Updated links
All checks were successful
Build / Build (push) Successful in 1h6m10s
2025-03-21 22:03:23 -04:00
ef2ed44383 Removed patch
Some checks failed
Build / Build (push) Failing after 4m12s
2025-03-20 20:56:04 -04:00
ddfd4a7422
Add build action
Some checks failed
Build / Build (push) Failing after 43s
2025-03-19 07:24:15 -04:00
ac1589f2b9 Fixed patch version 2024-12-20 08:10:02 -05:00
6159d531b3 Re-added vanilla input integration 2024-12-20 08:07:02 -05:00
652a17e0f9 Merge branch 'wip' 2024-12-20 08:02:55 -05:00
39e8be3872 Vanilla Input System Integration 2024-12-20 08:00:20 -05:00
87172fd904 Upgraded to Bevy 0.15 2024-12-11 18:40:02 -05:00
0d7e360c79
Thu Nov 21 04:09:57 PM EST 2024 2024-11-21 16:09:57 -05:00
43b32ff175
Thu Nov 21 04:08:03 PM EST 2024 2024-11-21 16:08:03 -05:00
d16c3f67bb
Thu Nov 21 03:57:24 PM EST 2024 2024-11-21 15:57:24 -05:00
c2d8977233
Thu Nov 21 12:33:58 PM EST 2024 2024-11-21 12:33:58 -05:00
e1eb0d99f1
update dependencies 2024-08-23 20:44:16 -04:00
a1648a1061
Added panic hooks 2024-08-01 14:54:42 -04:00
18 changed files with 595 additions and 299 deletions

View File

@ -0,0 +1,25 @@
name: Build
on: [push]
jobs:
Build:
env:
RUNNER_TOOL_CACHE: /toolcache
container: rust:alpine
steps:
- name: Install node
run: apk add nodejs gcc libc-dev pkgconf libx11-dev alsa-lib-dev eudev-dev tar
- name: Check out repository code
uses: actions/checkout@v4
- name: Restore cache
uses: actions/cache@v4
with:
path: |
~/.cargo/bin/
~/.cargo/registry/index/
~/.cargo/registry/cache/
~/.cargo/git/db/
target/
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
- name: Build
run: cargo build --release

View File

@ -1,50 +0,0 @@
name: Docs
on:
push:
branches: [master]
permissions:
contents: read
pages: write
id-token: write
concurrency:
group: deploy
cancel-in-progress: false
jobs:
build:
name: Build
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Install Dependencies
run: sudo apt-get update; sudo apt-get install --no-install-recommends libasound2-dev libudev-dev
- name: Setup Rust
uses: dtolnay/rust-toolchain@stable
- name: Configure cache
uses: Swatinem/rust-cache@v2
- name: Setup pages
id: pages
uses: actions/configure-pages@v4
- name: Clean docs folder
run: cargo clean --doc
- name: Build docs
run: cargo doc --no-deps
- name: Add redirect
run: echo '<meta http-equiv="refresh" content="0;url=bevy_terminal_display/index.html">' > target/doc/index.html
- name: Remove lock file
run: rm target/doc/.lock
- name: Upload artifact
uses: actions/upload-pages-artifact@v3
with:
path: target/doc
deploy:
name: Deploy
environment:
name: github-pages
url: ${{ steps.deployment.outputs.page_url }}
runs-on: ubuntu-latest
needs: build
steps:
- name: Deploy to GitHub Pages
id: deployment
uses: actions/deploy-pages@v4

View File

@ -1,24 +0,0 @@
name: Rust
on:
push:
branches: [ "master" ]
pull_request:
branches: [ "master" ]
env:
CARGO_TERM_COLOR: always
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Install Dependencies
run: sudo apt-get update; sudo apt-get install --no-install-recommends libasound2-dev libudev-dev
- name: Build
run: cargo build --verbose
- name: Run tests
run: cargo test --verbose

2
.gitignore vendored
View File

@ -1 +1,3 @@
/target
Cargo.lock
.cargo

View File

@ -1,18 +1,40 @@
[package]
name = "bevy_terminal_display"
version = "0.4.1"
version = "0.7.0"
edition = "2021"
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://github.com/exvacuum/bevy_terminal_display"
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"
crossterm = "0.27"
downcast-rs = "1.2"
downcast-rs = "2.0"
once_cell = "1.19"
bevy = "0.14"
bevy_headless_render = "0.1"
bevy_dither_post_process = "0.2"
ratatui = "0.26"
ratatui = "0.29"
color-eyre = "0.6"
leafwing-input-manager = "0.16"
serde = "1.0"
smol_str = "0.2"
[dependencies.bevy]
version = "0.15"
default-features = false
features = ["bevy_render"]
[dependencies.crossterm]
version = "0.28"
features = ["serde"]
# SOAOS DEPS
[dependencies.bevy_dither_post_process]
version = "0.3"
[dependencies.bevy_headless_render]
version = "0.2"

View File

@ -2,16 +2,13 @@
[![Crates](https://img.shields.io/crates/v/bevy_terminal_display)](https://crates.io/crates/bevy_terminal_display)
![License](https://img.shields.io/badge/license-0BSD%2FMIT%2FApache-blue.svg)
![Tag](https://img.shields.io/github/v/tag/exvacuum/bevy_terminal_display)
![Build](https://img.shields.io/github/actions/workflow/status/exvacuum/bevy_terminal_display/rust.yml)
[![Docs](https://img.shields.io/website?url=https%3A%2F%2Fexvacuum.github.io%2Fbevy_terminal_display%2F&label=docs)](https://exvacuum.github.io/bevy_terminal_display)
A (very experimental) plugin for the [Bevy](https://bevyengine.org) engine which allows for rendering to a terminal window.
WARNING: I highly recommend using the kitty terminal emulator with this, not only due to the gpu-accelerated rendering, but also the implementation of the kitty protocol which enables the advanced input detection.
Features Include:
- `TerminalDisplayBundle` automatically sets up a correctly-formatted render texture
- `TerminalDisplay` automatically sets up a correctly-formatted render texture
- Post-process dithers colors to pure black and white, which are then printed as braille characters to the terminal
- Responsiveness to terminal window resizing
- `TerminalInput` resource which keeps track of pressed & released keys
@ -20,12 +17,13 @@ Features Include:
- Log redirection
## Screenshots
![](./doc/screenshot.png)
![](doc/screenshot.png)
## Compatibility
| Crate Version | Bevy Version |
|--- |--- |
| 0.3-0.4 | 0.14 |
| ------------- | ------------ |
| 0.6-0.7 | 0.15 |
| 0.4 | 0.14 |
| 0.2 | 0.13 |
## Installation
@ -33,13 +31,13 @@ Features Include:
### crates.io
```toml
[dependencies]
bevy_terminal_display = "0.3"
bevy_terminal_display = "0.7"
```
### Using git URL in Cargo.toml
```toml
[dependencies.bevy_terminal_display]
git = "https://github.com/exvacuum/bevy_terminal_display.git"
git = "https://git.soaos.dev/soaos/bevy_terminal_display"
```
## Example Usage
@ -56,24 +54,15 @@ fn main() {
ScheduleRunnerPlugin::run_loop(Duration::from_secs_f32(1.0 / 60.0)),
bevy_terminal_display::TerminalDisplayPlugin::default(),
))
.insert_resource(Msaa::Off) // For post-process
.run();
}
```
When spawning a camera:
```rs
let terminal_display_bundle = bevy_terminal_display::display::components::TerminalDisplayBundle::new(3, &asset_server);
commands.spawn((
Camera3dBundle {
camera: Camera {
target: terminal_display_bundle.image_handle().into(),
..Default::default()
},
..Default::default()
},
terminal_display_bundle,
// Camera3d...
TerminalDisplay(3), // Field is level of dithering
));
```

View File

@ -1,26 +1,47 @@
use bevy::{prelude::*, render::render_resource::{Extent3d, TextureDescriptor, TextureDimension, TextureFormat, TextureUsages}};
use bevy::{
ecs::{component::ComponentId, world::DeferredWorld},
prelude::*,
render::render_resource::{
Extent3d, TextureDescriptor, TextureDimension, TextureFormat, TextureUsages,
},
};
use bevy_dither_post_process::components::DitherPostProcessSettings;
use bevy_headless_render::{components::{HeadlessRenderBundle, HeadlessRenderDestination}, render_assets::HeadlessRenderSource};
use bevy_headless_render::components::HeadlessRenderSource;
use ratatui::style::Style;
// TODO: MULTIPLE WINDOWS (probably behind feature flag)
// INFO: need abstraction for launching terminal emulators
//
// /// Structure to refer to a terminal window entity
// #[derive(Clone, Debug)]
// pub enum TerminalWindowRef {
// /// Refers to the primary window created by default in the terminal the command is run in
// Primary,
// /// Direct reference to an terminal window entity
// Entity(Entity),
// }
//
// #[derive(Component, Debug)]
// pub struct TerminalWindow;
/// Marker component for terminal display
#[derive(Component)]
pub struct TerminalDisplay;
/// Bundle for terminal display, contains a handle to an image to be used as a render target to
/// render to the terminal
#[derive(Bundle)]
pub struct TerminalDisplayBundle {
_terminal_display: TerminalDisplay,
_headless_render_bundle: HeadlessRenderBundle,
_dither_post_process_settings: DitherPostProcessSettings,
image_handle: Handle<Image>,
#[derive(Component, Debug)]
#[component(on_add = on_add_terminal_display)]
pub struct TerminalDisplay {
/// Level of dithering performed on image
pub dither_level: u32,
/// Style applied to rendered text
pub style: Style,
}
impl TerminalDisplayBundle {
/// Create a new terminal display with the given dither level. A higher level exponentially
/// increases the size of the bayer matrix used in the ordered dithering calculations. If in
/// doubt, 3 is a good starting value to test with.
pub fn new(dither_level: u32, asset_server: &AssetServer) -> Self {
fn on_add_terminal_display(mut world: DeferredWorld, entity: Entity, _id: ComponentId) {
let asset_server = world.get_resource::<AssetServer>().unwrap();
let dither_level = world
.entity(entity)
.get::<TerminalDisplay>()
.unwrap()
.dither_level;
let terminal_size = crossterm::terminal::size().unwrap();
let size = Extent3d {
width: (terminal_size.0 as u32) * 2,
@ -47,26 +68,20 @@ impl TerminalDisplayBundle {
image.resize(size);
let image_handle = asset_server.add(image);
let framebuffer_extract_source =
asset_server.add(HeadlessRenderSource(image_handle.clone()));
Self {
_terminal_display: TerminalDisplay,
_headless_render_bundle: HeadlessRenderBundle {
source: framebuffer_extract_source,
dest: HeadlessRenderDestination::default(),
},
image_handle,
_dither_post_process_settings: DitherPostProcessSettings::new(
dither_level,
asset_server,
),
}
}
/// Retrieves the handle to this display's target image. Anything written here will be
/// displayed.
pub fn image_handle(&self) -> Handle<Image> {
self.image_handle.clone()
let headless_render_source = HeadlessRenderSource::new(asset_server, image_handle.clone());
let post_process_settings = DitherPostProcessSettings::new(dither_level, asset_server);
world
.commands()
.entity(entity)
.insert((headless_render_source, post_process_settings));
if let Some(mut camera) = world.entity_mut(entity).get_mut::<Camera>() {
camera.target = image_handle.into();
} else {
world.commands().entity(entity).insert(Camera {
target: image_handle.into(),
hdr: true,
clear_color: ClearColorConfig::Custom(Color::LinearRgba(LinearRgba::BLACK)),
..Default::default()
});
}
}

View File

@ -20,9 +20,7 @@ impl Default for Terminal {
stdout().execute(EnterAlternateScreen).unwrap();
stdout().execute(EnableMouseCapture).unwrap();
stdout()
.execute(PushKeyboardEnhancementFlags(
KeyboardEnhancementFlags::REPORT_EVENT_TYPES,
))
.execute(PushKeyboardEnhancementFlags(KeyboardEnhancementFlags::all()))
.unwrap();
enable_raw_mode().unwrap();
let mut terminal = ratatui::Terminal::new(CrosstermBackend::new(stdout()))

View File

@ -2,16 +2,15 @@ use bevy::{
prelude::*,
render::render_resource::{Extent3d, TextureFormat},
};
use bevy_headless_render::{components::HeadlessRenderDestination, render_assets::HeadlessRenderSource};
use crossterm::event::Event;
use ratatui::{
style::Stylize,
widgets::{Paragraph, Wrap},
use bevy_headless_render::{
components::HeadlessRenderDestination, render_assets::HeadlessRenderSource,
};
use crossterm::event::Event;
use ratatui::widgets::{Paragraph, Wrap};
use crate::{input::events::TerminalInputEvent, widgets::components::Widget};
use super::resources::Terminal;
use super::{components::TerminalDisplay, resources::Terminal};
const BRAILLE_CODE_MIN: u16 = 0x2800;
const BRAILLE_CODE_MAX: u16 = 0x28FF;
@ -25,11 +24,13 @@ const BRAILLE_DOT_BIT_POSITIONS: [u8; 8] = [0, 1, 2, 6, 3, 4, 5, 7];
/// Prints out the contents of a render image to the terminal as braille characters
pub fn print_to_terminal(
mut terminal: ResMut<Terminal>,
image_exports: Query<&HeadlessRenderDestination>,
image_exports: Query<(&TerminalDisplay, &HeadlessRenderDestination)>,
mut widgets: Query<&mut Widget>,
) {
for image_export in image_exports.iter() {
let mut image = image_export
let display = image_exports.get_single();
let mut output_buffer = Vec::<char>::new();
if let Ok((_, image)) = display {
let mut image = image
.0
.lock()
.expect("Failed to get lock on output texture");
@ -44,7 +45,6 @@ pub fn print_to_terminal(
};
}
let mut output_buffer = Vec::<char>::new();
let width = image.width();
let height = image.height();
let data = &image.data;
@ -64,18 +64,20 @@ pub fn print_to_terminal(
output_buffer.push(braille_char(mask));
}
}
}
let string = output_buffer.into_iter().collect::<String>();
terminal
.0
.draw(|frame| {
if !string.is_empty() {
frame.render_widget(
Paragraph::new(string)
.white()
.bold()
.style(display.unwrap().0.style)
.wrap(Wrap { trim: true }),
frame.size(),
frame.area(),
);
}
let mut active_widgets = widgets
.iter_mut()
@ -83,11 +85,10 @@ pub fn print_to_terminal(
.collect::<Vec<_>>();
active_widgets.sort_by(|a, b| a.depth.cmp(&b.depth));
for mut widget in active_widgets {
widget.widget.render(frame, frame.size());
widget.widget.render(frame, frame.area());
}
})
.expect("Failed to draw terminal frame");
}
}
/// Utility function to convert a u8 into the corresponding braille character

4
src/input/components.rs Normal file
View File

@ -0,0 +1,4 @@
use bevy::prelude::*;
#[derive(Debug, Component)]
pub struct DummyWindow;

View File

@ -6,3 +6,6 @@ pub mod resources;
/// Systems for this module
pub(crate) mod systems;
/// Components for this module
pub(crate) mod components;

View File

@ -1,56 +1,7 @@
use bevy::{prelude::*, utils::HashSet};
use crossterm::event::{Event, KeyCode};
use bevy::prelude::*;
use crossterm::event::Event;
use std::sync::{Arc, Mutex};
/// Resource containing currently pressed and released keys
#[derive(Resource, Default)]
pub struct TerminalInput {
pressed_keys: HashSet<KeyCode>,
just_pressed_keys: HashSet<KeyCode>,
just_released_keys: HashSet<KeyCode>,
}
impl TerminalInput {
/// Gets whether the given key is pressed
pub fn is_pressed(&self, code: KeyCode) -> bool {
self.pressed_keys.contains(&code)
}
/// Gets whether the given key was just pressed
pub fn just_pressed(&self, code: KeyCode) -> bool {
self.just_pressed_keys.contains(&code)
}
/// Gets whether the given key was just released
pub fn just_released(&self, code: KeyCode) -> bool {
self.just_released_keys.contains(&code)
}
/// Sets given key to pressed
pub(super) fn press(&mut self, code: KeyCode) {
if !self.pressed_keys.contains(&code) {
self.pressed_keys.insert(code);
self.just_pressed_keys.insert(code);
}
}
/// Sets given key to released and removes pressed state
pub(super) fn release(&mut self, code: KeyCode) {
self.pressed_keys.remove(&code);
self.just_released_keys.insert(code);
}
/// Clears all just released keys
pub(super) fn clear_just_released(&mut self) {
self.just_released_keys.clear();
}
/// Clears all just pressed keys
pub(super) fn clear_just_pressed(&mut self) {
self.just_pressed_keys.clear();
}
}
/// Event queue for crossterm input event thread
#[derive(Resource, Default)]
pub(crate) struct EventQueue(pub(super) Arc<Mutex<Vec<Event>>>);

View File

@ -1,12 +1,17 @@
use std::cmp::Ordering;
use bevy::prelude::*;
use crossterm::event::{read, Event, KeyEvent, KeyEventKind};
use bevy::{
input::{keyboard::KeyboardInput, ButtonState},
prelude::*,
};
use crossterm::event::{read, Event, KeyEvent, KeyEventKind, MediaKeyCode, ModifierKeyCode};
use smol_str::SmolStr;
use super::{events::TerminalInputEvent, resources::{EventQueue, TerminalInput}};
use super::{components::DummyWindow, events::TerminalInputEvent, resources::EventQueue};
/// Initializes event queue and thread
pub fn setup_input(event_queue: Res<EventQueue>) {
pub fn setup_input(mut commands: Commands, event_queue: Res<EventQueue>) {
commands.spawn(DummyWindow);
let event_queue = event_queue.0.clone();
std::thread::spawn(move || {
loop {
@ -26,30 +31,305 @@ pub fn setup_input(event_queue: Res<EventQueue>) {
/// Reads events from queue and broadcasts corresponding `TerminalInputEvent`s
pub fn input_handling(
event_queue: Res<EventQueue>,
mut input: ResMut<TerminalInput>,
mut event_writer: EventWriter<TerminalInputEvent>,
dummy_window_query: Query<Entity, With<DummyWindow>>,
mut terminal_event_writer: EventWriter<TerminalInputEvent>,
mut key_event_writer: EventWriter<KeyboardInput>,
) {
input.clear_just_released();
input.clear_just_pressed();
let mut event_queue = event_queue.0.lock().unwrap();
let mut key_events = Vec::<KeyEvent>::new();
while let Some(event) = event_queue.pop() {
if let Event::Key(event) = event {
key_events.push(event);
}
event_writer.send(TerminalInputEvent(event));
terminal_event_writer.send(TerminalInputEvent(event));
}
key_events.sort_by(|&a, &b| a.kind.partial_cmp(&b.kind).unwrap_or(Ordering::Equal));
let window = dummy_window_query.single();
for event in key_events {
if let Some(key_code) = crossterm_keycode_to_bevy_keycode(event.code) {
if let Some(logical_key) = crossterm_keycode_to_bevy_key(event.code) {
match event.kind {
KeyEventKind::Press => {
input.press(event.code);
key_event_writer.send(KeyboardInput {
key_code,
logical_key,
state: ButtonState::Pressed,
window,
repeat: false,
});
}
KeyEventKind::Repeat => {
key_event_writer.send(KeyboardInput {
key_code,
logical_key,
state: ButtonState::Pressed,
window,
repeat: true,
});
}
KeyEventKind::Release => {
input.release(event.code);
key_event_writer.send(KeyboardInput {
key_code,
logical_key,
state: ButtonState::Released,
window,
repeat: false,
});
}
}
}
_ => (),
}
}
}
fn crossterm_keycode_to_bevy_keycode(
crossterm_keycode: crossterm::event::KeyCode,
) -> Option<bevy::input::keyboard::KeyCode> {
use bevy::input::keyboard::KeyCode as BKey;
use crossterm::event::KeyCode as CKey;
match crossterm_keycode {
CKey::Backspace => Some(BKey::Backspace),
CKey::Enter => Some(BKey::Enter),
CKey::Left => Some(BKey::ArrowLeft),
CKey::Right => Some(BKey::ArrowRight),
CKey::Up => Some(BKey::ArrowUp),
CKey::Down => Some(BKey::ArrowDown),
CKey::Home => Some(BKey::Home),
CKey::End => Some(BKey::End),
CKey::PageUp => Some(BKey::PageUp),
CKey::PageDown => Some(BKey::PageDown),
CKey::Tab | CKey::BackTab => Some(BKey::Tab),
CKey::Delete => Some(BKey::Delete),
CKey::Insert => Some(BKey::Insert),
CKey::F(num) => match num {
1 => Some(BKey::F1),
2 => Some(BKey::F2),
3 => Some(BKey::F3),
4 => Some(BKey::F4),
5 => Some(BKey::F5),
6 => Some(BKey::F6),
7 => Some(BKey::F7),
8 => Some(BKey::F8),
9 => Some(BKey::F9),
10 => Some(BKey::F10),
11 => Some(BKey::F11),
12 => Some(BKey::F12),
13 => Some(BKey::F13),
14 => Some(BKey::F14),
15 => Some(BKey::F15),
16 => Some(BKey::F16),
17 => Some(BKey::F17),
18 => Some(BKey::F18),
19 => Some(BKey::F19),
20 => Some(BKey::F20),
21 => Some(BKey::F21),
22 => Some(BKey::F22),
23 => Some(BKey::F23),
24 => Some(BKey::F24),
25 => Some(BKey::F25),
26 => Some(BKey::F26),
27 => Some(BKey::F27),
28 => Some(BKey::F28),
29 => Some(BKey::F29),
30 => Some(BKey::F30),
31 => Some(BKey::F31),
32 => Some(BKey::F32),
33 => Some(BKey::F33),
34 => Some(BKey::F34),
35 => Some(BKey::F35),
_ => None,
},
CKey::Char(c) => match c {
'1' | '!' => Some(BKey::Digit1),
'2' | '@' => Some(BKey::Digit2),
'3' | '#' => Some(BKey::Digit3),
'4' | '$' => Some(BKey::Digit4),
'5' | '%' => Some(BKey::Digit5),
'6' | '^' => Some(BKey::Digit5),
'7' | '&' => Some(BKey::Digit7),
'8' | '*' => Some(BKey::Digit8),
'9' | '(' => Some(BKey::Digit9),
'0' | ')' => Some(BKey::Digit0),
'-' | '_' => Some(BKey::Minus),
'=' | '+' => Some(BKey::Equal),
'`' | '~' => Some(BKey::Backquote),
'q' | 'Q' => Some(BKey::KeyQ),
'w' | 'W' => Some(BKey::KeyW),
'e' | 'E' => Some(BKey::KeyE),
'r' | 'R' => Some(BKey::KeyR),
't' | 'T' => Some(BKey::KeyT),
'y' | 'Y' => Some(BKey::KeyY),
'u' | 'U' => Some(BKey::KeyU),
'i' | 'I' => Some(BKey::KeyI),
'o' | 'O' => Some(BKey::KeyO),
'p' | 'P' => Some(BKey::KeyP),
'[' | '{' => Some(BKey::BracketLeft),
']' | '}' => Some(BKey::BracketRight),
'a' | 'A' => Some(BKey::KeyA),
's' | 'S' => Some(BKey::KeyS),
'd' | 'D' => Some(BKey::KeyD),
'f' | 'F' => Some(BKey::KeyF),
'g' | 'G' => Some(BKey::KeyG),
'h' | 'H' => Some(BKey::KeyH),
'j' | 'J' => Some(BKey::KeyJ),
'k' | 'K' => Some(BKey::KeyK),
'l' | 'L' => Some(BKey::KeyL),
';' | ':' => Some(BKey::Semicolon),
'\'' | '"' => Some(BKey::Slash),
'z' | 'Z' => Some(BKey::KeyZ),
'x' | 'X' => Some(BKey::KeyX),
'c' | 'C' => Some(BKey::KeyC),
'v' | 'V' => Some(BKey::KeyV),
'b' | 'B' => Some(BKey::KeyB),
'n' | 'N' => Some(BKey::KeyN),
'm' | 'M' => Some(BKey::KeyM),
',' | '<' => Some(BKey::Comma),
'.' | '>' => Some(BKey::Period),
'/' | '?' => Some(BKey::Slash),
' ' => Some(BKey::Space),
_ => None,
},
CKey::Null => None,
CKey::Esc => Some(BKey::Escape),
CKey::CapsLock => Some(BKey::CapsLock),
CKey::ScrollLock => Some(BKey::ScrollLock),
CKey::NumLock => Some(BKey::NumLock),
CKey::PrintScreen => Some(BKey::PrintScreen),
CKey::Pause => Some(BKey::Pause),
CKey::Menu => Some(BKey::ContextMenu),
CKey::KeypadBegin => None,
CKey::Media(media) => match media {
MediaKeyCode::Play => Some(BKey::MediaPlayPause),
MediaKeyCode::Pause => Some(BKey::Pause),
MediaKeyCode::PlayPause => Some(BKey::MediaPlayPause),
MediaKeyCode::Reverse => None,
MediaKeyCode::Stop => Some(BKey::MediaStop),
MediaKeyCode::FastForward => Some(BKey::MediaTrackNext),
MediaKeyCode::Rewind => Some(BKey::MediaTrackPrevious),
MediaKeyCode::TrackNext => Some(BKey::MediaTrackNext),
MediaKeyCode::TrackPrevious => Some(BKey::MediaTrackPrevious),
MediaKeyCode::Record => None,
MediaKeyCode::LowerVolume => Some(BKey::AudioVolumeDown),
MediaKeyCode::RaiseVolume => Some(BKey::AudioVolumeUp),
MediaKeyCode::MuteVolume => Some(BKey::AudioVolumeMute),
},
CKey::Modifier(modifier) => match modifier {
ModifierKeyCode::LeftShift => Some(BKey::ShiftLeft),
ModifierKeyCode::LeftControl => Some(BKey::ControlLeft),
ModifierKeyCode::LeftAlt => Some(BKey::AltLeft),
ModifierKeyCode::LeftSuper => Some(BKey::SuperLeft),
ModifierKeyCode::LeftHyper => Some(BKey::Hyper),
ModifierKeyCode::LeftMeta => Some(BKey::Meta),
ModifierKeyCode::RightShift => Some(BKey::ShiftRight),
ModifierKeyCode::RightControl => Some(BKey::ControlRight),
ModifierKeyCode::RightAlt => Some(BKey::AltLeft),
ModifierKeyCode::RightSuper => Some(BKey::SuperRight),
ModifierKeyCode::RightHyper => Some(BKey::Hyper),
ModifierKeyCode::RightMeta => Some(BKey::Meta),
ModifierKeyCode::IsoLevel3Shift => None,
ModifierKeyCode::IsoLevel5Shift => None,
},
}
}
fn crossterm_keycode_to_bevy_key(
crossterm_keycode: crossterm::event::KeyCode,
) -> Option<bevy::input::keyboard::Key> {
use bevy::input::keyboard::Key as BKey;
use crossterm::event::KeyCode as CKey;
match crossterm_keycode {
CKey::Backspace => Some(BKey::Backspace),
CKey::Enter => Some(BKey::Enter),
CKey::Left => Some(BKey::ArrowLeft),
CKey::Right => Some(BKey::ArrowRight),
CKey::Up => Some(BKey::ArrowUp),
CKey::Down => Some(BKey::ArrowDown),
CKey::Home => Some(BKey::Home),
CKey::End => Some(BKey::End),
CKey::PageUp => Some(BKey::PageUp),
CKey::PageDown => Some(BKey::PageDown),
CKey::Tab | CKey::BackTab => Some(BKey::Tab),
CKey::Delete => Some(BKey::Delete),
CKey::Insert => Some(BKey::Insert),
CKey::F(num) => match num {
1 => Some(BKey::F1),
2 => Some(BKey::F2),
3 => Some(BKey::F3),
4 => Some(BKey::F4),
5 => Some(BKey::F5),
6 => Some(BKey::F6),
7 => Some(BKey::F7),
8 => Some(BKey::F8),
9 => Some(BKey::F9),
10 => Some(BKey::F10),
11 => Some(BKey::F11),
12 => Some(BKey::F12),
13 => Some(BKey::F13),
14 => Some(BKey::F14),
15 => Some(BKey::F15),
16 => Some(BKey::F16),
17 => Some(BKey::F17),
18 => Some(BKey::F18),
19 => Some(BKey::F19),
20 => Some(BKey::F20),
21 => Some(BKey::F21),
22 => Some(BKey::F22),
23 => Some(BKey::F23),
24 => Some(BKey::F24),
25 => Some(BKey::F25),
26 => Some(BKey::F26),
27 => Some(BKey::F27),
28 => Some(BKey::F28),
29 => Some(BKey::F29),
30 => Some(BKey::F30),
31 => Some(BKey::F31),
32 => Some(BKey::F32),
33 => Some(BKey::F33),
34 => Some(BKey::F34),
35 => Some(BKey::F35),
_ => None,
},
CKey::Char(c) => Some(BKey::Character(SmolStr::from(c.encode_utf8(&mut [0; 4])))),
CKey::Null => None,
CKey::Esc => Some(BKey::Escape),
CKey::CapsLock => Some(BKey::CapsLock),
CKey::ScrollLock => Some(BKey::ScrollLock),
CKey::NumLock => Some(BKey::NumLock),
CKey::PrintScreen => Some(BKey::PrintScreen),
CKey::Pause => Some(BKey::Pause),
CKey::Menu => Some(BKey::ContextMenu),
CKey::KeypadBegin => None,
CKey::Media(media) => match media {
MediaKeyCode::Play => Some(BKey::MediaPlayPause),
MediaKeyCode::Pause => Some(BKey::Pause),
MediaKeyCode::PlayPause => Some(BKey::MediaPlayPause),
MediaKeyCode::Reverse => None,
MediaKeyCode::Stop => Some(BKey::MediaStop),
MediaKeyCode::FastForward => Some(BKey::MediaTrackNext),
MediaKeyCode::Rewind => Some(BKey::MediaTrackPrevious),
MediaKeyCode::TrackNext => Some(BKey::MediaTrackNext),
MediaKeyCode::TrackPrevious => Some(BKey::MediaTrackPrevious),
MediaKeyCode::Record => None,
MediaKeyCode::LowerVolume => Some(BKey::AudioVolumeDown),
MediaKeyCode::RaiseVolume => Some(BKey::AudioVolumeUp),
MediaKeyCode::MuteVolume => Some(BKey::AudioVolumeMute),
},
CKey::Modifier(modifier) => match modifier {
ModifierKeyCode::LeftShift => Some(BKey::Shift),
ModifierKeyCode::LeftControl => Some(BKey::Control),
ModifierKeyCode::LeftAlt => Some(BKey::Alt),
ModifierKeyCode::LeftSuper => Some(BKey::Super),
ModifierKeyCode::LeftHyper => Some(BKey::Hyper),
ModifierKeyCode::LeftMeta => Some(BKey::Meta),
ModifierKeyCode::RightShift => Some(BKey::Shift),
ModifierKeyCode::RightControl => Some(BKey::Control),
ModifierKeyCode::RightAlt => Some(BKey::Alt),
ModifierKeyCode::RightSuper => Some(BKey::Super),
ModifierKeyCode::RightHyper => Some(BKey::Hyper),
ModifierKeyCode::RightMeta => Some(BKey::Meta),
ModifierKeyCode::IsoLevel3Shift => Some(BKey::AltGraph),
ModifierKeyCode::IsoLevel5Shift => None,
},
}
}

View File

@ -4,21 +4,28 @@
use std::{
fs::OpenOptions,
io::{stdout, Write},
path::PathBuf,
sync::{Arc, Mutex},
};
use bevy::{
log::{
tracing_subscriber::{self, layer::SubscriberExt, EnvFilter, Layer, Registry},
Level,
}, prelude::*, utils::tracing::subscriber,
},
prelude::*,
utils::tracing::subscriber,
};
use bevy_dither_post_process::DitherPostProcessPlugin;
use bevy_headless_render::HeadlessRenderPlugin;
use color_eyre::config::HookBuilder;
pub use crossterm;
use once_cell::sync::Lazy;
use crossterm::{
event::{DisableMouseCapture, PopKeyboardEnhancementFlags},
terminal::{disable_raw_mode, LeaveAlternateScreen},
ExecutableCommand,
};
pub use ratatui;
/// Functions and types related to capture and display of world to terminal
@ -30,8 +37,6 @@ pub mod input;
/// Functions and types related to constructing and rendering TUI widgets
pub mod widgets;
static LOG_PATH: Lazy<Arc<Mutex<PathBuf>>> = 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,29 +53,40 @@ 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)
.with_filter(EnvFilter::builder().parse_lossy(format!("{},{}", Level::INFO, "wgpu=error,naga=warn")));
.with_filter(EnvFilter::builder().parse_lossy(format!(
"{},{}",
Level::INFO,
"wgpu=error,naga=warn"
)));
let subscriber = Registry::default().with(file_layer);
subscriber::set_global_default(subscriber).unwrap();
app.add_plugins((
DitherPostProcessPlugin,
HeadlessRenderPlugin,
))
let (panic, error) = HookBuilder::default().into_hooks();
let panic = panic.into_panic_hook();
let error = error.into_eyre_hook();
color_eyre::eyre::set_hook(Box::new(move |e| {
restore_terminal();
error(e)
}))
.unwrap();
std::panic::set_hook(Box::new(move |info| {
restore_terminal();
error!("{info}");
panic(info);
}));
app.add_plugins((DitherPostProcessPlugin, HeadlessRenderPlugin))
.add_systems(Startup, input::systems::setup_input)
.add_systems(
Update,
@ -79,11 +95,25 @@ impl Plugin for TerminalDisplayPlugin {
display::systems::resize_handling,
display::systems::print_to_terminal,
widgets::systems::widget_input_handling,
widgets::systems::update_widgets,
),
)
.insert_resource(display::resources::Terminal::default())
.insert_resource(input::resources::EventQueue::default())
.insert_resource(input::resources::TerminalInput::default())
.init_resource::<widgets::resources::FocusedWidget>()
.add_event::<input::events::TerminalInputEvent>();
}
}
fn restore_terminal() {
let _ = disable_raw_mode();
let mut stdout = stdout();
let _ = stdout
.execute(PopKeyboardEnhancementFlags)
.unwrap()
.execute(DisableMouseCapture)
.unwrap()
.execute(LeaveAlternateScreen)
.unwrap()
.flush();
}

23
src/widgets/commands.rs Normal file
View 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 TerminalWidgetCommands for Commands<'_, '_> {
fn focus_widget(&mut self, widget: Entity) {
self.queue(FocusWidgetCommand(widget));
}
}

View File

@ -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
@ -17,5 +23,8 @@ pub trait TerminalWidget: DowncastSync {
/// Called when a terminal input event is invoked to update any state accordingly
fn handle_events(&mut self, _event: &TerminalInputEvent, _commands: &mut Commands) {}
/// Called every frame during the Update schedule
fn update(&mut self, _time: &Time, _commands: &mut Commands) {}
}
impl_downcast!(sync TerminalWidget);

7
src/widgets/resources.rs Normal file
View 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>);

View File

@ -2,17 +2,28 @@ 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<TerminalInputEvent>,
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 {
for event in event_reader.read() {
for mut widget in widgets.iter_mut().filter(|widget| widget.enabled) {
widget.widget.handle_events(event, &mut commands);
}
}
}
}
}
pub fn update_widgets(mut widgets: Query<&mut Widget>, time: Res<Time>, mut commands: Commands) {
for mut widget in widgets.iter_mut() {
widget.widget.update(&time, &mut commands);
}
}