Compare commits

...

41 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
130ddd8117
Fix just pressed 2024-07-31 21:06:04 -04:00
15c97f12e9
Merge branch 'master' of github.com:exvacuum/bevy_terminal_display 2024-07-31 11:45:05 -04:00
ef0771d67c
Add "just pressed" input tracking 2024-07-31 11:44:07 -04:00
4a4b283b93
Update README.md 2024-07-24 15:49:19 -04:00
72da32574e
Removed unused imports in lib.rs 2024-07-24 12:05:01 -04:00
345e0665bf
Update to bevy 0.14 + Add 0BSD Option 2024-07-24 12:02:10 -04:00
f6b9eda55d
Merge branch 'master' of github.com:exvacuum/bevy_terminal_display 2024-06-06 10:27:13 -04:00
8b0ceeb84d
Fixed missing default call in readme 2024-06-06 10:27:00 -04:00
68503cdcc2
Update README.md 2024-06-05 16:06:33 -04:00
04b60ff33e
Removed unnecessary info logs 2024-06-05 13:45:53 -04:00
c30929cbd3
Ensured key release events are processed by input resource first 2024-06-05 13:40:23 -04:00
1addc01cac
Merge branch 'master' of github.com:exvacuum/bevy_terminal_display 2024-06-05 10:54:48 -04:00
2c44bf5efa
Moved widget rendering back into display system 2024-06-05 10:54:40 -04:00
10b09a1748
Moved widget rendering back into display system 2024-06-05 10:53:49 -04:00
8fe2406277
Update README.md 2024-06-05 08:32:23 -04:00
dd0f84a6dc
Update README.md 2024-06-04 19:10:42 -04:00
4f6e38b62e
Merge branch 'master' of github.com:exvacuum/bevy_terminal_display 2024-06-04 15:07:47 -04:00
60b1a6bb4f
Bump dependency versions 2024-06-04 15:07:31 -04:00
19 changed files with 631 additions and 324 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,24 +1,40 @@
[package]
name = "bevy_terminal_display"
version = "0.2.0"
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://git.soaos.dev/soaos/bevy_terminal_display"
[profile.dev]
opt-level = 1
[profile.dev.package."*"]
opt-level = 2
[dependencies]
crossterm = "0.27.0"
downcast-rs = "1.2.1"
once_cell = "1.19.0"
crossbeam-channel = "0.5"
downcast-rs = "2.0"
once_cell = "1.19"
ratatui = "0.29"
color-eyre = "0.6"
leafwing-input-manager = "0.16"
serde = "1.0"
smol_str = "0.2"
[dependencies.bevy]
version = "0.13"
version = "0.15"
default-features = false
features = ["bevy_render"]
[dependencies.bevy_framebuffer_extract]
git = "https://github.com/exvacuum/bevy_framebuffer_extract"
tag = "v0.1.1"
[dependencies.crossterm]
version = "0.28"
features = ["serde"]
# SOAOS DEPS
[dependencies.bevy_dither_post_process]
git = "https://github.com/exvacuum/bevy_dither_post_process"
tag = "v0.1.4"
[dependencies.ratatui]
version = "0.26.2"
version = "0.3"
[dependencies.bevy_headless_render]
version = "0.2"

5
LICENSE-0BSD Normal file
View File

@ -0,0 +1,5 @@
Copyright (C) 2024 by Silas Bartha silas@exvacuum.dev
Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.

View File

