From 86f3f56edbb53f9158a8ed18cd9cf87d5ddfb727 Mon Sep 17 00:00:00 2001 From: Brendan Mesters <32126115+BrendanMesters@users.noreply.github.com> Date: Wed, 2 Oct 2024 21:20:16 +0200 Subject: [PATCH] feat: add volume control & display (#4) * feat: added volume control I added a simple volume control to the program, using native functionality in `rodio::sink`. This logic has also been linked through to the UI so that users will be aware of this fucntionality (bound to the '-' and '+' keys) and adding a volume readout to the UI as well. * feat: add volume bindings which work without shift A small issue I noticed I had was that I had to press shift to hit '+', I now bound the volume up fucntionality to '+' and '=' and the volume down functionality to '-' and '_', to make both undependant of shift (assuming most default western keyboard layouts) * feat: support arrow keys * feat: add temporarily appearing audio bar * fix: polish input controls --------- Co-authored-by: talwat <83217276+talwat@users.noreply.github.com> --- src/player.rs | 19 ++++++ src/player/ui.rs | 168 ++++++++++++++++++++++++++++++++--------------- 2 files changed, 134 insertions(+), 53 deletions(-) diff --git a/src/player.rs b/src/player.rs index 2c79921..2b5b112 100644 --- a/src/player.rs +++ b/src/player.rs @@ -23,6 +23,7 @@ pub mod downloader; pub mod ui; /// Handles communication between the frontend & audio player. +#[derive(PartialEq)] pub enum Messages { /// Notifies the audio server that it should update the track. Next, @@ -36,6 +37,12 @@ pub enum Messages { /// Pauses the [Sink]. This will also unpause it if it is paused. Pause, + + /// Increase the volume of playback + VolumeUp, + + /// Decrease the volume of playback + VolumeDown, } const TIMEOUT: Duration = Duration::from_secs(8); @@ -183,6 +190,18 @@ impl Player { player.sink.pause(); } } + Messages::VolumeUp => { + // Increase the volume, if possible. + if player.sink.volume() < 1.0 { + player.sink.set_volume(player.sink.volume() + 0.1); + } + } + Messages::VolumeDown => { + // Decreaes the volume, if possible. + if player.sink.volume() > 0.0 { + player.sink.set_volume(player.sink.volume() - 0.1); + } + } } } } diff --git a/src/player/ui.rs b/src/player/ui.rs index 59b5227..b63c548 100644 --- a/src/player/ui.rs +++ b/src/player/ui.rs @@ -1,6 +1,13 @@ //! The module which manages all user interface, including inputs. -use std::{io::stderr, sync::Arc, time::Duration}; +use std::{ + io::stderr, + sync::{ + atomic::{AtomicUsize, Ordering}, + Arc, + }, + time::Duration, +}; use crate::tracks::TrackInfo; @@ -19,10 +26,26 @@ use tokio::{ use super::Messages; +/// The total width of the UI. +const WIDTH: usize = 27; + +/// The width of the progress bar, not including the borders (`[` and `]`) or padding. +const PROGRESS_WIDTH: usize = WIDTH - 16; + +/// The width of the audio bar, again not including borders or padding. +const AUDIO_WIDTH: usize = WIDTH - 17; + +/// Self explanitory. +const FPS: usize = 12; + +/// How long the audio bar will be visible for when audio is adjusted. +/// This is in frames. +const AUDIO_BAR_DURATION: usize = 9; + /// How long to wait in between frames. /// This is fairly arbitrary, but an ideal value should be enough to feel /// snappy but not require too many resources. -const FRAME_DELTA: f32 = 5.0 / 60.0; +const FRAME_DELTA: f32 = 1.0 / FPS as f32; /// Small helper function to format durations. fn format_duration(duration: &Duration) -> String { @@ -61,22 +84,59 @@ impl ActionBar { } } +/// Creates the progress bar, as well as all the padding needed. +fn progress_bar(player: &Arc) -> String { + let mut duration = Duration::new(0, 0); + let elapsed = player.sink.get_pos(); + + let mut filled = 0; + if let Some(current) = player.current.load().as_ref() { + if let Some(x) = current.duration { + duration = x; + + let elapsed = elapsed.as_secs() as f32 / duration.as_secs() as f32; + filled = (elapsed * PROGRESS_WIDTH as f32).round() as usize; + } + }; + + format!( + " [{}{}] {}/{} ", + "/".repeat(filled), + " ".repeat(PROGRESS_WIDTH.saturating_sub(filled)), + format_duration(&elapsed), + format_duration(&duration), + ) +} + +/// Creates the audio bar, as well as all the padding needed. +fn audio_bar(player: &Arc) -> String { + let volume = player.sink.volume(); + + let audio = (player.sink.volume() * AUDIO_WIDTH as f32).round() as usize; + let percentage = format!("{}%", (volume * 100.0).ceil().abs()); + + format!( + " volume: [{}{}] {}{} ", + "/".repeat(audio), + " ".repeat(AUDIO_WIDTH.saturating_sub(audio)), + " ".repeat(4usize.saturating_sub(percentage.len())), + percentage, + ) +} + /// The code for the interface itself. -async fn interface(queue: Arc) -> eyre::Result<()> { - /// The total width of the UI. - const WIDTH: usize = 27; - - /// The width of the progress bar, not including the borders (`[` and `]`) or padding. - const PROGRESS_WIDTH: usize = WIDTH - 16; - +/// +/// `volume_timer` is a bit strange, but it tracks how long the `volume` bar +/// has been displayed for, so that it's only displayed for a certain amount of frames. +async fn interface(player: Arc, volume_timer: Arc) -> eyre::Result<()> { loop { - let (mut main, len) = queue + let (mut main, len) = player .current .load() .as_ref() .map_or(ActionBar::Loading, |x| { let name = (*Arc::clone(x)).clone(); - if queue.sink.is_paused() { + if player.sink.is_paused() { ActionBar::Paused(name) } else { ActionBar::Playing(name) @@ -90,34 +150,26 @@ async fn interface(queue: Arc) -> eyre::Result<()> { main = format!("{}{}", main, " ".repeat(WIDTH - len)); } - let mut duration = Duration::new(0, 0); - let elapsed = queue.sink.get_pos(); - - let mut filled = 0; - if let Some(current) = queue.current.load().as_ref() { - if let Some(x) = current.duration { - duration = x; - - let elapsed = elapsed.as_secs() as f32 / duration.as_secs() as f32; - filled = (elapsed * PROGRESS_WIDTH as f32).round() as usize; - } + let timer = volume_timer.load(Ordering::Relaxed); + let middle = match timer { + 0 => progress_bar(&player), + _ => audio_bar(&player), }; - let progress = format!( - " [{}{}] {}/{} ", - "/".repeat(filled), - " ".repeat(PROGRESS_WIDTH.saturating_sub(filled)), - format_duration(&elapsed), - format_duration(&duration), - ); - let bar = [ + if timer > 0 && timer <= AUDIO_BAR_DURATION { + volume_timer.fetch_add(1, Ordering::Relaxed); + } else if timer > AUDIO_BAR_DURATION { + volume_timer.store(0, Ordering::Relaxed); + } + + let controls = [ format!("{}kip", "[s]".bold()), format!("{}ause", "[p]".bold()), format!("{}uit", "[q]".bold()), ]; // Formats the menu properly - let menu = [main, progress, bar.join(" ")] + let menu = [main, middle, controls.join(" ")] .map(|x| format!("│ {} │\r\n", x.reset()).to_string()); crossterm::execute!(stderr(), Clear(ClearType::FromCursorDown))?; @@ -155,37 +207,47 @@ pub async fn start( crossterm::execute!(stderr(), EnterAlternateScreen, MoveTo(0, 0))?; } - task::spawn(interface(Arc::clone(&queue))); + let volume_timer = Arc::new(AtomicUsize::new(0)); + + task::spawn(interface(Arc::clone(&queue), volume_timer.clone())); loop { let event::Event::Key(event) = event::read()? else { continue; }; - let KeyCode::Char(code) = event.code else { - continue; + let messages = match event.code { + // Arrow key volume controls. + KeyCode::Up | KeyCode::Right => Messages::VolumeUp, + KeyCode::Down | KeyCode::Left => Messages::VolumeDown, + KeyCode::Char(character) => match character { + // Ctrl+C + 'c' if event.modifiers == KeyModifiers::CONTROL => break, + + // Quit + 'q' => break, + + // Skip/Next + 's' | 'n' if !queue.current.load().is_none() => Messages::Next, + + // Pause + 'p' => Messages::Pause, + + // Volume up & down + '+' | '=' => Messages::VolumeUp, + '-' | '_' => Messages::VolumeDown, + _ => continue, + }, + _ => continue, }; - match code { - 'c' => { - // Handles Ctrl+C. - if event.modifiers == KeyModifiers::CONTROL { - break; - } - } - 'q' => { - break; - } - 's' => { - if !queue.current.load().is_none() { - sender.send(Messages::Next).await? - } - } - 'p' => { - sender.send(Messages::Pause).await?; - } - _ => {} + // If it's modifying the volume, then we'll set the `volume_timer` to 1 + // so that the ui thread will know that it should show the audio bar. + if messages == Messages::VolumeDown || messages == Messages::VolumeUp { + volume_timer.store(1, Ordering::Relaxed); } + + sender.send(messages).await?; } if alternate {