From 367d45a9e7abd6f5b8e50f4efa4c3ffad9d72699 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Orhun=20Parmaks=C4=B1z?= Date: Sun, 29 Dec 2024 00:07:26 +0300 Subject: [PATCH] style(tui): add generating animation --- Cargo.lock | 11 ------- git-cliff-tui/Cargo.toml | 1 - git-cliff-tui/src/effect.rs | 11 +++++-- git-cliff-tui/src/main.rs | 4 +++ git-cliff-tui/src/state.rs | 31 ++++++------------ git-cliff-tui/src/ui.rs | 65 ++++++++++--------------------------- 6 files changed, 39 insertions(+), 84 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 41cff6cc38..ad3541fe53 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1229,7 +1229,6 @@ dependencies = [ "lazy_static", "ratatui", "tachyonfx", - "throbber-widgets-tui", "tui-markdown", "unicode-width 0.2.0", ] @@ -3689,16 +3688,6 @@ dependencies = [ "syn", ] -[[package]] -name = "throbber-widgets-tui" -version = "0.8.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1d36b5738d666a2b4c91b7c24998a8588db724b3107258343ebf8824bf55b06d" -dependencies = [ - "rand", - "ratatui", -] - [[package]] name = "time" version = "0.3.36" diff --git a/git-cliff-tui/Cargo.toml b/git-cliff-tui/Cargo.toml index 0ab1e1511b..9b65ea92f5 100644 --- a/git-cliff-tui/Cargo.toml +++ b/git-cliff-tui/Cargo.toml @@ -8,7 +8,6 @@ edition = "2021" [dependencies] ratatui = { version = "0.29.0", features = ["unstable-widget-ref"] } copypasta = "0.10.1" -throbber-widgets-tui = "0.8.0" tui-markdown = "0.3.0" ansi-to-tui = "7.0.0" unicode-width = "0.2.0" diff --git a/git-cliff-tui/src/effect.rs b/git-cliff-tui/src/effect.rs index 7b4413ada7..067c900531 100644 --- a/git-cliff-tui/src/effect.rs +++ b/git-cliff-tui/src/effect.rs @@ -132,15 +132,20 @@ fn create_color_cycle( /// /// An Effect that animates a border around the specified area using cycled /// colors -pub fn create_border_effect(base_color: Color, area: Rect) -> Effect { - let color_cycle = create_color_cycle(base_color, 3); +pub fn create_border_effect( + base_color: Color, + speed: f32, + length: usize, + area: Rect, +) -> Effect { + let color_cycle = create_color_cycle(base_color, length); let effect = fx::effect_fn_buf(Instant::now(), u32::MAX, move |started_at, ctx, buf| { let elapsed = started_at.elapsed().as_secs_f32(); // speed n cells/s - let idx = (elapsed * 30.0) as usize; + let idx = (elapsed * speed) as usize; let area = ctx.area; diff --git a/git-cliff-tui/src/main.rs b/git-cliff-tui/src/main.rs index e0fa89c9b3..2aa2a6b0e7 100644 --- a/git-cliff-tui/src/main.rs +++ b/git-cliff-tui/src/main.rs @@ -40,9 +40,13 @@ fn main() -> Result<()> { if let Ok(event) = receiver.try_recv() { match event { Event::Generate => { + state.is_generating = true; + state.border_effect = None; state.generate_changelog()?; } Event::UpdateChangelog(contents) => { + state.is_generating = false; + state.border_effect = None; state.configs = contents; } Event::Quit => break, diff --git a/git-cliff-tui/src/state.rs b/git-cliff-tui/src/state.rs index 43d4a3b6c3..064d3cf258 100644 --- a/git-cliff-tui/src/state.rs +++ b/git-cliff-tui/src/state.rs @@ -9,7 +9,6 @@ use std::{ thread, }; use tachyonfx::Effect; -use throbber_widgets_tui::ThrobberState; use crate::event::Event; use crate::logo::Logo; @@ -26,25 +25,23 @@ pub struct Config { /// Application state. pub struct State { /// git-cliff arguments. - pub args: Args, + pub args: Args, /// Built-in configuration files. - pub configs: Vec, + pub configs: Vec, /// The state of the list. - pub list_state: ListState, + pub list_state: ListState, /// Event sender. - pub sender: mpsc::Sender, + pub sender: mpsc::Sender, /// Scroll index. - pub scroll_index: usize, + pub scroll_index: usize, /// Clipboard context. - pub clipboard: Option, - /// Throbber state. - pub throbber_state: ThrobberState, + pub clipboard: Option, /// Is generating? - pub is_generating: bool, + pub is_generating: bool, /// Logo widget. - pub logo: Logo, + pub logo: Logo, /// Border effect. - pub border_effect: Option, + pub border_effect: Option, } impl State { @@ -65,7 +62,6 @@ impl State { }, sender, scroll_index: 0, - throbber_state: ThrobberState::default(), clipboard: match ClipboardContext::new() { Ok(ctx) => Some(ctx), Err(e) => { @@ -73,7 +69,7 @@ impl State { None } }, - is_generating: false, + is_generating: true, logo: Logo::default(), args, border_effect: None, @@ -113,11 +109,4 @@ impl State { Ok(()) } - - /// Handles the tick event of the terminal. - pub fn tick(&mut self) { - if self.is_generating { - self.throbber_state.calc_next(); - } - } } diff --git a/git-cliff-tui/src/ui.rs b/git-cliff-tui/src/ui.rs index 9b8a6dfc47..02f67d2b12 100644 --- a/git-cliff-tui/src/ui.rs +++ b/git-cliff-tui/src/ui.rs @@ -12,7 +12,6 @@ use ratatui::{ }, style::{ Color, - Modifier, Style, Stylize, }, @@ -111,26 +110,19 @@ fn render_changelog(state: &mut State, frame: &mut Frame, area: Rect) { .unwrap_or_default(); frame.render_widget( Block::bordered() - .title_top("|Changelog|".yellow().into_left_aligned_line()) + .title_top( + if state.is_generating { + "|Generating|" + } else { + "|Changelog|" + } + .yellow() + .into_left_aligned_line(), + ) .title_bottom( - Line::from(if state.is_generating { + Line::from(if !contents.is_empty() { vec![ "|".fg(Color::Rgb(100, 100, 100)), - "> Generating...".white(), - "|".fg(Color::Rgb(100, 100, 100)), - ] - } else if !contents.is_empty() { - vec![ - "|".fg(Color::Rgb(100, 100, 100)), - state - .list_state - .selected() - .map(|i| state.configs[i].name.clone()) - .unwrap_or_default() - .white() - .italic(), - "|".fg(Color::Rgb(100, 100, 100)), - " |".fg(Color::Rgb(100, 100, 100)), "c".yellow().bold(), "opy".white(), "|".fg(Color::Rgb(100, 100, 100)), @@ -152,13 +144,9 @@ fn render_changelog(state: &mut State, frame: &mut Frame, area: Rect) { "|".fg(Color::Rgb(100, 100, 100)), ] } else { - vec![ - "|".fg(Color::Rgb(100, 100, 100)), - "Select config to start".white(), - "|".fg(Color::Rgb(100, 100, 100)), - ] + vec![] }) - .left_aligned(), + .right_aligned(), ) .border_type(BorderType::Rounded) .border_style(Style::default().fg(Color::Rgb(100, 100, 100))), @@ -189,35 +177,16 @@ fn render_changelog(state: &mut State, frame: &mut Frame, area: Rect) { &mut ScrollbarState::new(contents.len()).position(state.scroll_index), ); - if state.is_generating { - let throbber_area = Rect::new( - area.left().saturating_add(2), - area.bottom().saturating_sub(1), - 1, - 1, - ); - frame.render_stateful_widget( - throbber_widgets_tui::Throbber::default() - .style(Style::default().fg(Color::Yellow)) - .throbber_style( - Style::default() - .fg(Color::Yellow) - .add_modifier(Modifier::BOLD), - ) - .throbber_set(throbber_widgets_tui::BLACK_CIRCLE) - .use_type(throbber_widgets_tui::WhichUse::Spin), - throbber_area, - &mut state.throbber_state, - ); - } - match &mut state.border_effect { Some(effect) => { frame.render_effect(effect, area, Duration::from_millis(100)); } None => { - state.border_effect = - Some(effect::create_border_effect(Color::Gray, area)); + state.border_effect = Some(if state.is_generating { + effect::create_border_effect(Color::Rgb(205, 100, 42), 60., 1, area) + } else { + effect::create_border_effect(Color::Gray, 30., 3, area) + }); } } }