Compare commits

...

12 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
14 changed files with 207 additions and 4761 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

2
.gitignore vendored
View File

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

4656
Cargo.lock generated

File diff suppressed because it is too large Load Diff

View File

@ -1,17 +1,21 @@
[package] [package]
name = "bevy_terminal_display" name = "bevy_terminal_display"
version = "0.6.0" 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://git.exvacuum.dev/bevy_terminal_display" repository = "https://git.soaos.dev/soaos/bevy_terminal_display"
[profile.dev]
opt-level = 1
[profile.dev.package."*"]
opt-level = 2
[dependencies] [dependencies]
crossbeam-channel = "0.5" crossbeam-channel = "0.5"
downcast-rs = "1.2" downcast-rs = "2.0"
once_cell = "1.19" once_cell = "1.19"
bevy_headless_render = "0.2"
bevy_dither_post_process = "0.3"
ratatui = "0.29" ratatui = "0.29"
color-eyre = "0.6" color-eyre = "0.6"
leafwing-input-manager = "0.16" leafwing-input-manager = "0.16"
@ -26,3 +30,11 @@ features = ["bevy_render"]
[dependencies.crossterm] [dependencies.crossterm]
version = "0.28" version = "0.28"
features = ["serde"] features = ["serde"]
# SOAOS DEPS
[dependencies.bevy_dither_post_process]
version = "0.3"
[dependencies.bevy_headless_render]
version = "0.2"

View File

