120 lines
3.5 KiB
Rust

#![warn(missing_docs)]
//! Bevy plugin which allows a camera to render to a terminal window.
use std::{
fs::OpenOptions,
io::{stdout, Write},
path::PathBuf,
};
use bevy::{
log::{
tracing_subscriber::{self, layer::SubscriberExt, EnvFilter, Layer, Registry},
Level,
},
prelude::*,
utils::tracing::subscriber,
};
use bevy_dither_post_process::DitherPostProcessPlugin;
use bevy_headless_render::HeadlessRenderPlugin;
use color_eyre::config::HookBuilder;
pub use crossterm;
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
pub mod display;
/// Functions and types related to capturing and processing user keyboard input
pub mod input;
/// Functions and types related to constructing and rendering TUI widgets
pub mod widgets;
/// Plugin providing terminal display functionality
pub struct TerminalDisplayPlugin {
/// Path to redirect tracing logs to. Defaults to "debug.log"
pub log_path: PathBuf,
}
impl Default for TerminalDisplayPlugin {
fn default() -> Self {
Self {
log_path: "debug.log".into(),
}
}
}
impl Plugin for TerminalDisplayPlugin {
fn build(&self, app: &mut App) {
let log_path = self.log_path.clone();
let log_file = OpenOptions::new()
.write(true)
.create(true)
.truncate(true)
.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"
)));
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::widget_input_handling,
widgets::systems::update_widgets,
),
)
.insert_resource(display::resources::Terminal::default())
.insert_resource(input::resources::EventQueue::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();
}