Compare commits
21 Commits
Author | SHA1 | Date | |
---|---|---|---|
57a76e4e46 | |||
c356479a11 | |||
faa365ecc5 | |||
f818ae23e1 | |||
4d4e9a652b | |||
8514c007d5 | |||
9bcb260cca | |||
40a1de4b8b | |||
81caa78aa3 | |||
2cacc2f138 | |||
ef2ed44383 | |||
ddfd4a7422 | |||
ac1589f2b9 | |||
6159d531b3 | |||
652a17e0f9 | |||
39e8be3872 | |||
87172fd904 | |||
0d7e360c79 | |||
43b32ff175 | |||
d16c3f67bb | |||
c2d8977233 |
25
.gitea/workflows/build.yaml
Normal file
25
.gitea/workflows/build.yaml
Normal 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
|
50
.github/workflows/docs.yml
vendored
50
.github/workflows/docs.yml
vendored
@ -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
|
|
24
.github/workflows/rust.yml
vendored
24
.github/workflows/rust.yml
vendored
@ -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
2
.gitignore
vendored
@ -1 +1,3 @@
|
|||||||
/target
|
/target
|
||||||
|
Cargo.lock
|
||||||
|
.cargo
|
||||||
|
34
Cargo.toml
34
Cargo.toml
@ -1,22 +1,40 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "bevy_terminal_display"
|
name = "bevy_terminal_display"
|
||||||
version = "0.4.3"
|
version = "0.7.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
license = "0BSD OR MIT OR Apache-2.0"
|
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://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]
|
[dependencies]
|
||||||
crossbeam-channel = "0.5"
|
crossbeam-channel = "0.5"
|
||||||
crossterm = "0.28"
|
downcast-rs = "2.0"
|
||||||
downcast-rs = "1.2"
|
|
||||||
once_cell = "1.19"
|
once_cell = "1.19"
|
||||||
bevy_headless_render = "0.1"
|
ratatui = "0.29"
|
||||||
bevy_dither_post_process = "0.2"
|
|
||||||
ratatui = "0.28"
|
|
||||||
color-eyre = "0.6"
|
color-eyre = "0.6"
|
||||||
|
leafwing-input-manager = "0.16"
|
||||||
|
serde = "1.0"
|
||||||
|
smol_str = "0.2"
|
||||||
|
|
||||||
[dependencies.bevy]
|
[dependencies.bevy]
|
||||||
version = "0.14"
|
version = "0.15"
|
||||||
default-features = false
|
default-features = false
|
||||||
features = ["bevy_render"]
|
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"
|
29
README.md
29
README.md
@ -2,16 +2,13 @@
|
|||||||
|
|
||||||
[](https://crates.io/crates/bevy_terminal_display)
|
[](https://crates.io/crates/bevy_terminal_display)
|
||||||

|

|
||||||

|
|
||||||

|
|
||||||
[](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.
|
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.
|
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:
|
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
|
- Post-process dithers colors to pure black and white, which are then printed as braille characters to the terminal
|
||||||
- Responsiveness to terminal window resizing
|
- Responsiveness to terminal window resizing
|
||||||
- `TerminalInput` resource which keeps track of pressed & released keys
|
- `TerminalInput` resource which keeps track of pressed & released keys
|
||||||
@ -20,12 +17,13 @@ Features Include:
|
|||||||
- Log redirection
|
- Log redirection
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||

|

|
||||||
## Compatibility
|
## Compatibility
|
||||||
|
|
||||||
| Crate Version | Bevy Version |
|
| Crate Version | Bevy Version |
|
||||||
|--- |--- |
|
| ------------- | ------------ |
|
||||||
| 0.3-0.4 | 0.14 |
|
| 0.6-0.7 | 0.15 |
|
||||||
|
| 0.4 | 0.14 |
|
||||||
| 0.2 | 0.13 |
|
| 0.2 | 0.13 |
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
@ -33,13 +31,13 @@ Features Include:
|
|||||||
### crates.io
|
### crates.io
|
||||||
```toml
|
```toml
|
||||||
[dependencies]
|
[dependencies]
|
||||||
bevy_terminal_display = "0.3"
|
bevy_terminal_display = "0.7"
|
||||||
```
|
```
|
||||||
|
|
||||||
### Using git URL in Cargo.toml
|
### Using git URL in Cargo.toml
|
||||||
```toml
|
```toml
|
||||||
[dependencies.bevy_terminal_display]
|
[dependencies.bevy_terminal_display]
|
||||||
git = "https://github.com/exvacuum/bevy_terminal_display.git"
|
git = "https://git.soaos.dev/soaos/bevy_terminal_display"
|
||||||
```
|
```
|
||||||
|
|
||||||
## Example Usage
|
## Example Usage
|
||||||
@ -56,24 +54,15 @@ fn main() {
|
|||||||
ScheduleRunnerPlugin::run_loop(Duration::from_secs_f32(1.0 / 60.0)),
|
ScheduleRunnerPlugin::run_loop(Duration::from_secs_f32(1.0 / 60.0)),
|
||||||
bevy_terminal_display::TerminalDisplayPlugin::default(),
|
bevy_terminal_display::TerminalDisplayPlugin::default(),
|
||||||
))
|
))
|
||||||
.insert_resource(Msaa::Off) // For post-process
|
|
||||||
.run();
|
.run();
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
When spawning a camera:
|
When spawning a camera:
|
||||||
```rs
|
```rs
|
||||||
let terminal_display_bundle = bevy_terminal_display::display::components::TerminalDisplayBundle::new(3, &asset_server);
|
|
||||||
|
|
||||||
commands.spawn((
|
commands.spawn((
|
||||||
Camera3dBundle {
|
// Camera3d...
|
||||||
camera: Camera {
|
TerminalDisplay(3), // Field is level of dithering
|
||||||
target: terminal_display_bundle.image_handle().into(),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
terminal_display_bundle,
|
|
||||||
));
|
));
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -1,72 +1,87 @@
|
|||||||
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_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
|
/// Marker component for terminal display
|
||||||
#[derive(Component)]
|
#[derive(Component, Debug)]
|
||||||
pub struct TerminalDisplay;
|
#[component(on_add = on_add_terminal_display)]
|
||||||
|
pub struct TerminalDisplay {
|
||||||
/// Bundle for terminal display, contains a handle to an image to be used as a render target to
|
/// Level of dithering performed on image
|
||||||
/// render to the terminal
|
pub dither_level: u32,
|
||||||
#[derive(Bundle)]
|
/// Style applied to rendered text
|
||||||
pub struct TerminalDisplayBundle {
|
pub style: Style,
|
||||||
_terminal_display: TerminalDisplay,
|
|
||||||
_headless_render_bundle: HeadlessRenderBundle,
|
|
||||||
_dither_post_process_settings: DitherPostProcessSettings,
|
|
||||||
image_handle: Handle<Image>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TerminalDisplayBundle {
|
fn on_add_terminal_display(mut world: DeferredWorld, entity: Entity, _id: ComponentId) {
|
||||||
/// Create a new terminal display with the given dither level. A higher level exponentially
|
let asset_server = world.get_resource::<AssetServer>().unwrap();
|
||||||
/// increases the size of the bayer matrix used in the ordered dithering calculations. If in
|
let dither_level = world
|
||||||
/// doubt, 3 is a good starting value to test with.
|
.entity(entity)
|
||||||
pub fn new(dither_level: u32, asset_server: &AssetServer) -> Self {
|
.get::<TerminalDisplay>()
|
||||||
let terminal_size = crossterm::terminal::size().unwrap();
|
.unwrap()
|
||||||
let size = Extent3d {
|
.dither_level;
|
||||||
width: (terminal_size.0 as u32) * 2,
|
|
||||||
height: (terminal_size.1 as u32) * 4,
|
|
||||||
depth_or_array_layers: 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut image = Image {
|
let terminal_size = crossterm::terminal::size().unwrap();
|
||||||
texture_descriptor: TextureDescriptor {
|
let size = Extent3d {
|
||||||
label: None,
|
width: (terminal_size.0 as u32) * 2,
|
||||||
size,
|
height: (terminal_size.1 as u32) * 4,
|
||||||
dimension: TextureDimension::D2,
|
depth_or_array_layers: 1,
|
||||||
format: TextureFormat::R8Unorm,
|
};
|
||||||
mip_level_count: 1,
|
|
||||||
sample_count: 1,
|
|
||||||
usage: TextureUsages::TEXTURE_BINDING
|
|
||||||
| TextureUsages::COPY_SRC
|
|
||||||
| TextureUsages::RENDER_ATTACHMENT,
|
|
||||||
view_formats: &[],
|
|
||||||
},
|
|
||||||
..default()
|
|
||||||
};
|
|
||||||
|
|
||||||
image.resize(size);
|
let mut image = Image {
|
||||||
let image_handle = asset_server.add(image);
|
texture_descriptor: TextureDescriptor {
|
||||||
|
label: None,
|
||||||
|
size,
|
||||||
|
dimension: TextureDimension::D2,
|
||||||
|
format: TextureFormat::R8Unorm,
|
||||||
|
mip_level_count: 1,
|
||||||
|
sample_count: 1,
|
||||||
|
usage: TextureUsages::TEXTURE_BINDING
|
||||||
|
| TextureUsages::COPY_SRC
|
||||||
|
| TextureUsages::RENDER_ATTACHMENT,
|
||||||
|
view_formats: &[],
|
||||||
|
},
|
||||||
|
..default()
|
||||||
|
};
|
||||||
|
|
||||||
let framebuffer_extract_source =
|
image.resize(size);
|
||||||
asset_server.add(HeadlessRenderSource(image_handle.clone()));
|
let image_handle = asset_server.add(image);
|
||||||
|
|
||||||
Self {
|
let headless_render_source = HeadlessRenderSource::new(asset_server, image_handle.clone());
|
||||||
_terminal_display: TerminalDisplay,
|
let post_process_settings = DitherPostProcessSettings::new(dither_level, asset_server);
|
||||||
_headless_render_bundle: HeadlessRenderBundle {
|
world
|
||||||
source: framebuffer_extract_source,
|
.commands()
|
||||||
dest: HeadlessRenderDestination::default(),
|
.entity(entity)
|
||||||
},
|
.insert((headless_render_source, post_process_settings));
|
||||||
image_handle,
|
if let Some(mut camera) = world.entity_mut(entity).get_mut::<Camera>() {
|
||||||
_dither_post_process_settings: DitherPostProcessSettings::new(
|
camera.target = image_handle.into();
|
||||||
dither_level,
|
} else {
|
||||||
asset_server,
|
world.commands().entity(entity).insert(Camera {
|
||||||
),
|
target: image_handle.into(),
|
||||||
}
|
hdr: true,
|
||||||
}
|
clear_color: ClearColorConfig::Custom(Color::LinearRgba(LinearRgba::BLACK)),
|
||||||
|
..Default::default()
|
||||||
/// 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()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,9 +20,7 @@ impl Default for Terminal {
|
|||||||
stdout().execute(EnterAlternateScreen).unwrap();
|
stdout().execute(EnterAlternateScreen).unwrap();
|
||||||
stdout().execute(EnableMouseCapture).unwrap();
|
stdout().execute(EnableMouseCapture).unwrap();
|
||||||
stdout()
|
stdout()
|
||||||
.execute(PushKeyboardEnhancementFlags(
|
.execute(PushKeyboardEnhancementFlags(KeyboardEnhancementFlags::all()))
|
||||||
KeyboardEnhancementFlags::REPORT_EVENT_TYPES,
|
|
||||||
))
|
|
||||||
.unwrap();
|
.unwrap();
|
||||||
enable_raw_mode().unwrap();
|
enable_raw_mode().unwrap();
|
||||||
let mut terminal = ratatui::Terminal::new(CrosstermBackend::new(stdout()))
|
let mut terminal = ratatui::Terminal::new(CrosstermBackend::new(stdout()))
|
||||||
|
@ -2,16 +2,15 @@ use bevy::{
|
|||||||
prelude::*,
|
prelude::*,
|
||||||
render::render_resource::{Extent3d, TextureFormat},
|
render::render_resource::{Extent3d, TextureFormat},
|
||||||
};
|
};
|
||||||
use bevy_headless_render::{components::HeadlessRenderDestination, render_assets::HeadlessRenderSource};
|
use bevy_headless_render::{
|
||||||
use crossterm::event::Event;
|
components::HeadlessRenderDestination, render_assets::HeadlessRenderSource,
|
||||||
use ratatui::{
|
|
||||||
style::Stylize,
|
|
||||||
widgets::{Paragraph, Wrap},
|
|
||||||
};
|
};
|
||||||
|
use crossterm::event::Event;
|
||||||
|
use ratatui::widgets::{Paragraph, Wrap};
|
||||||
|
|
||||||
use crate::{input::events::TerminalInputEvent, widgets::components::Widget};
|
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_MIN: u16 = 0x2800;
|
||||||
const BRAILLE_CODE_MAX: u16 = 0x28FF;
|
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
|
/// Prints out the contents of a render image to the terminal as braille characters
|
||||||
pub fn print_to_terminal(
|
pub fn print_to_terminal(
|
||||||
mut terminal: ResMut<Terminal>,
|
mut terminal: ResMut<Terminal>,
|
||||||
image_exports: Query<&HeadlessRenderDestination>,
|
image_exports: Query<(&TerminalDisplay, &HeadlessRenderDestination)>,
|
||||||
mut widgets: Query<&mut Widget>,
|
mut widgets: Query<&mut Widget>,
|
||||||
) {
|
) {
|
||||||
for image_export in image_exports.iter() {
|
let display = image_exports.get_single();
|
||||||
let mut image = image_export
|
let mut output_buffer = Vec::<char>::new();
|
||||||
|
if let Ok((_, image)) = display {
|
||||||
|
let mut image = image
|
||||||
.0
|
.0
|
||||||
.lock()
|
.lock()
|
||||||
.expect("Failed to get lock on output texture");
|
.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 width = image.width();
|
||||||
let height = image.height();
|
let height = image.height();
|
||||||
let data = &image.data;
|
let data = &image.data;
|
||||||
@ -64,30 +64,31 @@ pub fn print_to_terminal(
|
|||||||
output_buffer.push(braille_char(mask));
|
output_buffer.push(braille_char(mask));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let string = output_buffer.into_iter().collect::<String>();
|
let string = output_buffer.into_iter().collect::<String>();
|
||||||
terminal
|
terminal
|
||||||
.0
|
.0
|
||||||
.draw(|frame| {
|
.draw(|frame| {
|
||||||
|
if !string.is_empty() {
|
||||||
frame.render_widget(
|
frame.render_widget(
|
||||||
Paragraph::new(string)
|
Paragraph::new(string)
|
||||||
.white()
|
.style(display.unwrap().0.style)
|
||||||
.bold()
|
|
||||||
.wrap(Wrap { trim: true }),
|
.wrap(Wrap { trim: true }),
|
||||||
frame.area(),
|
frame.area(),
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
let mut active_widgets = widgets
|
let mut active_widgets = widgets
|
||||||
.iter_mut()
|
.iter_mut()
|
||||||
.filter(|widget| widget.enabled)
|
.filter(|widget| widget.enabled)
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
active_widgets.sort_by(|a, b| a.depth.cmp(&b.depth));
|
active_widgets.sort_by(|a, b| a.depth.cmp(&b.depth));
|
||||||
for mut widget in active_widgets {
|
for mut widget in active_widgets {
|
||||||
widget.widget.render(frame, frame.area());
|
widget.widget.render(frame, frame.area());
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.expect("Failed to draw terminal frame");
|
.expect("Failed to draw terminal frame");
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Utility function to convert a u8 into the corresponding braille character
|
/// Utility function to convert a u8 into the corresponding braille character
|
||||||
|
4
src/input/components.rs
Normal file
4
src/input/components.rs
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
use bevy::prelude::*;
|
||||||
|
|
||||||
|
#[derive(Debug, Component)]
|
||||||
|
pub struct DummyWindow;
|
@ -6,3 +6,6 @@ pub mod resources;
|
|||||||
|
|
||||||
/// Systems for this module
|
/// Systems for this module
|
||||||
pub(crate) mod systems;
|
pub(crate) mod systems;
|
||||||
|
|
||||||
|
/// Components for this module
|
||||||
|
pub(crate) mod components;
|
||||||
|
@ -1,56 +1,7 @@
|
|||||||
use bevy::{prelude::*, utils::HashSet};
|
use bevy::prelude::*;
|
||||||
use crossterm::event::{Event, KeyCode};
|
use crossterm::event::Event;
|
||||||
use std::sync::{Arc, Mutex};
|
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
|
/// Event queue for crossterm input event thread
|
||||||
#[derive(Resource, Default)]
|
#[derive(Resource, Default)]
|
||||||
pub(crate) struct EventQueue(pub(super) Arc<Mutex<Vec<Event>>>);
|
pub(crate) struct EventQueue(pub(super) Arc<Mutex<Vec<Event>>>);
|
||||||
|
@ -1,12 +1,17 @@
|
|||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
|
|
||||||
use bevy::prelude::*;
|
use bevy::{
|
||||||
use crossterm::event::{read, Event, KeyEvent, KeyEventKind};
|
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
|
/// 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();
|
let event_queue = event_queue.0.clone();
|
||||||
std::thread::spawn(move || {
|
std::thread::spawn(move || {
|
||||||
loop {
|
loop {
|
||||||
@ -26,30 +31,305 @@ pub fn setup_input(event_queue: Res<EventQueue>) {
|
|||||||
/// Reads events from queue and broadcasts corresponding `TerminalInputEvent`s
|
/// Reads events from queue and broadcasts corresponding `TerminalInputEvent`s
|
||||||
pub fn input_handling(
|
pub fn input_handling(
|
||||||
event_queue: Res<EventQueue>,
|
event_queue: Res<EventQueue>,
|
||||||
mut input: ResMut<TerminalInput>,
|
dummy_window_query: Query<Entity, With<DummyWindow>>,
|
||||||
mut event_writer: EventWriter<TerminalInputEvent>,
|
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 event_queue = event_queue.0.lock().unwrap();
|
||||||
let mut key_events = Vec::<KeyEvent>::new();
|
let mut key_events = Vec::<KeyEvent>::new();
|
||||||
while let Some(event) = event_queue.pop() {
|
while let Some(event) = event_queue.pop() {
|
||||||
if let Event::Key(event) = event {
|
if let Event::Key(event) = event {
|
||||||
key_events.push(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));
|
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 {
|
for event in key_events {
|
||||||
match event.kind {
|
if let Some(key_code) = crossterm_keycode_to_bevy_keycode(event.code) {
|
||||||
KeyEventKind::Press => {
|
if let Some(logical_key) = crossterm_keycode_to_bevy_key(event.code) {
|
||||||
input.press(event.code);
|
match event.kind {
|
||||||
|
KeyEventKind::Press => {
|
||||||
|
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 => {
|
||||||
|
key_event_writer.send(KeyboardInput {
|
||||||
|
key_code,
|
||||||
|
logical_key,
|
||||||
|
state: ButtonState::Released,
|
||||||
|
window,
|
||||||
|
repeat: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
KeyEventKind::Release => {
|
|
||||||
input.release(event.code);
|
|
||||||
}
|
|
||||||
_ => (),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
91
src/lib.rs
91
src/lib.rs
@ -3,22 +3,29 @@
|
|||||||
//! 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, path::PathBuf, sync::{Arc, Mutex}
|
fs::OpenOptions,
|
||||||
|
io::{stdout, Write},
|
||||||
|
path::PathBuf,
|
||||||
};
|
};
|
||||||
|
|
||||||
use bevy::{
|
use bevy::{
|
||||||
log::{
|
log::{
|
||||||
tracing_subscriber::{self, layer::SubscriberExt, EnvFilter, Layer, Registry},
|
tracing_subscriber::{self, layer::SubscriberExt, EnvFilter, Layer, Registry},
|
||||||
Level,
|
Level,
|
||||||
}, prelude::*, utils::tracing::subscriber,
|
},
|
||||||
|
prelude::*,
|
||||||
|
utils::tracing::subscriber,
|
||||||
};
|
};
|
||||||
use bevy_dither_post_process::DitherPostProcessPlugin;
|
use bevy_dither_post_process::DitherPostProcessPlugin;
|
||||||
|
|
||||||
use bevy_headless_render::HeadlessRenderPlugin;
|
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::{
|
||||||
use once_cell::sync::Lazy;
|
event::{DisableMouseCapture, PopKeyboardEnhancementFlags},
|
||||||
|
terminal::{disable_raw_mode, LeaveAlternateScreen},
|
||||||
|
ExecutableCommand,
|
||||||
|
};
|
||||||
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 +37,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,23 +53,20 @@ 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)
|
||||||
.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);
|
let subscriber = Registry::default().with(file_layer);
|
||||||
subscriber::set_global_default(subscriber).unwrap();
|
subscriber::set_global_default(subscriber).unwrap();
|
||||||
|
|
||||||
@ -73,40 +75,45 @@ impl Plugin for TerminalDisplayPlugin {
|
|||||||
let error = error.into_eyre_hook();
|
let error = error.into_eyre_hook();
|
||||||
|
|
||||||
color_eyre::eyre::set_hook(Box::new(move |e| {
|
color_eyre::eyre::set_hook(Box::new(move |e| {
|
||||||
let _ = restore_terminal();
|
restore_terminal();
|
||||||
error(e)
|
error(e)
|
||||||
})).unwrap();
|
}))
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
std::panic::set_hook(Box::new(move |info| {
|
std::panic::set_hook(Box::new(move |info| {
|
||||||
let _ = restore_terminal();
|
restore_terminal();
|
||||||
|
error!("{info}");
|
||||||
panic(info);
|
panic(info);
|
||||||
}));
|
}));
|
||||||
|
|
||||||
app.add_plugins((
|
app.add_plugins((DitherPostProcessPlugin, HeadlessRenderPlugin))
|
||||||
DitherPostProcessPlugin,
|
.add_systems(Startup, input::systems::setup_input)
|
||||||
HeadlessRenderPlugin,
|
.add_systems(
|
||||||
))
|
Update,
|
||||||
.add_systems(Startup, input::systems::setup_input)
|
(
|
||||||
.add_systems(
|
input::systems::input_handling,
|
||||||
Update,
|
display::systems::resize_handling,
|
||||||
(
|
display::systems::print_to_terminal,
|
||||||
input::systems::input_handling,
|
widgets::systems::widget_input_handling,
|
||||||
display::systems::resize_handling,
|
widgets::systems::update_widgets,
|
||||||
display::systems::print_to_terminal,
|
),
|
||||||
widgets::systems::widget_input_handling,
|
)
|
||||||
),
|
.insert_resource(display::resources::Terminal::default())
|
||||||
)
|
.insert_resource(input::resources::EventQueue::default())
|
||||||
.insert_resource(display::resources::Terminal::default())
|
.init_resource::<widgets::resources::FocusedWidget>()
|
||||||
.insert_resource(input::resources::EventQueue::default())
|
.add_event::<input::events::TerminalInputEvent>();
|
||||||
.insert_resource(input::resources::TerminalInput::default())
|
|
||||||
.add_event::<input::events::TerminalInputEvent>();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn restore_terminal() {
|
fn restore_terminal() {
|
||||||
let mut stdout = stdout();
|
|
||||||
let _ = stdout.execute(PopKeyboardEnhancementFlags);
|
|
||||||
let _ = stdout.execute(DisableMouseCapture);
|
|
||||||
let _ = stdout.execute(LeaveAlternateScreen);
|
|
||||||
let _ = disable_raw_mode();
|
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
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 TerminalWidgetCommands for Commands<'_, '_> {
|
||||||
|
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
|
||||||
@ -17,5 +23,8 @@ pub trait TerminalWidget: DowncastSync {
|
|||||||
|
|
||||||
/// Called when a terminal input event is invoked to update any state accordingly
|
/// Called when a terminal input event is invoked to update any state accordingly
|
||||||
fn handle_events(&mut self, _event: &TerminalInputEvent, _commands: &mut Commands) {}
|
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);
|
impl_downcast!(sync TerminalWidget);
|
||||||
|
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,17 +2,28 @@ 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>,
|
||||||
) {
|
) {
|
||||||
for event in event_reader.read() {
|
if let Some(entity) = **focused_widget {
|
||||||
for mut widget in widgets.iter_mut().filter(|widget| widget.enabled) {
|
if let Ok(mut widget) = widgets.get_mut(entity) {
|
||||||
widget.widget.handle_events(event, &mut commands);
|
if widget.enabled {
|
||||||
|
for event in event_reader.read() {
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user