Compare commits

...

5 Commits

Author SHA1 Message Date
9f707cc322 Merge pull request 'option-select-event' (#3) from option-select-event into master
All checks were successful
Build / Build (push) Successful in 1h3m3s
Reviewed-on: #3
2025-04-02 20:51:21 -04:00
985e10689d option selection event handling
All checks were successful
Build / Build (push) Successful in 1h2m29s
2025-04-01 20:45:09 -04:00
5cdaf9a396 WIP - widget focus 2025-03-27 17:19:11 -04:00
16f4ee00c2 Merge pull request 'Add build action' (#2) from 1-add-build-action into master
All checks were successful
Build / Build (push) Successful in 3m39s
Reviewed-on: #2
2025-03-20 20:53:34 -04:00
671f74cff8
Added build action
All checks were successful
Build / Build (push) Successful in 1h14m44s
2025-03-19 07:13:24 -04:00
8 changed files with 128 additions and 5267 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

4
.gitignore vendored
View File

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

5210
Cargo.lock generated

File diff suppressed because it is too large Load Diff

View File

@ -1,21 +1,28 @@
[package] [package]
name = "bevy_terminal_dialog" name = "bevy_terminal_dialog"
version = "0.2.0" version = "0.2.2"
edition = "2021" edition = "2021"
[profile.dev]
opt-level = 1
[profile.dev.package.'*']
opt-level = 2
[dependencies] [dependencies]
bevy_terminal_display = "0.6"
textwrap = "0.16" textwrap = "0.16"
zalgo = "0.2" zalgo = "0.2"
arbitrary-chunks = "0.4" arbitrary-chunks = "0.4"
unicode-segmentation = "1.12" unicode-segmentation = "1.12"
anyhow = "1.0" anyhow = "1.0"
downcast-rs = "1.2" downcast-rs = "2.0"
lazy_static = "1.5" lazy_static = "1.5"
[dependencies.bevy_terminal_display]
version = "0.7"
[dependencies.bevy_basic_interaction] [dependencies.bevy_basic_interaction]
git = "https://git.exvacuum.dev/bevy_basic_interaction" git = "https://git.soaos.dev/soaos/bevy_basic_interaction"
tag = "v0.2.0" tag = "v0.2.0"
[dependencies.yarnspinner] [dependencies.yarnspinner]

6
src/events.rs Normal file
View File

@ -0,0 +1,6 @@
use bevy::prelude::*;
use yarnspinner::runtime::DialogueOption;
/// Event called when a dialog option is selected
#[derive(Event, Debug)]
pub struct OptionSelectedEvent(pub DialogueOption);

View File

@ -9,11 +9,15 @@ mod systems;
pub mod widgets; pub mod widgets;
pub mod util; pub mod util;
mod events;
pub use events::*;
/// Plugin which provides dialog functionality /// Plugin which provides dialog functionality
pub struct TerminalDialogPlugin; pub struct TerminalDialogPlugin;
impl Plugin for TerminalDialogPlugin { impl Plugin for TerminalDialogPlugin {
fn build(&self, app: &mut App) { fn build(&self, app: &mut App) {
app.add_systems(Startup, systems::setup); app.add_systems(Startup, systems::setup).add_event::<OptionSelectedEvent>();
} }
} }

View File

@ -18,53 +18,50 @@ pub fn style_line(line: &yarnspinner::runtime::Line) -> Vec<(String, Style)> {
Style::new(), Style::new(),
)); ));
for (i, attribute) in attributes.iter().enumerate() { for (i, attribute) in attributes.iter().enumerate() {
let mut attrib_text = line.text_for_attribute(&attribute).to_string(); let mut attrib_text = line.text_for_attribute(attribute).to_string();
let mut style = Style::new(); let mut style = Style::new();
match attribute.name.as_str() { if attribute.name.as_str() == "style" {
"style" => { for (property_name, property_value) in attribute.properties.iter() {
for (property_name, property_value) in attribute.properties.iter() { match property_name.as_str() {
match property_name.as_str() { "bold" => {
"bold" => { if let MarkupValue::Bool(value) = property_value {
if let MarkupValue::Bool(value) = property_value { if *value {
if *value { style = style.bold();
style = style.bold();
}
} }
} }
"italic" => {
if let MarkupValue::Bool(value) = property_value {
if *value {
style = style.italic();
}
}
}
"color" => {
if let MarkupValue::Integer(value) = property_value {
style = style.fg(Color::Indexed(*value as u8))
}
}
"zalgo" => {
if let MarkupValue::Bool(value) = property_value {
if *value {
let mut generator = Generator::new();
let mut out = String::new();
let args =
GeneratorArgs::new(true, true, true, ZalgoSize::Mini);
generator.gen(&attrib_text, &mut out, &args);
attrib_text = out;
}
}
}
"bg" => {
if let MarkupValue::Integer(value) = property_value {
style = style.bg(Color::Indexed(*value as u8))
}
}
_ => (),
} }
"italic" => {
if let MarkupValue::Bool(value) = property_value {
if *value {
style = style.italic();
}
}
}
"color" => {
if let MarkupValue::Integer(value) = property_value {
style = style.fg(Color::Indexed(*value as u8))
}
}
"zalgo" => {
if let MarkupValue::Bool(value) = property_value {
if *value {
let mut generator = Generator::new();
let mut out = String::new();
let args =
GeneratorArgs::new(true, true, true, ZalgoSize::Mini);
generator.gen(&attrib_text, &mut out, &args);
attrib_text = out;
}
}
}
"bg" => {
if let MarkupValue::Integer(value) = property_value {
style = style.bg(Color::Indexed(*value as u8))
}
}
_ => (),
} }
} }
_ => (),
} }
if attribute.name != "character" { if attribute.name != "character" {
line_segments.push((attrib_text, style)) line_segments.push((attrib_text, style))

View File

@ -5,7 +5,11 @@ use std::time::{Duration, Instant};
use arbitrary_chunks::ArbitraryChunks as _; use arbitrary_chunks::ArbitraryChunks as _;
use bevy::prelude::*; use bevy::prelude::*;
use bevy_terminal_display::{ use bevy_terminal_display::{
crossterm, crossterm::{
self,
event::{Event, KeyCode, KeyEvent, KeyEventKind},
},
input::events::TerminalInputEvent,
ratatui::{ ratatui::{
layout::{Constraint, Flex, Layout}, layout::{Constraint, Flex, Layout},
style::{Style, Stylize}, style::{Style, Stylize},
@ -20,6 +24,8 @@ use bevy_terminal_display::{
use unicode_segmentation::UnicodeSegmentation as _; use unicode_segmentation::UnicodeSegmentation as _;
use yarnspinner::runtime::DialogueOption; use yarnspinner::runtime::DialogueOption;
use crate::OptionSelectedEvent;
/// Dialog box widget marker /// Dialog box widget marker
#[derive(Component)] #[derive(Component)]
pub struct DialogBox; pub struct DialogBox;
@ -45,9 +51,9 @@ impl DialogBoxWidget {
pub fn new(character: Option<String>, text: Vec<(String, Style)>, speed: Duration) -> Self { pub fn new(character: Option<String>, text: Vec<(String, Style)>, speed: Duration) -> Self {
Self { Self {
character, character,
character_count: text character_count: text.iter().fold(0, |count, (string, _)| {
.iter() count + string.graphemes(true).count()
.fold(0, |count, (string, _)| count + string.graphemes(true).count()), }),
text, text,
speed, speed,
typewriter_index: 0, typewriter_index: 0,
@ -62,7 +68,9 @@ impl DialogBoxWidget {
/// Sets the text contained inside the dialog box /// Sets the text contained inside the dialog box
pub fn set_text(&mut self, text: Vec<(String, Style)>) { pub fn set_text(&mut self, text: Vec<(String, Style)>) {
self.character_count = text.iter().fold(0, |count, (string, _)| count + string.graphemes(true).count()); self.character_count = text.iter().fold(0, |count, (string, _)| {
count + string.graphemes(true).count()
});
self.text = text; self.text = text;
self.typewriter_index = 0; self.typewriter_index = 0;
self.last_character = Instant::now(); self.last_character = Instant::now();
@ -117,11 +125,12 @@ impl TerminalWidget for DialogBoxWidget {
} }
fn update(&mut self, _time: &Time, _commands: &mut Commands) { fn update(&mut self, _time: &Time, _commands: &mut Commands) {
if self.character_count > 0 && self.typewriter_index < self.character_count { if self.character_count > 0
if self.last_character.elapsed() >= self.speed { && self.typewriter_index < self.character_count
self.typewriter_index += 1; && self.last_character.elapsed() >= self.speed
self.last_character = Instant::now(); {
} self.typewriter_index += 1;
self.last_character = Instant::now();
} }
} }
} }
@ -220,4 +229,25 @@ impl TerminalWidget for OptionsBoxWidget {
frame.render_widget(Clear, area); frame.render_widget(Clear, area);
frame.render_stateful_widget(list, area, &mut self.state); frame.render_stateful_widget(list, area, &mut self.state);
} }
fn handle_events(&mut self, event: &TerminalInputEvent, commands: &mut Commands) {
if let TerminalInputEvent(Event::Key(KeyEvent { code, kind, .. })) = event {
if kind == &KeyEventKind::Press {
match code {
KeyCode::Up => {
self.state.select_previous();
}
KeyCode::Down => {
self.state.select_next();
}
KeyCode::Char('e') => {
commands.send_event(OptionSelectedEvent(
self.options[self.state.selected().unwrap()].0.clone(),
));
}
_ => (),
}
}
}
}
} }