@ -1,33 +1,43 @@
# bevy_terminal_display
[![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)
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
- `Widget` component for rendering ratatui TUI widgets
- `TerminalWidget` trait for creating custom TUI widget components
- Logging redirected to `output.log`
- Log redirection
## Screenshots
![](./doc/screenshot.png)
![](doc/screenshot.png)
## Compatibility
| Crate Version | Bevy Version |
|--- |--- |
| ------------- | ------------ |
| 0.6-0.7 | 0.15 |
| 0.4 | 0.14 |
| 0.2 | 0.13 |
## Installation
### crates.io
```toml
[dependencies]
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
@ -42,28 +52,21 @@ fn main() {
.add_plugins((
DefaultPlugins.build().disable::<WinitPlugin>().disable::<LogPlugin>,
ScheduleRunnerPlugin::run_loop(Duration::from_secs_f32(1.0 / 60.0)),
bevy_terminal_display::TerminalDisplayPlugin,
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
));
```
## License
This crate is licensed under your choice of 0BSD, Apache-2.0, or MIT license.

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_framebuffer_extract::{components::{ExtractFramebufferBundle, FramebufferExtractDestination}, render_assets::FramebufferExtractSource};
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,
extract_framebuffer_bundle: ExtractFramebufferBundle,
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(FramebufferExtractSource(image_handle.clone()));
Self {
_terminal_display: TerminalDisplay,
extract_framebuffer_bundle: ExtractFramebufferBundle {
source: framebuffer_extract_source,
dest: FramebufferExtractDestination::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,18 +2,15 @@ use bevy::{
prelude::*,
render::render_resource::{Extent3d, TextureFormat},
};
use bevy_headless_render::{
components::HeadlessRenderDestination, render_assets::HeadlessRenderSource,
};
use crossterm::event::Event;
use bevy_framebuffer_extract::{
components::FramebufferExtractDestination, render_assets::FramebufferExtractSource,
};
use ratatui::{
style::Stylize,
widgets::{Paragraph, Wrap},
};
use ratatui::widgets::{Paragraph, Wrap};
use crate::input::events::TerminalInputEvent;
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;
@ -27,10 +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<&FramebufferExtractDestination>,
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");
@ -45,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;
@ -65,22 +64,32 @@ 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()
.filter(|widget| widget.enabled)
.collect::<Vec<_>>();
active_widgets.sort_by(|a, b| a.depth.cmp(&b.depth));
for mut widget in active_widgets {
widget.widget.render(frame, frame.area());
}
})
.expect("Failed to draw terminal frame");
}
}
/// Utility function to convert a u8 into the corresponding braille character
fn braille_char(mask: u8) -> char {
@ -98,7 +107,7 @@ fn braille_char(mask: u8) -> char {
/// Watches for terminal resize events and resizes the render image accordingly
pub fn resize_handling(
mut images: ResMut<Assets<Image>>,
mut sources: ResMut<Assets<FramebufferExtractSource>>,
mut sources: ResMut<Assets<HeadlessRenderSource>>,
mut event_reader: EventReader<TerminalInputEvent>,
) {
for event in event_reader.read() {

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,48 +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>,
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 is released
pub fn is_released(&self, code: KeyCode) -> bool {
self.released_keys.contains(&code)
}
/// Sets given key to pressed
pub(super) fn press(&mut self, code: KeyCode) {
if !self.is_pressed(code) {
self.pressed_keys.insert(code);
}
}
/// Sets given key to released and removes pressed state
pub(super) fn release(&mut self, code: KeyCode) {
if self.is_pressed(code) {
self.pressed_keys.remove(&code);
}
if !self.is_released(code) {
self.released_keys.insert(code);
}
}
/// Clears all released keys
pub(super) fn clear_released(&mut self) {
self.released_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,10 +1,17 @@
use bevy::prelude::*;
use crossterm::event::{read, Event, KeyEventKind};
use std::cmp::Ordering;
use super::{events::TerminalInputEvent, resources::{EventQueue, TerminalInput}};
use bevy::{
input::{keyboard::KeyboardInput, ButtonState},
prelude::*,
};
use crossterm::event::{read, Event, KeyEvent, KeyEventKind, MediaKeyCode, ModifierKeyCode};
use smol_str::SmolStr;
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 {
@ -24,23 +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_released();
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);
}
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,
});
}
}
event_writer.send(TerminalInputEvent(event));
}
}
}
}
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,23 +4,28 @@
use std::{
fs::OpenOptions,
io::{stdout, Write},
path::PathBuf,
sync::{Arc, Mutex},
};
use bevy::{
log::{
tracing_subscriber::{self, prelude::*, Registry},
Level, LogPlugin,
tracing_subscriber::{self, layer::SubscriberExt, EnvFilter, Layer, Registry},
Level,
},
prelude::*,
utils::tracing::level_filters::LevelFilter,
utils::tracing::subscriber,
};
use bevy_dither_post_process::DitherPostProcessPlugin;
use bevy_framebuffer_extract::FramebufferExtractPlugin;
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
@ -32,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"
@ -50,50 +53,67 @@ 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();
app.add_plugins((
DitherPostProcessPlugin,
FramebufferExtractPlugin,
LogPlugin {
update_subscriber: Some(|_| {
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(LevelFilter::from_level(Level::INFO));
Box::new(Registry::default().with(file_layer))
}),
..Default::default()
},
))
.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();
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,
(
input::systems::input_handling,
display::systems::resize_handling,
(
display::systems::print_to_terminal,
widgets::systems::draw_widgets,
)
.chain(),
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

@ -1,35 +1,29 @@
use bevy::prelude::*;
use crate::{display::resources::Terminal, input::events::TerminalInputEvent};
use crate::input::events::TerminalInputEvent;
use super::components::Widget;
use super::{components::Widget, resources::FocusedWidget};
/// Invokes every enabled widget's `render` method
pub fn draw_widgets(mut terminal: ResMut<Terminal>, mut widgets: Query<&mut Widget>) {
terminal
.0
.draw(|frame| {
let mut active_widgets = widgets
.iter_mut()
.filter(|widget| widget.enabled)
.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());
}
})
.unwrap();
}
/// 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);
}
}