@ -2,8 +2,6 @@
[![Crates](https://img.shields.io/crates/v/bevy_terminal_display)](https://crates.io/crates/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) ![License](https://img.shields.io/badge/license-0BSD%2FMIT%2FApache-blue.svg)
![Tag](https://img.shields.io/github/v/tag/exvacuum/bevy_terminal_display)
[![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. A (very experimental) plugin for the [Bevy](https://bevyengine.org) engine which allows for rendering to a terminal window.
@ -19,12 +17,12 @@ Features Include:
- Log redirection - Log redirection
## Screenshots ## Screenshots
![](https://git.exvacuum.dev/bevy_terminal_display/plain/doc/screenshot.png) ![](doc/screenshot.png)
## Compatibility ## Compatibility
| Crate Version | Bevy Version | | Crate Version | Bevy Version |
| ------------- | ------------ | | ------------- | ------------ |
| 0.6 | 0.15 | | 0.6-0.7 | 0.15 |
| 0.4 | 0.14 | | 0.4 | 0.14 |
| 0.2 | 0.13 | | 0.2 | 0.13 |
@ -33,13 +31,13 @@ Features Include:
### crates.io ### crates.io
```toml ```toml
[dependencies] [dependencies]
bevy_terminal_display = "0.5" 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://git.exvacuum.dev/bevy_terminal_display" git = "https://git.soaos.dev/soaos/bevy_terminal_display"
``` ```
## Example Usage ## Example Usage

View File

@ -1,8 +1,5 @@
use bevy::{ use bevy::{
ecs::{ ecs::{component::ComponentId, world::DeferredWorld},
component::ComponentId,
world::DeferredWorld,
},
prelude::*, prelude::*,
render::render_resource::{ render::render_resource::{
Extent3d, TextureDescriptor, TextureDimension, TextureFormat, TextureUsages, Extent3d, TextureDescriptor, TextureDimension, TextureFormat, TextureUsages,
@ -10,15 +7,40 @@ use bevy::{
}; };
use bevy_dither_post_process::components::DitherPostProcessSettings; use bevy_dither_post_process::components::DitherPostProcessSettings;
use bevy_headless_render::components::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, Debug)] #[derive(Component, Debug)]
#[component(on_add = on_add_terminal_display)] #[component(on_add = on_add_terminal_display)]
pub struct TerminalDisplay(pub u32); pub struct TerminalDisplay {
/// Level of dithering performed on image
pub dither_level: u32,
/// Style applied to rendered text
pub style: Style,
}
fn on_add_terminal_display(mut world: DeferredWorld, entity: Entity, _id: ComponentId) { fn on_add_terminal_display(mut world: DeferredWorld, entity: Entity, _id: ComponentId) {
let asset_server = world.get_resource::<AssetServer>().unwrap(); let asset_server = world.get_resource::<AssetServer>().unwrap();
let dither_level = world.entity(entity).get::<TerminalDisplay>().unwrap().0; let dither_level = world
.entity(entity)
.get::<TerminalDisplay>()
.unwrap()
.dither_level;
let terminal_size = crossterm::terminal::size().unwrap(); let terminal_size = crossterm::terminal::size().unwrap();
let size = Extent3d { let size = Extent3d {
@ -46,17 +68,19 @@ fn on_add_terminal_display(mut world: DeferredWorld, entity: Entity, _id: Compon
image.resize(size); image.resize(size);
let image_handle = asset_server.add(image); let image_handle = asset_server.add(image);
let headless_render_source = HeadlessRenderSource::new(&asset_server, image_handle.clone()); let headless_render_source = HeadlessRenderSource::new(asset_server, image_handle.clone());
let post_process_settings = DitherPostProcessSettings::new(dither_level, &asset_server); let post_process_settings = DitherPostProcessSettings::new(dither_level, asset_server);
world world
.commands() .commands()
.entity(entity) .entity(entity)
.insert((headless_render_source, post_process_settings)); .insert((headless_render_source, post_process_settings));
if let Some(mut camera) = world.entity_mut(entity).get_mut::<Camera>() { if let Some(mut camera) = world.entity_mut(entity).get_mut::<Camera>() {
camera.target = image_handle.into(); camera.target = image_handle.into();
} else { } else {
world.commands().entity(entity).insert(Camera { world.commands().entity(entity).insert(Camera {
target: image_handle.into(), target: image_handle.into(),
hdr: true,
clear_color: ClearColorConfig::Custom(Color::LinearRgba(LinearRgba::BLACK)),
..Default::default() ..Default::default()
}); });
} }

View File

@ -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()))

View File

@ -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,29 +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)
.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

View File

@ -7,11 +7,7 @@ use bevy::{
use crossterm::event::{read, Event, KeyEvent, KeyEventKind, MediaKeyCode, ModifierKeyCode}; use crossterm::event::{read, Event, KeyEvent, KeyEventKind, MediaKeyCode, ModifierKeyCode};
use smol_str::SmolStr; use smol_str::SmolStr;
use super::{ use super::{components::DummyWindow, events::TerminalInputEvent, resources::EventQueue};
components::DummyWindow,
events::TerminalInputEvent,
resources::EventQueue,
};
/// Initializes event queue and thread /// Initializes event queue and thread
pub fn setup_input(mut commands: Commands, event_queue: Res<EventQueue>) { pub fn setup_input(mut commands: Commands, event_queue: Res<EventQueue>) {
@ -294,7 +290,7 @@ fn crossterm_keycode_to_bevy_key(
35 => Some(BKey::F35), 35 => Some(BKey::F35),
_ => None, _ => None,
}, },
CKey::Char(c) => Some(BKey::Character(SmolStr::from(c.encode_utf8(&mut [0;4])))), CKey::Char(c) => Some(BKey::Character(SmolStr::from(c.encode_utf8(&mut [0; 4])))),
CKey::Null => None, CKey::Null => None,
CKey::Esc => Some(BKey::Escape), CKey::Esc => Some(BKey::Escape),
CKey::CapsLock => Some(BKey::CapsLock), CKey::CapsLock => Some(BKey::CapsLock),

View File

@ -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, Write}, 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,43 +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}"); 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, )
widgets::systems::update_widgets, .insert_resource(display::resources::Terminal::default())
), .insert_resource(input::resources::EventQueue::default())
) .init_resource::<widgets::resources::FocusedWidget>()
.insert_resource(display::resources::Terminal::default()) .add_event::<input::events::TerminalInputEvent>();
.insert_resource(input::resources::EventQueue::default())
.add_event::<input::events::TerminalInputEvent>();
} }
} }
fn restore_terminal() -> Result<(), Box<dyn std::error::Error>>{ fn restore_terminal() {
disable_raw_mode()?; let _ = disable_raw_mode();
let mut stdout = stdout(); let mut stdout = stdout();
stdout.execute(PopKeyboardEnhancementFlags)? let _ = stdout
.execute(DisableMouseCapture)? .execute(PopKeyboardEnhancementFlags)
.execute(LeaveAlternateScreen)? .unwrap()
.flush()?; .execute(DisableMouseCapture)
Ok(()) .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 /// Components for this module
pub mod components; pub mod components;
/// Resources for this module
pub mod resources;
/// Systems for this module /// Systems for this module
pub(crate) mod systems; pub(crate) mod systems;
/// Commands for this module
pub mod commands;
/// Trait which defines an interface for terminal widgets /// Trait which defines an interface for terminal widgets
pub trait TerminalWidget: DowncastSync { pub trait TerminalWidget: DowncastSync {
/// Called every frame to render the widget /// Called every frame to render the widget

7
src/widgets/resources.rs Normal file
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,22 @@ 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);
}
}
} }
} }
} }