Compare commits
5 Commits
Author | SHA1 | Date | |
---|---|---|---|
9f707cc322 | |||
985e10689d | |||
5cdaf9a396 | |||
16f4ee00c2 | |||
671f74cff8 |
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
|
4
.gitignore
vendored
4
.gitignore
vendored
@ -1 +1,3 @@
|
||||
/target
|
||||
target
|
||||
.cargo/config.toml
|
||||
Cargo.lock
|
||||
|
5210
Cargo.lock
generated
5210
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
15
Cargo.toml
15
Cargo.toml
@ -1,21 +1,28 @@
|
||||
[package]
|
||||
name = "bevy_terminal_dialog"
|
||||
version = "0.2.0"
|
||||
version = "0.2.2"
|
||||
edition = "2021"
|
||||
|
||||
[profile.dev]
|
||||
opt-level = 1
|
||||
|
||||
[profile.dev.package.'*']
|
||||
opt-level = 2
|
||||
|
||||
[dependencies]
|
||||
bevy_terminal_display = "0.6"
|
||||
textwrap = "0.16"
|
||||
zalgo = "0.2"
|
||||
arbitrary-chunks = "0.4"
|
||||
unicode-segmentation = "1.12"
|
||||
anyhow = "1.0"
|
||||
downcast-rs = "1.2"
|
||||
downcast-rs = "2.0"
|
||||
lazy_static = "1.5"
|
||||
|
||||
[dependencies.bevy_terminal_display]
|
||||
version = "0.7"
|
||||
|
||||
[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"
|
||||
|
||||
[dependencies.yarnspinner]
|
||||
|
6
src/events.rs
Normal file
6
src/events.rs
Normal 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);
|
@ -9,11 +9,15 @@ mod systems;
|
||||
pub mod widgets;
|
||||
pub mod util;
|
||||
|
||||
mod events;
|
||||
pub use events::*;
|
||||
|
||||
/// Plugin which provides dialog functionality
|
||||
pub struct TerminalDialogPlugin;
|
||||
|
||||
impl Plugin for TerminalDialogPlugin {
|
||||
fn build(&self, app: &mut App) {
|
||||
app.add_systems(Startup, systems::setup);
|
||||
app.add_systems(Startup, systems::setup).add_event::<OptionSelectedEvent>();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -18,10 +18,9 @@ pub fn style_line(line: &yarnspinner::runtime::Line) -> Vec<(String, Style)> {
|
||||
Style::new(),
|
||||
));
|
||||
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();
|
||||
match attribute.name.as_str() {
|
||||
"style" => {
|
||||
if attribute.name.as_str() == "style" {
|
||||
for (property_name, property_value) in attribute.properties.iter() {
|
||||
match property_name.as_str() {
|
||||
"bold" => {
|
||||
@ -64,8 +63,6 @@ pub fn style_line(line: &yarnspinner::runtime::Line) -> Vec<(String, Style)> {
|
||||
}
|
||||
}
|
||||
}
|
||||
_ => (),
|
||||
}
|
||||
if attribute.name != "character" {
|
||||
line_segments.push((attrib_text, style))
|
||||
}
|
||||
|
@ -5,7 +5,11 @@ use std::time::{Duration, Instant};
|
||||
use arbitrary_chunks::ArbitraryChunks as _;
|
||||
use bevy::prelude::*;
|
||||
use bevy_terminal_display::{
|
||||
crossterm,
|
||||
crossterm::{
|
||||
self,
|
||||
event::{Event, KeyCode, KeyEvent, KeyEventKind},
|
||||
},
|
||||
input::events::TerminalInputEvent,
|
||||
ratatui::{
|
||||
layout::{Constraint, Flex, Layout},
|
||||
style::{Style, Stylize},
|
||||
@ -20,6 +24,8 @@ use bevy_terminal_display::{
|
||||
use unicode_segmentation::UnicodeSegmentation as _;
|
||||
use yarnspinner::runtime::DialogueOption;
|
||||
|
||||
use crate::OptionSelectedEvent;
|
||||
|
||||
/// Dialog box widget marker
|
||||
#[derive(Component)]
|
||||
pub struct DialogBox;
|
||||
@ -45,9 +51,9 @@ impl DialogBoxWidget {
|
||||
pub fn new(character: Option<String>, text: Vec<(String, Style)>, speed: Duration) -> Self {
|
||||
Self {
|
||||
character,
|
||||
character_count: text
|
||||
.iter()
|
||||
.fold(0, |count, (string, _)| count + string.graphemes(true).count()),
|
||||
character_count: text.iter().fold(0, |count, (string, _)| {
|
||||
count + string.graphemes(true).count()
|
||||
}),
|
||||
text,
|
||||
speed,
|
||||
typewriter_index: 0,
|
||||
@ -62,7 +68,9 @@ impl DialogBoxWidget {
|
||||
|
||||
/// Sets the text contained inside the dialog box
|
||||
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.typewriter_index = 0;
|
||||
self.last_character = Instant::now();
|
||||
@ -117,14 +125,15 @@ impl TerminalWidget for DialogBoxWidget {
|
||||
}
|
||||
|
||||
fn update(&mut self, _time: &Time, _commands: &mut Commands) {
|
||||
if self.character_count > 0 && self.typewriter_index < self.character_count {
|
||||
if self.last_character.elapsed() >= self.speed {
|
||||
if self.character_count > 0
|
||||
&& self.typewriter_index < self.character_count
|
||||
&& self.last_character.elapsed() >= self.speed
|
||||
{
|
||||
self.typewriter_index += 1;
|
||||
self.last_character = Instant::now();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Option selection box widget marker
|
||||
#[derive(Component)]
|
||||
@ -220,4 +229,25 @@ impl TerminalWidget for OptionsBoxWidget {
|
||||
frame.render_widget(Clear, area);
|
||||
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(),
|
||||
));
|
||||
}
|
||||
_ => (),
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user