Compare commits

...

47 Commits

Author SHA1 Message Date
arabianq 330c3d79d4 cargo update 2026-01-28 03:40:43 +03:00
arabianq dff20daace deps: bump clap to 4.5.55 2026-01-28 03:39:39 +03:00
arabianq cdc44328a8 docs: update README to include new features for collapsible audio tracks, drag and drop directories, and automatic device detection 2026-01-28 03:38:22 +03:00
arabianq ac61a71dcb feat: bump version to 1.5.0 2026-01-28 03:35:04 +03:00
arabianq 71c800c396 docs(assets): update screenshot image 2026-01-28 03:34:32 +03:00
arabianq 577a6d279b fix(daemon): remove unnecessary ExecStartPre sleep command 2026-01-28 03:32:29 +03:00
arabianq 49e01f0318 fix(gui): correct calculation of vertical separator's position 2026-01-28 03:31:54 +03:00
arabianq 5ea9b3b0ba feat(daemon): implementet get full-state command 2026-01-28 02:41:33 +03:00
arabianq ca85d4c369 refactor: remove redundant device linking in play method 2026-01-28 02:28:23 +03:00
arabianq 4499b1d3aa feat(gui): now directories can be reordered using drag and drop 2026-01-28 02:10:36 +03:00
arabianq d385e5356e refactor: simplify device retrieval in link_player_to_virtual_mic function 2026-01-28 01:30:03 +03:00
arabianq b4a0dc6a83 feat: now pwsp will automatically detect when input device is connected/disconnected and properly link/unlink it 2026-01-28 01:26:43 +03:00
arabianq 2e570b3bb0 fix: navigating through files using keyboard now works correctly with filtered files 2026-01-28 00:45:52 +03:00
arabianq ee4554286e refactor: improved filtering functionality 2026-01-28 00:45:20 +03:00
arabianq 2c6f0d932e refactor: refactor input handling for Enter key and directory navigation 2026-01-28 00:34:44 +03:00
arabianq 4e7606fdc6 feat: remove escape key functionality from input handling 2026-01-28 00:28:34 +03:00
arabianq 03df631690 refactor: enhance search field focus functionality and input handling 2026-01-28 00:28:08 +03:00
arabianq 6df826f210 feat: you can now collapse every audio track 2026-01-28 00:03:56 +03:00
arabianq cdf306cfe9 feat: make vertical separator in GUI adjustable 2026-01-27 23:51:14 +03:00
arabianq 74a436b171 fix: add serde default attribute to DaemonConfig and GuiConfig structs 2026-01-27 23:50:50 +03:00
arabianq 2f33c48bcc change version to 1.4.0 2026-01-25 00:34:21 +03:00
arabianq 1f48dad71c cargo update 2026-01-25 00:33:49 +03:00
Tarasov Aleksandr 8066f6d96b Merge pull request #8 from arabianq/feature/parallel-sounds
[Feature] Parallel sounds
2026-01-25 00:25:21 +03:00
arabianq 5c3bc30a6e readme: update screenshot 2026-01-25 00:23:42 +03:00
arabianq 90ed05d88d readme: add information about hotkeys and mouse controls 2026-01-25 00:21:53 +03:00
arabianq f59050ef04 feat: shift + enter is now equal to shift + left mouse 2026-01-25 00:20:55 +03:00
arabianq bae10edc99 feat: replace synchronous requests with asynchronous counterparts for improved performance 2026-01-25 00:14:14 +03:00
arabianq cfa2681ba3 feat: enhance file playback controls with shift and ctrl modifiers 2026-01-25 00:00:11 +03:00
arabianq 03e936ac34 feat: sort tracks by their id in AudioPlayer on get_tracks 2026-01-24 23:59:23 +03:00
arabianq f7f96abcbb feat: add functionality to stop all audio tracks with backspace key 2026-01-24 23:46:51 +03:00
arabianq 4d54443593 feat: add master volume slider 2026-01-24 23:45:33 +03:00
arabianq 5afe3dd45b refactor: remove unused requests and variables from start_app_state_thread 2026-01-24 22:36:26 +03:00
arabianq bd75ac6190 refactor: remove unused fields from AudioPlayerState 2026-01-24 22:34:25 +03:00
arabianq 3e6a8b6e79 feat: first attemp to support playing multiple tracks in parallel 2026-01-24 22:18:42 +03:00
arabianq c1c8deb1b3 README: small changes 2026-01-24 20:46:59 +03:00
arabianq dc61342af8 change version to 1.3.2 2026-01-16 16:19:17 +03:00
arabianq fded7f7b3f deps: cargo update 2026-01-16 16:18:44 +03:00
arabianq ec83680333 deps: bump rfd to 0.17.2 2026-01-16 16:18:11 +03:00
arabianq 16e94e71d3 fix: update input device properties to output for capture_MONO 2026-01-16 16:16:27 +03:00
arabianq 56040934e9 cargo update 2026-01-08 02:56:37 +03:00
arabianq 89975cb124 change version to 1.3.1 2026-01-08 02:53:50 +03:00
arabianq 03e20ffa7e deps: bump rfd to 0.17.1 2026-01-08 02:53:16 +03:00
arabianq 8f7ea09ef5 deps: bump clap to 4.5.54 2026-01-08 02:49:54 +03:00
arabianq c639721f2a deps: bump serde_json to 1.0.149 2026-01-08 02:49:11 +03:00
arabianq 84512efda8 deps: bump tokio to 1.49.0 2026-01-08 02:48:23 +03:00
arabianq 6cf4a9744d replace println! with eprintln! for errors 2026-01-08 02:46:44 +03:00
arabianq d9ced4e650 fix: replace .except() with correct error handling 2026-01-08 02:45:07 +03:00
21 changed files with 1320 additions and 1321 deletions
Generated
+149 -623
View File
File diff suppressed because it is too large Load Diff
+7 -5
View File
@@ -1,6 +1,6 @@
[package] [package]
name = "pwsp" name = "pwsp"
version = "1.3.0" version = "1.5.0"
edition = "2024" edition = "2024"
authors = ["arabian"] authors = ["arabian"]
description = "PWSP lets you play audio files through your microphone. Has both CLI and GUI clients." description = "PWSP lets you play audio files through your microphone. Has both CLI and GUI clients."
@@ -12,22 +12,24 @@ keywords = ["soundpad", "pipewire", "linux", "cli", "gui"]
[dependencies] [dependencies]
tokio = { version = "1.48.0", features = ["full"] } tokio = { version = "1.49.0", features = ["full"] }
async-trait = "0.1.89" async-trait = "0.1.89"
serde = { version = "1.0.228", features = ["derive"] } serde = { version = "1.0.228", features = ["derive"] }
serde_json = "1.0.148" serde_json = "1.0.149"
clap = { version = "4.5.53", default-features = false, features = ["std", "suggestions", "help", "usage", "error-context", "derive"] } clap = { version = "4.5.55", default-features = false, features = ["std", "suggestions", "help", "usage", "error-context", "derive"] }
dirs = "6.0.0" dirs = "6.0.0"
itertools = "0.14.0"
rodio = { version = "0.21.1", default-features = false, features = ["symphonia-all", "playback"] } rodio = { version = "0.21.1", default-features = false, features = ["symphonia-all", "playback"] }
pipewire = "0.9.2" pipewire = "0.9.2"
rfd = "0.16.0" rfd = { version = "0.17.2", default-features = false, features = ["xdg-portal"]}
egui = { version = "0.33.3", default-features = false, features = ["default_fonts", "rayon"] } egui = { version = "0.33.3", default-features = false, features = ["default_fonts", "rayon"] }
eframe = { version = "0.33.3", default-features = false, features = ["default_fonts", "glow", "x11", "wayland"] } eframe = { version = "0.33.3", default-features = false, features = ["default_fonts", "glow", "x11", "wayland"] }
egui_material_icons = "0.5.0" egui_material_icons = "0.5.0"
egui_dnd = "0.14.0"
[[bin]] [[bin]]
name = "pwsp-daemon" name = "pwsp-daemon"
+32 -7
View File
@@ -24,6 +24,9 @@ chats on platforms like **Discord, Zoom, or Teamspeak**.
* **Position slider** to fast-forward or rewind the audio. * **Position slider** to fast-forward or rewind the audio.
* **Persistent Configuration**: The list of added directories and your selected audio output device are saved * **Persistent Configuration**: The list of added directories and your selected audio output device are saved
automatically, so you won't need to reconfigure them every time you launch the application. automatically, so you won't need to reconfigure them every time you launch the application.
* **Collapsible Audio Tracks**: You can collapse every audio track to save space.
* **Drag and Drop Directories**: Reorder your sound directories easily using drag and drop.
* **Automatic Device Detection**: PWSP automatically detects when an input device is connected or disconnected and handles linking/unlinking.
# **⚙️ How It Works** # **⚙️ How It Works**
@@ -35,8 +38,8 @@ three main components:
* Creating and managing virtual audio devices. * Creating and managing virtual audio devices.
* Linking these devices within the PipeWire graph. * Linking these devices within the PipeWire graph.
* Handling all audio playback. * Handling all audio playback.
* **pwsp-gui**: This is the graphical user interface. It acts as a client that communicates with pwsp-daemon via a * * **pwsp-gui**: This is the graphical user interface. It acts as a client that communicates with pwsp-daemon via a
*UnixSocket**. This is how you interact with your sound collection, control playback, and configure settings. **UnixSocket**. This is how you interact with your sound collection, control playback, and configure settings.
* **pwsp-cli**: This is the command-line interface, also acting as a client. It provides a way to control the daemon * **pwsp-cli**: This is the command-line interface, also acting as a client. It provides a way to control the daemon
without a GUI, allowing for scripting or quick command-based actions. without a GUI, allowing for scripting or quick command-based actions.
@@ -44,10 +47,10 @@ three main components:
## **Pre-built Packages** ## **Pre-built Packages**
You can download pre-built binaries, .deb and .rpm packages from You can download pre-built binaries and .deb packages from
the [releases page](https://github.com/arabianq/pipewire-soundpad/releases). the [releases page](https://github.com/arabianq/pipewire-soundpad/releases).
## **Fedora Linux** ## **Fedora Linux (and derivatives)**
If you're using Fedora, you can install PWSP from a dedicated repository using DNF. If you're using Fedora, you can install PWSP from a dedicated repository using DNF.
@@ -137,12 +140,12 @@ You can start the daemon from the terminal or enable the systemd service for aut
### **Using the GUI** ### **Using the GUI**
1. **Add Sounds**: Click the **"Add Directory"** button and select a folder containing your audio files. The application 1. **Add Sounds**: Click the **"+"** button and select a folder containing your audio files. The application
will automatically list all supported files. will automatically list all supported files.
2. **Select Microphone**: In the main application window, select your **physical microphone**. PWSP will automatically 2. **Select Microphone**: In the main application window, select your microphone. PWSP will automatically
create a virtual microphone and feed it sound from two sources: **your microphone** and the **audio files**. create a virtual microphone and feed it sound from two sources: **your microphone** and the **audio files**.
3. **Playback**: Click on a file in the list to load it, then use the **"Play"** and **"Pause"** buttons to control 3. **Playback**: Click on a file in the list to load it, then use the **"Play"** and **"Pause"** buttons to control
playback. playback. You can also play single file once using **"Play File"** button.
### **Using the CLI** ### **Using the CLI**
@@ -173,6 +176,28 @@ pwsp-cli --help
pwsp-cli set position 20 pwsp-cli set position 20
``` ```
### **Hotkeys & Controls**
#### **Keyboard Shortcuts**
| Key | Action |
| :----------------------- | :--------------------------------------------------- |
| **Space** | Pause / Resume audio |
| **Backspace** | Stop all audio tracks |
| **Enter** | Play selected file (stops all other tracks) |
| **Ctrl + Enter** | Add selected file to playback (plays simultaneously) |
| **Shift + Enter** | Replace the last added track with the selected one |
| **I** | Open / Close settings |
| **/** | Focus search field |
| **Ctrl + ↑ / ↓** | Navigate through files |
| **Ctrl + Shift + ↑ / ↓** | Navigate through directories |
#### **Mouse Controls**
* **Left Click**: Play track (stops all other tracks).
* **Ctrl + Left Click**: Add track (plays simultaneously with current tracks).
* **Shift + Left Click**: Replace the last added track with the selected one.
# **🤝 Contributing** # **🤝 Contributing**
Contributions are welcome\! If you have ideas for improvements or find a bug, feel free to create Contributions are welcome\! If you have ideas for improvements or find a bug, feel free to create
-1
View File
@@ -3,7 +3,6 @@ Description=Pipewire Soundpad Daemon
After=pipewire.service After=pipewire.service
[Service] [Service]
ExecStartPre=/usr/bin/sleep 10
ExecStart=/usr/bin/pwsp-daemon ExecStart=/usr/bin/pwsp-daemon
Restart=no Restart=no
RuntimeDirectory=pwsp RuntimeDirectory=pwsp
Binary file not shown.

Before

Width:  |  Height:  |  Size: 200 KiB

After

Width:  |  Height:  |  Size: 84 KiB

+1 -1
View File
@@ -4,7 +4,7 @@
%global cargo_install_lib 0 %global cargo_install_lib 0
Name: pwsp Name: pwsp
Version: 1.3.0 Version: 1.5.0
Release: %autorelease Release: %autorelease
Summary: Lets you play audio files through your microphone Summary: Lets you play audio files through your microphone
+68 -28
View File
@@ -36,17 +36,36 @@ enum Actions {
/// Ping the daemon /// Ping the daemon
Ping, Ping,
/// Pause audio playback /// Pause audio playback
Pause, Pause {
#[clap(short, long)]
id: Option<u32>,
},
/// Resume audio playback /// Resume audio playback
Resume, Resume {
#[clap(short, long)]
id: Option<u32>,
},
/// Toggle pause /// Toggle pause
TogglePause, TogglePause {
#[clap(short, long)]
id: Option<u32>,
},
/// Stop audio playback and clear the queue /// Stop audio playback and clear the queue
Stop, Stop {
#[clap(short, long)]
id: Option<u32>,
},
/// Play a file /// Play a file
Play { file_path: PathBuf }, Play {
file_path: PathBuf,
#[clap(short, long)]
concurrent: bool,
},
/// Toggle loop /// Toggle loop
ToggleLoop, ToggleLoop {
#[clap(short, long)]
id: Option<u32>,
},
} }
#[derive(Subcommand, Debug)] #[derive(Subcommand, Debug)]
@@ -56,31 +75,49 @@ enum GetCommands {
/// Playback volume /// Playback volume
Volume, Volume,
/// Playback position (in seconds) /// Playback position (in seconds)
Position, Position {
#[clap(short, long)]
id: Option<u32>,
},
/// Duration of the current file /// Duration of the current file
Duration, Duration {
#[clap(short, long)]
id: Option<u32>,
},
/// Player state (Playing, Paused or Stopped) /// Player state (Playing, Paused or Stopped)
State, State,
/// Current playing file path /// Get all playing tracks
CurrentFilePath, Tracks,
/// Current audio input /// Current audio input
Input, Input,
/// All audio inputs /// All audio inputs
Inputs, Inputs,
/// Is loop enabled (true or false) /// Full player state
Loop, FullState,
} }
#[derive(Subcommand, Debug)] #[derive(Subcommand, Debug)]
enum SetCommands { enum SetCommands {
/// Playback volume /// Playback volume
Volume { volume: f32 }, Volume {
volume: f32,
#[clap(short, long)]
id: Option<u32>,
},
/// Playback position (in seconds) /// Playback position (in seconds)
Position { position: f32 }, Position {
position: f32,
#[clap(short, long)]
id: Option<u32>,
},
/// Audio input id (see pwsp-cli get inputs) /// Audio input id (see pwsp-cli get inputs)
Input { name: String }, Input { name: String },
/// Enable or disable loop (true or false) /// Enable or disable loop (true or false)
Loop { enabled: String }, Loop {
enabled: String,
#[clap(short, long)]
id: Option<u32>,
},
} }
#[tokio::main] #[tokio::main]
@@ -92,29 +129,32 @@ async fn main() -> Result<(), Box<dyn Error>> {
let request = match cli.command { let request = match cli.command {
Commands::Action { action } => match action { Commands::Action { action } => match action {
Actions::Ping => Request::ping(), Actions::Ping => Request::ping(),
Actions::Pause => Request::pause(), Actions::Pause { id } => Request::pause(id),
Actions::Resume => Request::resume(), Actions::Resume { id } => Request::resume(id),
Actions::TogglePause => Request::toggle_pause(), Actions::TogglePause { id } => Request::toggle_pause(id),
Actions::Stop => Request::stop(), Actions::Stop { id } => Request::stop(id),
Actions::Play { file_path } => Request::play(file_path.to_str().unwrap()), Actions::Play {
Actions::ToggleLoop => Request::toggle_loop(), file_path,
concurrent,
} => Request::play(file_path.to_str().unwrap(), concurrent),
Actions::ToggleLoop { id } => Request::toggle_loop(id),
}, },
Commands::Get { parameter } => match parameter { Commands::Get { parameter } => match parameter {
GetCommands::IsPaused => Request::get_is_paused(), GetCommands::IsPaused => Request::get_is_paused(),
GetCommands::Volume => Request::get_volume(), GetCommands::Volume => Request::get_volume(),
GetCommands::Position => Request::get_position(), GetCommands::Position { id } => Request::get_position(id),
GetCommands::Duration => Request::get_duration(), GetCommands::Duration { id } => Request::get_duration(id),
GetCommands::State => Request::get_state(), GetCommands::State => Request::get_state(),
GetCommands::CurrentFilePath => Request::get_current_file_path(), GetCommands::Tracks => Request::get_tracks(),
GetCommands::Input => Request::get_input(), GetCommands::Input => Request::get_input(),
GetCommands::Inputs => Request::get_inputs(), GetCommands::Inputs => Request::get_inputs(),
GetCommands::Loop => Request::get_loop(), GetCommands::FullState => Request::get_full_state(),
}, },
Commands::Set { parameter } => match parameter { Commands::Set { parameter } => match parameter {
SetCommands::Volume { volume } => Request::set_volume(volume), SetCommands::Volume { volume, id } => Request::set_volume(volume, id),
SetCommands::Position { position } => Request::seek(position), SetCommands::Position { position, id } => Request::seek(position, id),
SetCommands::Input { name } => Request::set_input(&name), SetCommands::Input { name } => Request::set_input(&name),
SetCommands::Loop { enabled } => Request::set_loop(&enabled), SetCommands::Loop { enabled, id } => Request::set_loop(&enabled, id),
}, },
}; };
+2 -16
View File
@@ -1,8 +1,5 @@
use pwsp::{ use pwsp::{
types::{ types::socket::{Request, Response},
audio_player::PlayerState,
socket::{Request, Response},
},
utils::{ utils::{
commands::parse_command, commands::parse_command,
daemon::{ daemon::{
@@ -122,18 +119,7 @@ async fn player_loop() {
loop { loop {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
// Start playback again if loop is enabled audio_player.update().await;
let should_play = audio_player.get_state() == PlayerState::Stopped
&& audio_player.current_file_path.is_some()
&& audio_player.looped;
if should_play {
let file_path = audio_player.current_file_path.clone().unwrap();
audio_player
.play(&file_path)
.await
.expect("Something went wrong while trying to play the file");
}
sleep(Duration::from_millis(100)).await; sleep(Duration::from_millis(100)).await;
} }
+223 -93
View File
@@ -1,14 +1,36 @@
use crate::gui::{SUPPORTED_EXTENSIONS, SoundpadGui}; use crate::gui::SoundpadGui;
use egui::{ use egui::{
Align, AtomExt, Button, Color32, ComboBox, FontFamily, Label, Layout, RichText, ScrollArea, Align, AtomExt, Button, CollapsingHeader, Color32, ComboBox, CursorIcon, FontFamily, Label,
Slider, TextEdit, Ui, Vec2, Layout, RichText, ScrollArea, Sense, Slider, TextEdit, Ui, Vec2,
}; };
use egui_dnd::dnd;
use egui_material_icons::icons; use egui_material_icons::icons;
use pwsp::types::audio_player::PlayerState; use pwsp::types::audio_player::TrackInfo;
use pwsp::utils::gui::format_time_pair; use pwsp::utils::gui::format_time_pair;
use std::{error::Error, path::PathBuf}; use std::{error::Error, time::Instant};
use pwsp::types::gui::AppState;
enum TrackAction {
Pause(u32),
Resume(u32),
ToggleLoop(u32),
Stop(u32),
}
impl SoundpadGui { impl SoundpadGui {
fn get_volume_icon(volume: f32) -> &'static str {
if volume > 0.7 {
icons::ICON_VOLUME_UP
} else if volume <= 0.0 {
icons::ICON_VOLUME_OFF
} else if volume < 0.3 {
icons::ICON_VOLUME_MUTE
} else {
icons::ICON_VOLUME_DOWN
}
}
pub fn draw_waiting_for_daemon(&mut self, ui: &mut Ui) { pub fn draw_waiting_for_daemon(&mut self, ui: &mut Ui) {
ui.centered_and_justified(|ui| { ui.centered_and_justified(|ui| {
ui.label( ui.label(
@@ -74,45 +96,101 @@ impl SoundpadGui {
fn draw_header(&mut self, ui: &mut Ui) { fn draw_header(&mut self, ui: &mut Ui) {
ui.vertical_centered_justified(|ui| { ui.vertical_centered_justified(|ui| {
// Current file name if self.audio_player_state.tracks.is_empty() {
ui.label( ui.label("No tracks playing");
RichText::new( return;
self.audio_player_state }
.current_file_path
.file_stem() let tracks = self.audio_player_state.tracks.clone();
.unwrap_or_default() let mut action = None;
.to_str()
.unwrap_or_default(), for track in tracks {
CollapsingHeader::new(
RichText::new(
track
.path
.file_stem()
.unwrap_or_default()
.to_str()
.unwrap_or_default(),
)
.color(Color32::WHITE)
.family(FontFamily::Monospace),
) )
.color(Color32::WHITE) .default_open(true)
.family(FontFamily::Monospace), .show(ui, |ui| {
); if let Some(act) = Self::draw_track_control(ui, &mut self.app_state, &track) {
// Media controls action = Some(act);
self.draw_controls(ui); }
ui.separator(); });
ui.separator();
}
if let Some(action) = action {
match action {
TrackAction::Pause(id) => self.pause(Some(id)),
TrackAction::Resume(id) => self.resume(Some(id)),
TrackAction::ToggleLoop(id) => self.toggle_loop(Some(id)),
TrackAction::Stop(id) => self.stop(Some(id)),
}
}
}); });
} }
fn draw_controls(&mut self, ui: &mut Ui) { fn draw_track_control(
ui: &mut Ui,
app_state: &mut AppState,
track: &TrackInfo,
) -> Option<TrackAction> {
let ui_state = app_state.track_ui_states.entry(track.id).or_default();
let should_update_position = !ui_state.position_dragged
&& ui_state
.ignore_position_update_until
.map(|t| Instant::now() > t)
.unwrap_or(true);
if should_update_position {
ui_state.position_slider_value = track.position;
}
let should_update_volume = !ui_state.volume_dragged
&& ui_state
.ignore_volume_update_until
.map(|t| Instant::now() > t)
.unwrap_or(true);
if should_update_volume {
ui_state.volume_slider_value = track.volume;
}
let mut action = None;
ui.horizontal_top(|ui| { ui.horizontal_top(|ui| {
// ---------- Play Button ---------- // ---------- Play Button ----------
let play_button = Button::new(match self.audio_player_state.state { let play_button = Button::new(if track.paused {
PlayerState::Playing => icons::ICON_PAUSE, icons::ICON_PLAY_ARROW
PlayerState::Paused | PlayerState::Stopped => icons::ICON_PLAY_ARROW, } else {
icons::ICON_PAUSE
}) })
.corner_radius(15.0); .corner_radius(15.0);
let play_button_response = ui.add_sized([30.0, 30.0], play_button); let play_button_response = ui.add_sized([30.0, 30.0], play_button);
if play_button_response.clicked() { if play_button_response.clicked() {
self.play_toggle(); if track.paused {
action = Some(TrackAction::Resume(track.id));
} else {
action = Some(TrackAction::Pause(track.id));
}
} }
// -------------------------------- // --------------------------------
// ---------- Loop Button ---------- // ---------- Loop Button ----------
let loop_button = Button::new( let loop_button = Button::new(
RichText::new(match self.audio_player_state.looped { RichText::new(if track.looped {
true => icons::ICON_REPEAT_ONE, icons::ICON_REPEAT_ONE
false => icons::ICON_REPEAT, } else {
icons::ICON_REPEAT
}) })
.size(18.0), .size(18.0),
) )
@@ -120,17 +198,15 @@ impl SoundpadGui {
let loop_button_response = ui.add_sized([15.0, 30.0], loop_button); let loop_button_response = ui.add_sized([15.0, 30.0], loop_button);
if loop_button_response.clicked() { if loop_button_response.clicked() {
self.toggle_loop(); action = Some(TrackAction::ToggleLoop(track.id));
} }
// -------------------------------- // --------------------------------
// ---------- Position Slider ---------- // ---------- Position Slider ----------
let position_slider = Slider::new( let duration = track.duration.unwrap_or(1.0);
&mut self.app_state.position_slider_value, let position_slider = Slider::new(&mut ui_state.position_slider_value, 0.0..=duration)
0.0..=self.audio_player_state.duration, .show_value(false)
) .step_by(0.01);
.show_value(false)
.step_by(0.01);
let default_slider_width = ui.spacing().slider_width; let default_slider_width = ui.spacing().slider_width;
let position_slider_width = ui.available_width() let position_slider_width = ui.available_width()
@@ -140,57 +216,81 @@ impl SoundpadGui {
ui.spacing_mut().slider_width = position_slider_width; ui.spacing_mut().slider_width = position_slider_width;
let position_slider_response = ui.add_sized([30.0, 30.0], position_slider); let position_slider_response = ui.add_sized([30.0, 30.0], position_slider);
if position_slider_response.drag_stopped() { if position_slider_response.drag_stopped() {
self.app_state.position_dragged = true; ui_state.position_dragged = true;
} }
// -------------------------------- // --------------------------------
// ---------- Time Label ---------- // ---------- Time Label ----------
let time_label = Label::new( let time_label =
RichText::new(format_time_pair( Label::new(RichText::new(format_time_pair(track.position, duration)).monospace());
self.audio_player_state.position,
self.audio_player_state.duration,
))
.monospace(),
);
ui.add_sized([30.0, 30.0], time_label); ui.add_sized([30.0, 30.0], time_label);
// -------------------------------- // --------------------------------
// ---------- Volume Icon ---------- // ---------- Volume Icon ----------
let volume_icon = if self.audio_player_state.volume > 0.7 { let volume_icon = Self::get_volume_icon(track.volume);
icons::ICON_VOLUME_UP let volume_label = Label::new(RichText::new(volume_icon).size(18.0));
} else if self.audio_player_state.volume == 0.0 { ui.add_sized([30.0, 30.0], volume_label)
icons::ICON_VOLUME_OFF .on_hover_text(format!("Volume: {:.0}%", track.volume * 100.0));
} else if self.audio_player_state.volume < 0.3 {
icons::ICON_VOLUME_MUTE
} else {
icons::ICON_VOLUME_DOWN
};
let volume_icon = Label::new(RichText::new(volume_icon).size(18.0));
ui.add_sized([30.0, 30.0], volume_icon);
// -------------------------------- // --------------------------------
// ---------- Volume Slider ---------- // ---------- Volume Slider ----------
let volume_slider = Slider::new(&mut self.app_state.volume_slider_value, 0.0..=1.0) let volume_slider = Slider::new(&mut ui_state.volume_slider_value, 0.0..=1.0)
.show_value(false) .show_value(false)
.step_by(0.01); .step_by(0.01);
ui.spacing_mut().slider_width = default_slider_width; ui.spacing_mut().slider_width = default_slider_width - 30.0;
ui.spacing_mut().item_spacing.x = 0.0; ui.spacing_mut().item_spacing.x = 0.0;
let volume_slider_response = ui.add_sized([30.0, 30.0], volume_slider); let volume_slider_response = ui.add_sized([30.0, 30.0], volume_slider);
if volume_slider_response.drag_stopped() { if volume_slider_response.drag_stopped() {
self.app_state.volume_dragged = true; ui_state.volume_dragged = true;
}
// --------------------------------
// ---------- Stop Button ---------
let stop_button = Button::new(icons::ICON_CLOSE).frame(false);
let stop_button_response = ui.add_sized([30.0, 30.0], stop_button);
if stop_button_response.clicked() {
action = Some(TrackAction::Stop(track.id));
} }
// -------------------------------- // --------------------------------
}); });
action
} }
fn draw_body(&mut self, ui: &mut Ui) { fn draw_body(&mut self, ui: &mut Ui) {
let dirs_size = Vec2::new(ui.available_width() / 4.0, ui.available_height() - 40.0); let left_panel_width = self
.config
.left_panel_width
.max(100.0)
.min(ui.available_width() - 100.0);
let dirs_size = Vec2::new(left_panel_width, ui.available_height() - 40.0);
ui.horizontal(|ui| { ui.horizontal(|ui| {
self.draw_dirs(ui, dirs_size); self.draw_dirs(ui, dirs_size);
ui.separator();
let (rect, response) = ui.allocate_at_least(
Vec2::new(ui.spacing().item_spacing.x, ui.available_height()),
Sense::click_and_drag(),
);
if ui.is_rect_visible(rect) {
let stroke = ui.visuals().widgets.noninteractive.bg_stroke;
ui.painter().vline(rect.center().x, rect.y_range(), stroke);
}
let vertical_separator_response =
response.on_hover_and_drag_cursor(CursorIcon::ResizeHorizontal);
if vertical_separator_response.dragged() {
self.config.left_panel_width += vertical_separator_response.drag_delta().x;
self.config.left_panel_width = self.config.left_panel_width.clamp(100.0, 500.0);
}
if vertical_separator_response.drag_stopped() {
self.config.save_to_file().ok();
}
let files_size = Vec2::new(ui.available_width(), ui.available_height() - 40.0); let files_size = Vec2::new(ui.available_width(), ui.available_height() - 40.0);
self.draw_files(ui, files_size); self.draw_files(ui, files_size);
@@ -205,10 +305,14 @@ impl SoundpadGui {
ScrollArea::vertical().id_salt(0).show(ui, |ui| { ScrollArea::vertical().id_salt(0).show(ui, |ui| {
ui.set_min_width(area_size.x); ui.set_min_width(area_size.x);
let mut dirs: Vec<PathBuf> = self.app_state.dirs.iter().cloned().collect(); let mut dirs = self.app_state.dirs.clone();
dirs.sort();
for path in dirs.iter() { dnd(ui, "dnd_directories").show_vec(&mut dirs, |ui, item, handle, _state| {
let path = item.clone();
ui.horizontal(|ui| { ui.horizontal(|ui| {
handle.ui(ui, |ui| {
ui.label(icons::ICON_DRAG_INDICATOR);
});
let name = path let name = path
.file_name() .file_name()
.map(|s| s.to_string_lossy().to_string()) .map(|s| s.to_string_lossy().to_string())
@@ -216,7 +320,7 @@ impl SoundpadGui {
let mut dir_button_text = RichText::new(name.clone()); let mut dir_button_text = RichText::new(name.clone());
if let Some(current_dir) = &self.app_state.current_dir { if let Some(current_dir) = &self.app_state.current_dir {
if current_dir.eq(path) { if current_dir.eq(&path) {
dir_button_text = dir_button_text.color(Color32::WHITE); dir_button_text = dir_button_text.color(Color32::WHITE);
} }
} }
@@ -226,7 +330,7 @@ impl SoundpadGui {
let dir_button_response = ui.add(dir_button); let dir_button_response = ui.add(dir_button);
if dir_button_response.clicked() { if dir_button_response.clicked() {
self.open_dir(path); self.open_dir(&path);
} }
let delete_dir_button = Button::new(icons::ICON_DELETE).frame(false); let delete_dir_button = Button::new(icons::ICON_DELETE).frame(false);
@@ -236,7 +340,8 @@ impl SoundpadGui {
self.remove_dir(&path.clone()); self.remove_dir(&path.clone());
} }
}); });
} });
self.app_state.dirs = dirs;
ui.horizontal(|ui| { ui.horizontal(|ui| {
let add_dirs_button = Button::new(icons::ICON_ADD).frame(false); let add_dirs_button = Button::new(icons::ICON_ADD).frame(false);
@@ -260,12 +365,17 @@ impl SoundpadGui {
fn draw_files(&mut self, ui: &mut Ui, area_size: Vec2) { fn draw_files(&mut self, ui: &mut Ui, area_size: Vec2) {
ui.vertical(|ui| { ui.vertical(|ui| {
ui.horizontal(|ui| { ui.horizontal(|ui| {
let search_field = ui.add_sized( let search_field_response = ui.add_sized(
[ui.available_width(), 22.0], [ui.available_width(), 22.0],
TextEdit::singleline(&mut self.app_state.search_query).hint_text("Search..."), TextEdit::singleline(&mut self.app_state.search_query).hint_text("Search..."),
); );
self.app_state.search_field_id = Some(search_field.id); if self.app_state.force_focus_search {
search_field_response.request_focus();
self.app_state.force_focus_search = false;
}
self.app_state.search_field_id = Some(search_field_response.id);
}); });
ui.separator(); ui.separator();
@@ -275,37 +385,15 @@ impl SoundpadGui {
ui.set_min_height(area_size.y); ui.set_min_height(area_size.y);
ui.vertical(|ui| { ui.vertical(|ui| {
let mut files: Vec<PathBuf> = self.app_state.files.iter().cloned().collect(); let files = self.get_filtered_files();
files.sort();
for entry_path in files { for entry_path in files {
if entry_path.is_dir() {
continue;
}
if !SUPPORTED_EXTENSIONS
.contains(&entry_path.extension().unwrap_or_default().to_str().unwrap())
{
continue;
}
let file_name = entry_path let file_name = entry_path
.file_name() .file_name()
.unwrap() .unwrap()
.to_string_lossy() .to_string_lossy()
.to_string(); .to_string();
let search_query = self
.app_state
.search_query
.to_lowercase()
.trim()
.to_string();
if !file_name.to_lowercase().contains(search_query.as_str()) {
continue;
}
let mut file_button_text = RichText::new(file_name); let mut file_button_text = RichText::new(file_name);
if let Some(current_file) = &self.app_state.selected_file { if let Some(current_file) = &self.app_state.selected_file {
if current_file.eq(&entry_path) { if current_file.eq(&entry_path) {
@@ -316,7 +404,18 @@ impl SoundpadGui {
let file_button = Button::new(file_button_text).frame(false); let file_button = Button::new(file_button_text).frame(false);
let file_button_response = ui.add(file_button); let file_button_response = ui.add(file_button);
if file_button_response.clicked() { if file_button_response.clicked() {
self.play_file(&entry_path); ui.input(|i| {
if i.modifiers.ctrl {
self.play_file(&entry_path, true);
} else if i.modifiers.shift
&& let Some(last_track) = self.audio_player_state.tracks.last()
{
self.stop(Some(last_track.id));
self.play_file(&entry_path, true);
} else {
self.play_file(&entry_path, false);
}
});
self.app_state.selected_file = Some(entry_path); self.app_state.selected_file = Some(entry_path);
} }
} }
@@ -327,7 +426,7 @@ impl SoundpadGui {
fn draw_footer(&mut self, ui: &mut Ui) { fn draw_footer(&mut self, ui: &mut Ui) {
ui.add_space(5.0); ui.add_space(5.0);
ui.horizontal_top(|ui| { ui.horizontal(|ui| {
// ---------- Microphone selection ---------- // ---------- Microphone selection ----------
let mut mics: Vec<(&String, &String)> = let mut mics: Vec<(&String, &String)> =
self.audio_player_state.all_inputs.iter().collect(); self.audio_player_state.all_inputs.iter().collect();
@@ -336,6 +435,7 @@ impl SoundpadGui {
let mut selected_input = self.audio_player_state.current_input.to_owned(); let mut selected_input = self.audio_player_state.current_input.to_owned();
let prev_input = selected_input.to_owned(); let prev_input = selected_input.to_owned();
ComboBox::from_label("Choose microphone") ComboBox::from_label("Choose microphone")
.height(30.0)
.selected_text( .selected_text(
self.audio_player_state self.audio_player_state
.all_inputs .all_inputs
@@ -353,10 +453,40 @@ impl SoundpadGui {
} }
// -------------------------------- // --------------------------------
// ---------- Master Volume Slider ----------
let volume_icon = Self::get_volume_icon(self.audio_player_state.volume);
let volume_label = Label::new(RichText::new(volume_icon).size(18.0));
ui.add_sized([18.0, 18.0], volume_label)
.on_hover_text(format!(
"Master Volume: {:.0}%",
self.audio_player_state.volume * 100.0
));
let should_update_volume = !self.app_state.volume_dragged
&& self
.app_state
.ignore_volume_update_until
.map(|t| Instant::now() > t)
.unwrap_or(true);
if should_update_volume {
self.app_state.volume_slider_value = self.audio_player_state.volume;
}
let volume_slider = Slider::new(&mut self.app_state.volume_slider_value, 0.0..=1.0)
.show_value(false)
.step_by(0.01);
let volume_slider_response = ui.add_sized([150.0, 18.0], volume_slider);
if volume_slider_response.drag_stopped() {
self.app_state.volume_dragged = true;
}
// ------------------------------------------
ui.add_space(ui.available_width() - 18.0 - ui.spacing().item_spacing.x); ui.add_space(ui.available_width() - 18.0 - ui.spacing().item_spacing.x);
// ---------- Settings button ---------- // ---------- Settings button ----------
let settings_button = Button::new(icons::ICON_SETTINGS).frame(false); let settings_button =
Button::new(icons::ICON_SETTINGS.atom_size(Vec2::new(18.0, 18.0))).frame(false);
let settings_button_response = ui.add_sized([18.0, 18.0], settings_button); let settings_button_response = ui.add_sized([18.0, 18.0], settings_button);
if settings_button_response.clicked() { if settings_button_response.clicked() {
self.app_state.show_settings = true; self.app_state.show_settings = true;
+92 -84
View File
@@ -1,107 +1,115 @@
use crate::gui::SoundpadGui; use crate::gui::SoundpadGui;
use egui::{Context, Key}; use egui::{Context, Key, Modifiers};
use std::path::PathBuf; use std::path::PathBuf;
impl SoundpadGui { impl SoundpadGui {
fn key_pressed(&self, ctx: &Context, key: Key) -> bool {
ctx.input(|i| i.key_pressed(key))
}
fn modifiers(&self, ctx: &Context) -> Modifiers {
ctx.input(|i| i.modifiers)
}
pub fn handle_input(&mut self, ctx: &Context) { pub fn handle_input(&mut self, ctx: &Context) {
if ctx.memory(|reader| { reader.focused() }.is_some()) { let modifiers = self.modifiers(ctx);
return;
// Open/close settings
if self.key_pressed(ctx, Key::I) {
self.app_state.show_settings = !self.app_state.show_settings;
} }
ctx.input(|i| { if !self.app_state.show_settings {
// Close app on espace // Pause / resume audio on space
if i.key_pressed(Key::Escape) { if self.key_pressed(ctx, Key::Space) {
std::process::exit(0); self.play_toggle();
} }
// Open/close settings // Stop all audio tracks on backspace
if i.key_pressed(Key::I) { if self.key_pressed(ctx, Key::Backspace) {
self.app_state.show_settings = !self.app_state.show_settings; self.stop(None);
} }
if i.key_pressed(Key::Enter) && self.app_state.selected_file.is_some() { // Focus search field
self.play_file(&self.app_state.selected_file.clone().unwrap()); if self.key_pressed(ctx, Key::Slash) {
self.app_state.force_focus_search = true;
} }
if !self.app_state.show_settings { // Play selected file on Enter
// Pause / resume audio on space if self.key_pressed(ctx, Key::Enter) && self.app_state.selected_file.is_some() {
if i.key_pressed(Key::Space) { let path = &self.app_state.selected_file.clone().unwrap();
self.play_toggle(); if modifiers.ctrl {
self.play_file(path, true);
} else if modifiers.shift
&& let Some(last_track) = self.audio_player_state.tracks.last()
{
self.stop(Some(last_track.id));
self.play_file(path, true);
} else {
self.play_file(path, false);
} }
}
// Focus search field // Iterate through dirs and files with Ctrl + Up/Down
if i.key_pressed(Key::Slash) && self.app_state.search_field_id.is_some() { let arrow_up_pressed = self.key_pressed(ctx, Key::ArrowUp);
self.app_state.force_focus_id = self.app_state.search_field_id; let arrow_down_pressed = self.key_pressed(ctx, Key::ArrowDown);
} if modifiers.ctrl && (arrow_up_pressed || arrow_down_pressed) {
if modifiers.shift && !self.app_state.dirs.is_empty() {
let mut dirs: Vec<PathBuf> = self.app_state.dirs.iter().cloned().collect();
dirs.sort();
// Iterate through dirs if there are some let current_dir_index: i8;
if i.modifiers.ctrl { if let Some(current_dir) = &self.app_state.current_dir {
let arrow_up_pressed = i.key_pressed(Key::ArrowUp); if let Some(index) = dirs.iter().position(|x| x == current_dir) {
let arrow_down_pressed = i.key_pressed(Key::ArrowDown); current_dir_index = index as i8;
} else {
if arrow_up_pressed || arrow_down_pressed { current_dir_index = -1;
if i.modifiers.shift && !self.app_state.dirs.is_empty() {
let mut dirs: Vec<PathBuf> =
self.app_state.dirs.iter().cloned().collect();
dirs.sort();
let current_dir_index: i8;
if let Some(current_dir) = &self.app_state.current_dir {
if let Some(index) = dirs.iter().position(|x| x == current_dir) {
current_dir_index = index as i8;
} else {
current_dir_index = -1;
}
} else {
current_dir_index = -1;
}
let mut new_dir_index: i8;
new_dir_index = current_dir_index - arrow_up_pressed as i8
+ arrow_down_pressed as i8;
if new_dir_index < 0 {
new_dir_index = (dirs.len() - 1) as i8;
} else if new_dir_index >= dirs.len() as i8 {
new_dir_index = 0;
}
self.open_dir(&dirs[new_dir_index as usize]);
} else if self.app_state.current_dir.is_some() {
let mut files: Vec<PathBuf> =
self.app_state.files.iter().cloned().collect();
files.sort();
let current_files_index: i64;
if let Some(selected_file) = &self.app_state.selected_file {
if let Some(index) = files.iter().position(|x| x == selected_file) {
current_files_index = index as i64;
} else {
current_files_index = -1;
}
} else {
current_files_index = -1;
}
let mut new_files_index: i64;
new_files_index = current_files_index - arrow_up_pressed as i64
+ arrow_down_pressed as i64;
if new_files_index < 0 {
new_files_index = (files.len() - 1) as i64;
} else if new_files_index >= files.len() as i64 {
new_files_index = 0;
}
self.app_state.selected_file =
Some(files[new_files_index as usize].clone());
} }
} else {
current_dir_index = -1;
} }
let mut new_dir_index: i8;
new_dir_index =
current_dir_index - arrow_up_pressed as i8 + arrow_down_pressed as i8;
if new_dir_index < 0 {
new_dir_index = (dirs.len() - 1) as i8;
} else if new_dir_index >= dirs.len() as i8 {
new_dir_index = 0;
}
self.open_dir(&dirs[new_dir_index as usize]);
} else if self.app_state.current_dir.is_some() {
let files = self.get_filtered_files();
if files.is_empty() {
return;
}
let current_files_index = self
.app_state
.selected_file
.as_ref()
.and_then(|f| files.iter().position(|x| x == f))
.map(|i| i as i64)
.unwrap_or(-1);
let mut new_files_index =
current_files_index - arrow_up_pressed as i64 + arrow_down_pressed as i64;
if new_files_index < 0 {
new_files_index = (files.len() - 1) as i64;
} else if new_files_index >= files.len() as i64 {
new_files_index = 0;
}
self.app_state.selected_file = Some(files[new_files_index as usize].clone());
} }
} }
}); }
// });
} }
} }
+69 -14
View File
@@ -4,6 +4,7 @@ mod update;
use eframe::{HardwareAcceleration, NativeOptions, icon_data::from_png_bytes, run_native}; use eframe::{HardwareAcceleration, NativeOptions, icon_data::from_png_bytes, run_native};
use egui::{Context, Vec2, ViewportBuilder}; use egui::{Context, Vec2, ViewportBuilder};
use itertools::Itertools;
use pwsp::{ use pwsp::{
types::{ types::{
audio_player::PlayerState, audio_player::PlayerState,
@@ -13,13 +14,13 @@ use pwsp::{
}, },
utils::{ utils::{
daemon::get_daemon_config, daemon::get_daemon_config,
gui::{get_gui_config, make_request_sync, start_app_state_thread}, gui::{get_gui_config, make_request_async, make_request_sync, start_app_state_thread},
}, },
}; };
use rfd::FileDialog; use rfd::FileDialog;
use std::path::PathBuf;
use std::{ use std::{
error::Error, error::Error,
path::PathBuf,
sync::{Arc, Mutex}, sync::{Arc, Mutex},
}; };
@@ -59,14 +60,14 @@ impl SoundpadGui {
let (new_state, request) = { let (new_state, request) = {
let guard = self.audio_player_state_shared.lock().unwrap(); let guard = self.audio_player_state_shared.lock().unwrap();
match guard.state { match guard.state {
PlayerState::Playing => (Some(PlayerState::Paused), Some(Request::pause())), PlayerState::Playing => (Some(PlayerState::Paused), Some(Request::pause(None))),
PlayerState::Paused => (Some(PlayerState::Playing), Some(Request::resume())), PlayerState::Paused => (Some(PlayerState::Playing), Some(Request::resume(None))),
PlayerState::Stopped => (None, None), PlayerState::Stopped => (None, None),
} }
}; };
if let Some(req) = request { if let Some(req) = request {
make_request_sync(req).ok(); make_request_async(req);
} }
if let Some(state) = new_state { if let Some(state) = new_state {
@@ -79,7 +80,7 @@ impl SoundpadGui {
pub fn open_file(&mut self) { pub fn open_file(&mut self) {
let file_dialog = FileDialog::new().add_filter("Audio File", &SUPPORTED_EXTENSIONS); let file_dialog = FileDialog::new().add_filter("Audio File", &SUPPORTED_EXTENSIONS);
if let Some(path) = file_dialog.pick_file() { if let Some(path) = file_dialog.pick_file() {
self.play_file(&path); self.play_file(&path, false);
} }
} }
@@ -87,15 +88,16 @@ impl SoundpadGui {
let file_dialog = FileDialog::new(); let file_dialog = FileDialog::new();
if let Some(paths) = file_dialog.pick_folders() { if let Some(paths) = file_dialog.pick_folders() {
for path in paths { for path in paths {
self.app_state.dirs.insert(path); self.app_state.dirs.push(path);
} }
self.app_state.dirs = self.app_state.dirs.iter().unique().cloned().collect();
self.config.dirs = self.app_state.dirs.clone(); self.config.dirs = self.app_state.dirs.clone();
self.config.save_to_file().ok(); self.config.save_to_file().ok();
} }
} }
pub fn remove_dir(&mut self, path: &PathBuf) { pub fn remove_dir(&mut self, path: &PathBuf) {
self.app_state.dirs.remove(path); self.app_state.dirs.retain(|x| x != path);
if let Some(current_dir) = &self.app_state.current_dir if let Some(current_dir) = &self.app_state.current_dir
&& current_dir == path && current_dir == path
{ {
@@ -116,12 +118,12 @@ impl SoundpadGui {
.collect(); .collect();
} }
pub fn play_file(&mut self, path: &PathBuf) { pub fn play_file(&mut self, path: &PathBuf, concurrent: bool) {
make_request_sync(Request::play(path.to_str().unwrap())).ok(); make_request_async(Request::play(path.to_str().unwrap(), concurrent));
} }
pub fn set_input(&mut self, name: String) { pub fn set_input(&mut self, name: String) {
make_request_sync(Request::set_input(&name)).ok(); make_request_async(Request::set_input(&name));
if self.config.save_input { if self.config.save_input {
let mut daemon_config = get_daemon_config(); let mut daemon_config = get_daemon_config();
@@ -130,8 +132,61 @@ impl SoundpadGui {
} }
} }
pub fn toggle_loop(&mut self) { pub fn toggle_loop(&mut self, id: Option<u32>) {
make_request_sync(Request::toggle_loop()).ok(); make_request_async(Request::toggle_loop(id));
}
pub fn pause(&mut self, id: Option<u32>) {
make_request_async(Request::pause(id));
}
pub fn resume(&mut self, id: Option<u32>) {
make_request_async(Request::resume(id));
}
pub fn stop(&mut self, id: Option<u32>) {
make_request_async(Request::stop(id));
}
pub fn get_filtered_files(&self) -> Vec<PathBuf> {
let mut files: Vec<PathBuf> = self.app_state.files.iter().cloned().collect();
files.sort();
let search_query = self.app_state.search_query.to_lowercase();
let search_query = search_query.trim();
files
.into_iter()
.filter(|entry_path| {
if entry_path.is_dir() {
return false;
}
if !SUPPORTED_EXTENSIONS.contains(
&entry_path
.extension()
.unwrap_or_default()
.to_str()
.unwrap_or_default(),
) {
return false;
}
if !search_query.is_empty() {
let file_name = entry_path
.file_name()
.unwrap_or_default()
.to_string_lossy()
.to_string();
if !file_name.to_lowercase().contains(search_query) {
return false;
}
}
true
})
.collect()
} }
} }
@@ -163,7 +218,7 @@ pub async fn run() -> Result<(), Box<dyn Error>> {
Ok(_) => { Ok(_) => {
let config = get_gui_config(); let config = get_gui_config();
if config.pause_on_exit { if config.pause_on_exit {
make_request_sync(Request::pause()).ok(); make_request_sync(Request::pause(None)).ok();
} }
Ok(()) Ok(())
} }
+63 -42
View File
@@ -3,19 +3,74 @@ use eframe::{App, Frame as EFrame};
use egui::{CentralPanel, Context}; use egui::{CentralPanel, Context};
use pwsp::{ use pwsp::{
types::socket::Request, types::socket::Request,
utils::{ utils::{daemon::get_daemon_config, gui::make_request_async},
daemon::{get_daemon_config, is_daemon_running},
gui::make_request_sync,
},
}; };
use std::time::{Duration, Instant};
impl App for SoundpadGui { impl App for SoundpadGui {
fn update(&mut self, ctx: &Context, _frame: &mut EFrame) { fn update(&mut self, ctx: &Context, _frame: &mut EFrame) {
// Save directories if changed
if !self.config.dirs.eq(&self.app_state.dirs) {
self.config.dirs = self.app_state.dirs.clone();
self.config.save_to_file().ok();
}
// Seek and volume requests
let mut seek_requests = vec![];
let mut volume_requests = vec![];
for (id, ui_state) in &mut self.app_state.track_ui_states {
if ui_state.position_dragged {
seek_requests.push((*id, ui_state.position_slider_value));
}
if ui_state.volume_dragged {
volume_requests.push((*id, ui_state.volume_slider_value));
ui_state.volume_dragged = false;
}
}
for (id, pos) in seek_requests {
make_request_async(Request::seek(pos, Some(id)));
if let Some(ui_state) = self.app_state.track_ui_states.get_mut(&id) {
ui_state.position_dragged = false;
ui_state.ignore_position_update_until =
Some(Instant::now() + Duration::from_millis(300));
}
}
for (id, vol) in volume_requests {
make_request_async(Request::set_volume(vol, Some(id)));
if let Some(ui_state) = self.app_state.track_ui_states.get_mut(&id) {
ui_state.volume_dragged = false;
ui_state.ignore_volume_update_until =
Some(Instant::now() + Duration::from_millis(300));
}
}
if self.app_state.volume_dragged {
make_request_async(Request::set_volume(
self.app_state.volume_slider_value,
None,
));
self.app_state.volume_dragged = false;
self.app_state.ignore_volume_update_until =
Some(Instant::now() + Duration::from_millis(300));
if self.config.save_volume {
let mut daemon_config = get_daemon_config();
daemon_config.default_volume = Some(self.app_state.volume_slider_value);
daemon_config.save_to_file().ok();
}
}
// Sync audio player state
{ {
let guard = self.audio_player_state_shared.lock().unwrap(); let guard = self.audio_player_state_shared.lock().unwrap();
self.audio_player_state = guard.clone(); self.audio_player_state = guard.clone();
} }
// Handle scale factor changes
let old_scale_factor = self.config.scale_factor; let old_scale_factor = self.config.scale_factor;
let new_scale_factor = ctx.zoom_factor().clamp(0.5, 2.0); let new_scale_factor = ctx.zoom_factor().clamp(0.5, 2.0);
@@ -26,10 +81,12 @@ impl App for SoundpadGui {
self.config.save_to_file().ok(); self.config.save_to_file().ok();
} }
// Handle input
self.handle_input(ctx); self.handle_input(ctx);
// Draw UI
CentralPanel::default().show(ctx, |ui| { CentralPanel::default().show(ctx, |ui| {
if !is_daemon_running().unwrap() { if !self.audio_player_state.is_daemon_running {
self.draw_waiting_for_daemon(ui); self.draw_waiting_for_daemon(ui);
return; return;
} }
@@ -40,45 +97,9 @@ impl App for SoundpadGui {
} }
self.draw(ui).ok(); self.draw(ui).ok();
if let Some(force_focus_id) = self.app_state.force_focus_id {
ui.memory_mut(|reder| {
reder.request_focus(force_focus_id);
});
self.app_state.force_focus_id = None;
}
}); });
if self.app_state.position_dragged { // Request repaint
make_request_sync(Request::seek(self.app_state.position_slider_value)).ok();
let mut guard = self.audio_player_state_shared.lock().unwrap();
guard.new_position = Some(self.app_state.position_slider_value);
guard.position = self.app_state.position_slider_value;
self.app_state.position_dragged = false;
} else {
self.app_state.position_slider_value = self.audio_player_state.position;
}
if self.app_state.volume_dragged {
let new_volume = self.app_state.volume_slider_value;
make_request_sync(Request::set_volume(new_volume)).ok();
let mut guard = self.audio_player_state_shared.lock().unwrap();
guard.new_volume = Some(self.app_state.volume_slider_value);
guard.volume = self.app_state.volume_slider_value;
self.app_state.volume_dragged = false;
if self.config.save_volume {
let mut daemon_config = get_daemon_config();
daemon_config.default_volume = Some(new_volume);
daemon_config.save_to_file().ok();
}
} else {
self.app_state.volume_slider_value = self.audio_player_state.volume;
}
ctx.request_repaint_after_secs(1.0 / 60.0); ctx.request_repaint_after_secs(1.0 / 60.0);
} }
} }
+289 -144
View File
@@ -1,13 +1,14 @@
use crate::{ use crate::{
types::pipewire::{AudioDevice, DeviceType, Terminate}, types::pipewire::{DeviceType, Terminate},
utils::{ utils::{
daemon::get_daemon_config, daemon::get_daemon_config,
pipewire::{create_link, get_all_devices, get_device}, pipewire::{create_link, get_device},
}, },
}; };
use rodio::{Decoder, OutputStream, OutputStreamBuilder, Sink, Source}; use rodio::{Decoder, OutputStream, OutputStreamBuilder, Sink, Source};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use std::{ use std::{
collections::HashMap,
error::Error, error::Error,
fs, fs,
path::{Path, PathBuf}, path::{Path, PathBuf},
@@ -22,53 +23,65 @@ pub enum PlayerState {
Playing, Playing,
} }
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct TrackInfo {
pub id: u32,
pub path: PathBuf,
pub duration: Option<f32>,
pub position: f32,
pub volume: f32,
pub looped: bool,
pub paused: bool,
}
#[derive(Default, Debug, Clone, Serialize, Deserialize)]
pub struct FullState {
pub state: PlayerState,
pub tracks: Vec<TrackInfo>,
pub volume: f32,
pub current_input: String,
pub all_inputs: HashMap<String, String>,
}
pub struct PlayingSound {
pub id: u32,
pub sink: Sink,
pub path: PathBuf,
pub duration: Option<f32>,
pub looped: bool,
pub volume: f32,
}
pub struct AudioPlayer { pub struct AudioPlayer {
_stream_handle: OutputStream, pub stream_handle: OutputStream,
sink: Sink, pub tracks: HashMap<u32, PlayingSound>,
pub next_id: u32,
input_link_sender: Option<pipewire::channel::Sender<Terminate>>, input_link_sender: Option<pipewire::channel::Sender<Terminate>>,
pub current_input_device: Option<AudioDevice>, pub input_device_name: Option<String>,
pub volume: f32, pub volume: f32, // Master volume
pub duration: Option<f32>,
pub current_file_path: Option<PathBuf>,
pub looped: bool,
} }
impl AudioPlayer { impl AudioPlayer {
pub async fn new() -> Result<Self, Box<dyn Error>> { pub async fn new() -> Result<Self, Box<dyn Error>> {
let daemon_config = get_daemon_config(); let daemon_config = get_daemon_config();
let default_volume = daemon_config.default_volume.unwrap_or(1.0); let default_volume = daemon_config.default_volume.unwrap_or(1.0);
let mut default_input_device: Option<AudioDevice> = None;
if let Some(name) = daemon_config.default_input_name
&& let Ok(device) = get_device(&name).await
&& device.device_type == DeviceType::Input
{
default_input_device = Some(device);
}
let stream_handle = OutputStreamBuilder::open_default_stream()?; let stream_handle = OutputStreamBuilder::open_default_stream()?;
let sink = Sink::connect_new(stream_handle.mixer());
sink.set_volume(default_volume);
let mut audio_player = AudioPlayer { let mut audio_player = AudioPlayer {
_stream_handle: stream_handle, stream_handle,
sink, tracks: HashMap::new(),
next_id: 1,
input_link_sender: None, input_link_sender: None,
current_input_device: default_input_device.clone(), input_device_name: daemon_config.default_input_name.clone(),
volume: default_volume, volume: default_volume,
duration: None,
current_file_path: None,
looped: false,
}; };
if default_input_device.is_some() { if audio_player.input_device_name.is_some() {
audio_player.link_devices().await?; audio_player.link_devices().await?;
} }
@@ -78,8 +91,11 @@ impl AudioPlayer {
fn abort_link_thread(&mut self) { fn abort_link_thread(&mut self) {
if let Some(sender) = &self.input_link_sender { if let Some(sender) = &self.input_link_sender {
match sender.send(Terminate {}) { match sender.send(Terminate {}) {
Ok(_) => println!("Sent terminate signal to link thread"), Ok(_) => {
Err(_) => println!("Failed to send terminate signal to link thread"), println!("Sent terminate signal to link thread");
self.input_link_sender = None;
}
Err(_) => eprintln!("Failed to send terminate signal to link thread"),
} }
} }
} }
@@ -87,118 +103,169 @@ impl AudioPlayer {
async fn link_devices(&mut self) -> Result<(), Box<dyn Error>> { async fn link_devices(&mut self) -> Result<(), Box<dyn Error>> {
self.abort_link_thread(); self.abort_link_thread();
if self.current_input_device.is_none() { let input_device;
println!("No input device selected, skipping device linking"); if let Some(input_device_name) = &self.input_device_name {
return Ok(()); if let Ok(device) = get_device(input_device_name).await {
} input_device = device;
} else {
let (input_devices, _) = get_all_devices().await?; eprintln!(
"Could not find selected input device {}, skipping device linking",
let mut pwsp_daemon_input: Option<AudioDevice> = None; input_device_name
for input_device in input_devices { );
if input_device.name == "pwsp-virtual-mic" { return Ok(());
pwsp_daemon_input = Some(input_device);
break;
} }
} } else {
eprintln!("No input device selected, skipping device linking");
if pwsp_daemon_input.is_none() {
println!("Could not find pwsp-daemon input device, skipping device linking");
return Ok(()); return Ok(());
} }
let pwsp_daemon_input = pwsp_daemon_input.unwrap(); let daemon_input;
if let Ok(device) = get_device("pwsp-virtual-mic").await {
daemon_input = device;
} else {
eprintln!("Could not find pwsp-virtual-mic device, skipping device linking");
return Ok(());
}
let Some(output_fl) = input_device.output_fl.clone() else {
eprintln!("Failed to get pwsp-daemon output_fl");
return Ok(());
};
let Some(output_fr) = input_device.output_fr.clone() else {
eprintln!("Failed to get pwsp-daemon output_fr");
return Ok(());
};
let Some(input_fl) = daemon_input.input_fl.clone() else {
eprintln!("Failed to get pwsp-daemon input_fl");
return Ok(());
};
let Some(input_fr) = daemon_input.input_fr.clone() else {
eprintln!("Failed to get pwsp-daemon input_fr");
return Ok(());
};
let current_input_device = self.current_input_device.clone().unwrap();
let output_fl = current_input_device
.clone()
.output_fl
.expect("Failed to get pwsp-daemon output_fl");
let output_fr = current_input_device
.clone()
.output_fr
.expect("Failed to get pwsp-daemon output_fl");
let input_fl = pwsp_daemon_input
.clone()
.input_fl
.expect("Failed to get pwsp-daemon input_fl");
let input_fr = pwsp_daemon_input
.clone()
.input_fr
.expect("Failed to get pwsp-daemon input_fr");
self.input_link_sender = Some(create_link(output_fl, output_fr, input_fl, input_fr)?); self.input_link_sender = Some(create_link(output_fl, output_fr, input_fl, input_fr)?);
Ok(()) Ok(())
} }
pub fn pause(&mut self) { pub fn pause(&mut self, id: Option<u32>) {
if self.get_state() == PlayerState::Playing { if let Some(id) = id {
self.sink.pause(); if let Some(sound) = self.tracks.get_mut(&id) {
} sound.sink.pause();
} }
pub fn resume(&mut self) {
if self.get_state() == PlayerState::Paused {
self.sink.play();
}
}
pub fn stop(&mut self) {
self.sink.stop();
}
pub fn is_paused(&self) -> bool {
self.sink.is_paused()
}
pub fn get_state(&self) -> PlayerState {
if self.sink.len() == 0 {
return PlayerState::Stopped;
}
if self.sink.is_paused() {
return PlayerState::Paused;
}
PlayerState::Playing
}
pub fn set_volume(&mut self, volume: f32) {
self.volume = volume;
self.sink.set_volume(volume);
}
pub fn get_position(&self) -> f32 {
if self.get_state() == PlayerState::Stopped {
return 0.0;
}
self.sink.get_pos().as_secs_f32()
}
pub fn seek(&mut self, mut position: f32) -> Result<(), Box<dyn Error>> {
if position < 0.0 {
position = 0.0;
}
match self.sink.try_seek(Duration::from_secs_f32(position)) {
Ok(_) => Ok(()),
Err(err) => Err(err.into()),
}
}
pub fn get_duration(&mut self) -> Result<f32, Box<dyn Error>> {
if self.get_state() == PlayerState::Stopped {
Err("Nothing is playing right now".into())
} else { } else {
match self.duration { for sound in self.tracks.values_mut() {
Some(duration) => Ok(duration), sound.sink.pause();
None => Err("Couldn't determine duration for current file".into()),
} }
} }
} }
pub async fn play(&mut self, file_path: &Path) -> Result<(), Box<dyn Error>> { pub fn resume(&mut self, id: Option<u32>) {
if let Some(id) = id {
if let Some(sound) = self.tracks.get_mut(&id) {
sound.sink.play();
}
} else {
for sound in self.tracks.values_mut() {
sound.sink.play();
}
}
}
pub fn stop(&mut self, id: Option<u32>) {
if let Some(id) = id {
self.tracks.remove(&id);
} else {
self.tracks.clear();
}
}
pub fn is_paused(&self) -> bool {
if self.tracks.is_empty() {
return false;
}
self.tracks.values().all(|s| s.sink.is_paused())
}
pub fn get_state(&self) -> PlayerState {
if self.tracks.is_empty() {
return PlayerState::Stopped;
}
if self
.tracks
.values()
.any(|s| !s.sink.is_paused() && !s.sink.empty())
{
return PlayerState::Playing;
}
if self.is_paused() {
return PlayerState::Paused;
}
PlayerState::Stopped
}
pub fn set_volume(&mut self, volume: f32, id: Option<u32>) {
if let Some(id) = id {
if let Some(sound) = self.tracks.get_mut(&id) {
sound.volume = volume;
sound.sink.set_volume(self.volume * volume);
}
} else {
self.volume = volume;
for sound in self.tracks.values_mut() {
sound.sink.set_volume(self.volume * sound.volume);
}
}
}
pub fn get_position(&self, id: Option<u32>) -> f32 {
if let Some(id) = id {
if let Some(sound) = self.tracks.get(&id) {
return sound.sink.get_pos().as_secs_f32();
}
} else if let Some(sound) = self.tracks.values().last() {
// Fallback to last added track if no ID
return sound.sink.get_pos().as_secs_f32();
}
0.0
}
pub fn seek(&mut self, position: f32, id: Option<u32>) -> Result<(), Box<dyn Error>> {
let position = if position < 0.0 { 0.0 } else { position };
if let Some(id) = id {
if let Some(sound) = self.tracks.get_mut(&id) {
sound.sink.try_seek(Duration::from_secs_f32(position))?;
}
} else {
// Seek all? Or last? Let's seek all for now if no ID provided
for sound in self.tracks.values_mut() {
sound.sink.try_seek(Duration::from_secs_f32(position)).ok();
}
}
Ok(())
}
pub fn get_duration(&mut self, id: Option<u32>) -> Result<f32, Box<dyn Error>> {
if let Some(id) = id {
if let Some(sound) = self.tracks.get(&id) {
return sound.duration.ok_or("Unknown duration".into());
}
} else if let Some(sound) = self.tracks.values().last() {
return sound.duration.ok_or("Unknown duration".into());
}
Err("No track playing".into())
}
pub async fn play(
&mut self,
file_path: &Path,
concurrent: bool,
) -> Result<u32, Box<dyn Error>> {
if !file_path.exists() { if !file_path.exists() {
return Err(format!("File does not exist: {}", file_path.display()).into()); return Err(format!("File does not exist: {}", file_path.display()).into());
} }
@@ -206,30 +273,108 @@ impl AudioPlayer {
let file = fs::File::open(file_path)?; let file = fs::File::open(file_path)?;
match Decoder::try_from(file) { match Decoder::try_from(file) {
Ok(source) => { Ok(source) => {
self.current_file_path = Some(file_path.to_path_buf()); if !concurrent {
self.tracks.clear();
if let Some(duration) = source.total_duration() {
self.duration = Some(duration.as_secs_f32());
} else {
self.duration = None;
} }
self.sink.stop(); let id = self.next_id;
self.sink.append(source); self.next_id += 1;
self.sink.play();
self.link_devices().await?;
Ok(()) let duration = source.total_duration().map(|d| d.as_secs_f32());
let sink = Sink::connect_new(self.stream_handle.mixer());
sink.set_volume(self.volume); // Default volume is 1.0 * master
sink.append(source);
sink.play();
let sound = PlayingSound {
id,
sink,
path: file_path.to_path_buf(),
duration,
looped: false,
volume: 1.0,
};
self.tracks.insert(id, sound);
Ok(id)
} }
Err(err) => Err(err.into()), Err(err) => Err(err.into()),
} }
} }
pub fn get_current_file_path(&mut self) -> &Option<PathBuf> { pub fn set_loop(&mut self, enabled: bool, id: Option<u32>) {
if self.get_state() == PlayerState::Stopped && !self.looped { if let Some(id) = id {
self.current_file_path = None; if let Some(sound) = self.tracks.get_mut(&id) {
sound.looped = enabled;
}
} else {
// Set loop for all? Or just last?
// Let's set for all.
for sound in self.tracks.values_mut() {
sound.looped = enabled;
}
} }
&self.current_file_path }
pub fn get_tracks(&self) -> Vec<TrackInfo> {
let mut tracks: Vec<_> = self
.tracks
.values()
.map(|sound| TrackInfo {
id: sound.id,
path: sound.path.clone(),
duration: sound.duration,
position: sound.sink.get_pos().as_secs_f32(),
volume: sound.volume,
looped: sound.looped,
paused: sound.sink.is_paused(),
})
.collect();
tracks.sort_by_key(|t| t.id);
tracks
}
pub async fn update(&mut self) {
if let Some(input_device_name) = &self.input_device_name {
// Unlink devices if selected input device was removed
if self.input_link_sender.is_some() && get_device(input_device_name).await.is_err() {
// Selected input device was removed
eprintln!(
"Selected input device {} was removed, unlinking devices",
input_device_name
);
self.abort_link_thread();
}
// Link devices if not linked
else if self.input_link_sender.is_none() {
self.link_devices().await.ok();
}
}
// Handle looped sounds
let mut restarts = vec![];
for (id, sound) in &self.tracks {
if sound.sink.empty() && sound.looped {
restarts.push(*id);
}
}
for id in restarts {
if let Some(sound) = self.tracks.get_mut(&id) {
if let Ok(file) = fs::File::open(&sound.path) {
if let Ok(source) = Decoder::try_from(file) {
sound.sink.append(source);
sound.sink.play();
}
}
}
}
self.tracks
.retain(|_, sound| !sound.sink.empty() || sound.looped);
} }
pub async fn set_current_input_device(&mut self, name: &str) -> Result<(), Box<dyn Error>> { pub async fn set_current_input_device(&mut self, name: &str) -> Result<(), Box<dyn Error>> {
@@ -239,7 +384,7 @@ impl AudioPlayer {
return Err("Selected device is not an input device".into()); return Err("Selected device is not an input device".into());
} }
self.current_input_device = Some(input_device); self.input_device_name = Some(name.to_string());
self.link_devices().await?; self.link_devices().await?;
+135 -51
View File
@@ -1,9 +1,15 @@
use crate::{ use crate::{
types::{audio_player::PlayerState, socket::Response}, types::{
utils::{daemon::get_audio_player, pipewire::get_all_devices}, audio_player::{FullState, PlayerState},
socket::Response,
},
utils::{
daemon::get_audio_player,
pipewire::{get_all_devices, get_device},
},
}; };
use async_trait::async_trait; use async_trait::async_trait;
use std::path::PathBuf; use std::{collections::HashMap, path::PathBuf};
#[async_trait] #[async_trait]
pub trait Executable { pub trait Executable {
@@ -12,13 +18,21 @@ pub trait Executable {
pub struct PingCommand {} pub struct PingCommand {}
pub struct PauseCommand {} pub struct PauseCommand {
pub id: Option<u32>,
}
pub struct ResumeCommand {} pub struct ResumeCommand {
pub id: Option<u32>,
}
pub struct TogglePauseCommand {} pub struct TogglePauseCommand {
pub id: Option<u32>,
}
pub struct StopCommand {} pub struct StopCommand {
pub id: Option<u32>,
}
pub struct IsPausedCommand {} pub struct IsPausedCommand {}
@@ -28,21 +42,28 @@ pub struct GetVolumeCommand {}
pub struct SetVolumeCommand { pub struct SetVolumeCommand {
pub volume: Option<f32>, pub volume: Option<f32>,
pub id: Option<u32>,
} }
pub struct GetPositionCommand {} pub struct GetPositionCommand {
pub id: Option<u32>,
}
pub struct SeekCommand { pub struct SeekCommand {
pub position: Option<f32>, pub position: Option<f32>,
pub id: Option<u32>,
} }
pub struct GetDurationCommand {} pub struct GetDurationCommand {
pub id: Option<u32>,
}
pub struct PlayCommand { pub struct PlayCommand {
pub file_path: Option<PathBuf>, pub file_path: Option<PathBuf>,
pub concurrent: Option<bool>,
} }
pub struct GetCurrentFilePathCommand {} pub struct GetTracksCommand {}
pub struct GetCurrentInputCommand {} pub struct GetCurrentInputCommand {}
@@ -52,13 +73,16 @@ pub struct SetCurrentInputCommand {
pub name: Option<String>, pub name: Option<String>,
} }
pub struct GetLoopCommand {}
pub struct SetLoopCommand { pub struct SetLoopCommand {
pub enabled: Option<bool>, pub enabled: Option<bool>,
pub id: Option<u32>,
} }
pub struct ToggleLoopCommand {} pub struct ToggleLoopCommand {
pub id: Option<u32>,
}
pub struct GetFullStateCommand {}
#[async_trait] #[async_trait]
impl Executable for PingCommand { impl Executable for PingCommand {
@@ -71,7 +95,7 @@ impl Executable for PingCommand {
impl Executable for PauseCommand { impl Executable for PauseCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
audio_player.pause(); audio_player.pause(self.id);
Response::new(true, "Audio was paused") Response::new(true, "Audio was paused")
} }
} }
@@ -80,7 +104,7 @@ impl Executable for PauseCommand {
impl Executable for ResumeCommand { impl Executable for ResumeCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
audio_player.resume(); audio_player.resume(self.id);
Response::new(true, "Audio was resumed") Response::new(true, "Audio was resumed")
} }
} }
@@ -94,12 +118,31 @@ impl Executable for TogglePauseCommand {
return Response::new(false, "Audio is not playing"); return Response::new(false, "Audio is not playing");
} }
if audio_player.is_paused() { // This logic is a bit tricky with multiple tracks.
audio_player.resume(); // If ID is provided, toggle that track.
Response::new(true, "Audio was resumed") // If not, toggle global pause state?
// For now, let's just use pause/resume based on global state if no ID.
if let Some(id) = self.id {
if let Some(track) = audio_player.tracks.get(&id) {
if track.sink.is_paused() {
audio_player.resume(Some(id));
Response::new(true, "Audio was resumed")
} else {
audio_player.pause(Some(id));
Response::new(true, "Audio was paused")
}
} else {
Response::new(false, "Track not found")
}
} else { } else {
audio_player.pause(); if audio_player.is_paused() {
Response::new(true, "Audio was paused") audio_player.resume(None);
Response::new(true, "Audio was resumed")
} else {
audio_player.pause(None);
Response::new(true, "Audio was paused")
}
} }
} }
} }
@@ -108,7 +151,7 @@ impl Executable for TogglePauseCommand {
impl Executable for StopCommand { impl Executable for StopCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
audio_player.stop(); audio_player.stop(self.id);
Response::new(true, "Audio was stopped") Response::new(true, "Audio was stopped")
} }
} }
@@ -145,7 +188,7 @@ impl Executable for SetVolumeCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
if let Some(volume) = self.volume { if let Some(volume) = self.volume {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
audio_player.set_volume(volume); audio_player.set_volume(volume, self.id);
Response::new(true, format!("Audio volume was set to {}", volume)) Response::new(true, format!("Audio volume was set to {}", volume))
} else { } else {
Response::new(false, "Invalid volume value") Response::new(false, "Invalid volume value")
@@ -157,7 +200,7 @@ impl Executable for SetVolumeCommand {
impl Executable for GetPositionCommand { impl Executable for GetPositionCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let audio_player = get_audio_player().await.lock().await; let audio_player = get_audio_player().await.lock().await;
let position = audio_player.get_position(); let position = audio_player.get_position(self.id);
Response::new(true, position.to_string()) Response::new(true, position.to_string())
} }
} }
@@ -167,7 +210,7 @@ impl Executable for SeekCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
if let Some(position) = self.position { if let Some(position) = self.position {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
match audio_player.seek(position) { match audio_player.seek(position, self.id) {
Ok(_) => Response::new(true, format!("Audio position was set to {}", position)), Ok(_) => Response::new(true, format!("Audio position was set to {}", position)),
Err(err) => Response::new(false, err.to_string()), Err(err) => Response::new(false, err.to_string()),
} }
@@ -181,7 +224,7 @@ impl Executable for SeekCommand {
impl Executable for GetDurationCommand { impl Executable for GetDurationCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
match audio_player.get_duration() { match audio_player.get_duration(self.id) {
Ok(duration) => Response::new(true, duration.to_string()), Ok(duration) => Response::new(true, duration.to_string()),
Err(err) => Response::new(false, err.to_string()), Err(err) => Response::new(false, err.to_string()),
} }
@@ -193,8 +236,11 @@ impl Executable for PlayCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
if let Some(file_path) = &self.file_path { if let Some(file_path) = &self.file_path {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
match audio_player.play(file_path).await { match audio_player
Ok(_) => Response::new(true, format!("Now playing {}", file_path.display())), .play(file_path, self.concurrent.unwrap_or(false))
.await
{
Ok(id) => Response::new(true, id.to_string()),
Err(err) => Response::new(false, err.to_string()), Err(err) => Response::new(false, err.to_string()),
} }
} else { } else {
@@ -204,15 +250,11 @@ impl Executable for PlayCommand {
} }
#[async_trait] #[async_trait]
impl Executable for GetCurrentFilePathCommand { impl Executable for GetTracksCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let mut audio_player = get_audio_player().await.lock().await; let audio_player = get_audio_player().await.lock().await;
let current_file_path = audio_player.get_current_file_path(); let tracks = audio_player.get_tracks();
if let Some(current_file_path) = current_file_path { Response::new(true, serde_json::to_string(&tracks).unwrap())
Response::new(true, current_file_path.to_str().unwrap())
} else {
Response::new(false, "No file is playing")
}
} }
} }
@@ -220,11 +262,15 @@ impl Executable for GetCurrentFilePathCommand {
impl Executable for GetCurrentInputCommand { impl Executable for GetCurrentInputCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let audio_player = get_audio_player().await.lock().await; let audio_player = get_audio_player().await.lock().await;
if let Some(input_device) = &audio_player.current_input_device { if let Some(input_device_name) = &audio_player.input_device_name {
Response::new( if let Ok(input_device) = get_device(input_device_name).await {
true, Response::new(
format!("{} - {}", input_device.name, input_device.nick), true,
) format!("{} - {}", input_device.name, input_device.nick),
)
} else {
Response::new(false, "Failed to get current input device")
}
} else { } else {
Response::new(false, "No input device selected") Response::new(false, "No input device selected")
} }
@@ -265,14 +311,6 @@ impl Executable for SetCurrentInputCommand {
} }
} }
#[async_trait]
impl Executable for GetLoopCommand {
async fn execute(&self) -> Response {
let audio_player = get_audio_player().await.lock().await;
Response::new(true, audio_player.looped.to_string())
}
}
#[async_trait] #[async_trait]
impl Executable for SetLoopCommand { impl Executable for SetLoopCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
@@ -280,7 +318,7 @@ impl Executable for SetLoopCommand {
match self.enabled { match self.enabled {
Some(enabled) => { Some(enabled) => {
audio_player.looped = enabled; audio_player.set_loop(enabled, self.id);
Response::new(true, format!("Loop was set to {}", enabled)) Response::new(true, format!("Loop was set to {}", enabled))
} }
None => Response::new(false, "Invalid enabled value"), None => Response::new(false, "Invalid enabled value"),
@@ -292,7 +330,53 @@ impl Executable for SetLoopCommand {
impl Executable for ToggleLoopCommand { impl Executable for ToggleLoopCommand {
async fn execute(&self) -> Response { async fn execute(&self) -> Response {
let mut audio_player = get_audio_player().await.lock().await; let mut audio_player = get_audio_player().await.lock().await;
audio_player.looped = !audio_player.looped; if let Some(id) = self.id {
Response::new(true, format!("Loop was set to {}", audio_player.looped)) if let Some(track) = audio_player.tracks.get_mut(&id) {
track.looped = !track.looped;
Response::new(true, format!("Loop was set to {}", track.looped))
} else {
Response::new(false, "Track not found")
}
} else {
// Toggle all?
for track in audio_player.tracks.values_mut() {
track.looped = !track.looped;
}
Response::new(true, "Loop toggled for all tracks")
}
}
}
#[async_trait]
impl Executable for GetFullStateCommand {
async fn execute(&self) -> Response {
let (input_devices, _output_devices) = get_all_devices().await.unwrap();
let mut all_inputs = HashMap::new();
let mut current_input_nick = String::new();
let audio_player = get_audio_player().await.lock().await;
for device in input_devices {
if device.name == "pwsp-virtual-mic" {
continue;
}
if let Some(current_input_name) = &audio_player.input_device_name {
if device.name == *current_input_name {
current_input_nick = format!("{} - {}", device.name, device.nick);
}
}
all_inputs.insert(device.name, device.nick);
}
let full_state = FullState {
state: audio_player.get_state(),
tracks: audio_player.get_tracks(),
volume: audio_player.volume,
current_input: current_input_nick,
all_inputs,
};
Response::new(true, serde_json::to_string(&full_state).unwrap())
} }
} }
+7 -3
View File
@@ -1,8 +1,9 @@
use crate::utils::config::get_config_path; use crate::utils::config::get_config_path;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use std::{collections::HashSet, error::Error, fs, path::PathBuf}; use std::{error::Error, fs, path::PathBuf};
#[derive(Default, Clone, Serialize, Deserialize)] #[derive(Default, Clone, Serialize, Deserialize)]
#[serde(default)]
pub struct DaemonConfig { pub struct DaemonConfig {
pub default_input_name: Option<String>, pub default_input_name: Option<String>,
pub default_volume: Option<f32>, pub default_volume: Option<f32>,
@@ -30,28 +31,31 @@ impl DaemonConfig {
} }
#[derive(Clone, Serialize, Deserialize)] #[derive(Clone, Serialize, Deserialize)]
#[serde(default)]
pub struct GuiConfig { pub struct GuiConfig {
pub scale_factor: f32, pub scale_factor: f32,
pub left_panel_width: f32,
pub save_volume: bool, pub save_volume: bool,
pub save_input: bool, pub save_input: bool,
pub save_scale_factor: bool, pub save_scale_factor: bool,
pub pause_on_exit: bool, pub pause_on_exit: bool,
pub dirs: HashSet<PathBuf>, pub dirs: Vec<PathBuf>,
} }
impl Default for GuiConfig { impl Default for GuiConfig {
fn default() -> Self { fn default() -> Self {
GuiConfig { GuiConfig {
scale_factor: 1.0, scale_factor: 1.0,
left_panel_width: 280.0,
save_volume: false, save_volume: false,
save_input: false, save_input: false,
save_scale_factor: false, save_scale_factor: false,
pause_on_exit: false, pause_on_exit: false,
dirs: HashSet::default(), dirs: vec![],
} }
} }
} }
+26 -16
View File
@@ -1,49 +1,59 @@
use crate::types::audio_player::PlayerState; use crate::types::audio_player::{PlayerState, TrackInfo};
use egui::Id; use egui::Id;
use std::{ use std::{
collections::{HashMap, HashSet}, collections::{HashMap, HashSet},
path::PathBuf, path::PathBuf,
time::Instant,
}; };
#[derive(Default, Debug)] #[derive(Default, Debug)]
pub struct AppState { pub struct TrackUiState {
pub search_query: String,
pub position_slider_value: f32, pub position_slider_value: f32,
pub volume_slider_value: f32, pub volume_slider_value: f32,
pub position_dragged: bool, pub position_dragged: bool,
pub volume_dragged: bool, pub volume_dragged: bool,
pub ignore_position_update_until: Option<Instant>,
pub ignore_volume_update_until: Option<Instant>,
}
#[derive(Default, Debug)]
pub struct AppState {
pub search_query: String,
pub track_ui_states: HashMap<u32, TrackUiState>,
pub show_settings: bool, pub show_settings: bool,
pub volume_dragged: bool,
pub force_focus_search: bool,
pub volume_slider_value: f32,
pub search_field_id: Option<Id>,
pub ignore_volume_update_until: Option<Instant>,
pub current_dir: Option<PathBuf>, pub current_dir: Option<PathBuf>,
pub dirs: HashSet<PathBuf>, pub dirs: Vec<PathBuf>,
pub selected_file: Option<PathBuf>, pub selected_file: Option<PathBuf>,
pub files: HashSet<PathBuf>, pub files: HashSet<PathBuf>,
pub search_field_id: Option<Id>,
pub force_focus_id: Option<Id>,
} }
#[derive(Default, Debug, Clone)] #[derive(Default, Debug, Clone)]
pub struct AudioPlayerState { pub struct AudioPlayerState {
pub state: PlayerState, pub state: PlayerState,
pub new_state: Option<PlayerState>, pub new_state: Option<PlayerState>,
pub current_file_path: PathBuf,
pub is_paused: bool, pub tracks: Vec<TrackInfo>,
pub looped: bool,
pub volume: f32, pub volume: f32, // Master volume
pub new_volume: Option<f32>,
pub position: f32,
pub new_position: Option<f32>,
pub duration: f32,
pub current_input: String, pub current_input: String,
pub all_inputs: HashMap<String, String>, pub all_inputs: HashMap<String, String>,
pub is_daemon_running: bool,
} }
+86 -26
View File
@@ -24,24 +24,54 @@ impl Request {
Request::new("ping", vec![]) Request::new("ping", vec![])
} }
pub fn pause() -> Self { pub fn pause(id: Option<u32>) -> Self {
Request::new("pause", vec![]) let mut args = vec![];
let id_str;
if let Some(id) = id {
id_str = id.to_string();
args.push(("id", id_str.as_str()));
}
Request::new("pause", args)
} }
pub fn resume() -> Self { pub fn resume(id: Option<u32>) -> Self {
Request::new("resume", vec![]) let mut args = vec![];
let id_str;
if let Some(id) = id {
id_str = id.to_string();
args.push(("id", id_str.as_str()));
}
Request::new("resume", args)
} }
pub fn toggle_pause() -> Self { pub fn toggle_pause(id: Option<u32>) -> Self {
Request::new("toggle_pause", vec![]) let mut args = vec![];
let id_str;
if let Some(id) = id {
id_str = id.to_string();
args.push(("id", id_str.as_str()));
}
Request::new("toggle_pause", args)
} }
pub fn stop() -> Self { pub fn stop(id: Option<u32>) -> Self {
Request::new("stop", vec![]) let mut args = vec![];
let id_str;
if let Some(id) = id {
id_str = id.to_string();
args.push(("id", id_str.as_str()));
}
Request::new("stop", args)
} }
pub fn play(file_path: &str) -> Self { pub fn play(file_path: &str, concurrent: bool) -> Self {
Request::new("play", vec![("file_path", file_path)]) Request::new(
"play",
vec![
("file_path", file_path),
("concurrent", &concurrent.to_string()),
],
)
} }
pub fn get_is_paused() -> Self { pub fn get_is_paused() -> Self {
@@ -52,20 +82,32 @@ impl Request {
Request::new("get_volume", vec![]) Request::new("get_volume", vec![])
} }
pub fn get_position() -> Self { pub fn get_position(id: Option<u32>) -> Self {
Request::new("get_position", vec![]) let mut args = vec![];
let id_str;
if let Some(id) = id {
id_str = id.to_string();
args.push(("id", id_str.as_str()));
}
Request::new("get_position", args)
} }
pub fn get_duration() -> Self { pub fn get_duration(id: Option<u32>) -> Self {
Request::new("get_duration", vec![]) let mut args = vec![];
let id_str;
if let Some(id) = id {
id_str = id.to_string();
args.push(("id", id_str.as_str()));
}
Request::new("get_duration", args)
} }
pub fn get_state() -> Self { pub fn get_state() -> Self {
Request::new("get_state", vec![]) Request::new("get_state", vec![])
} }
pub fn get_current_file_path() -> Self { pub fn get_tracks() -> Self {
Request::new("get_current_file_path", vec![]) Request::new("get_tracks", vec![])
} }
pub fn get_input() -> Self { pub fn get_input() -> Self {
@@ -76,28 +118,46 @@ impl Request {
Request::new("get_inputs", vec![]) Request::new("get_inputs", vec![])
} }
pub fn set_volume(volume: f32) -> Self { pub fn set_volume(volume: f32, id: Option<u32>) -> Self {
Request::new("set_volume", vec![("volume", &volume.to_string())]) let mut args = vec![("volume".to_string(), volume.to_string())];
if let Some(id) = id {
args.push(("id".to_string(), id.to_string()));
}
Request::new("set_volume".to_string(), args)
} }
pub fn seek(position: f32) -> Self { pub fn seek(position: f32, id: Option<u32>) -> Self {
Request::new("seek", vec![("position", &position.to_string())]) let mut args = vec![("position".to_string(), position.to_string())];
if let Some(id) = id {
args.push(("id".to_string(), id.to_string()));
}
Request::new("seek".to_string(), args)
} }
pub fn set_input(name: &str) -> Self { pub fn set_input(name: &str) -> Self {
Request::new("set_input", vec![("input_name", name)]) Request::new("set_input", vec![("input_name", name)])
} }
pub fn get_loop() -> Self { pub fn set_loop(enabled: &str, id: Option<u32>) -> Self {
Request::new("get_loop", vec![]) let mut args = vec![("enabled".to_string(), enabled.to_string())];
if let Some(id) = id {
args.push(("id".to_string(), id.to_string()));
}
Request::new("set_loop".to_string(), args)
} }
pub fn set_loop(enabled: &str) -> Self { pub fn toggle_loop(id: Option<u32>) -> Self {
Request::new("set_loop", vec![("enabled", enabled)]) let mut args = vec![];
let id_str;
if let Some(id) = id {
id_str = id.to_string();
args.push(("id", id_str.as_str()));
}
Request::new("toggle_loop", args)
} }
pub fn toggle_loop() -> Self { pub fn get_full_state() -> Self {
Request::new("toggle_loop", vec![]) Request::new("get_full_state", vec![])
} }
} }
+24 -13
View File
@@ -3,12 +3,14 @@ use crate::types::{commands::*, socket::Request};
use std::path::PathBuf; use std::path::PathBuf;
pub fn parse_command(request: &Request) -> Option<Box<dyn Executable + Send>> { pub fn parse_command(request: &Request) -> Option<Box<dyn Executable + Send>> {
let id = request.args.get("id").and_then(|s| s.parse::<u32>().ok());
match request.name.as_str() { match request.name.as_str() {
"ping" => Some(Box::new(PingCommand {})), "ping" => Some(Box::new(PingCommand {})),
"pause" => Some(Box::new(PauseCommand {})), "pause" => Some(Box::new(PauseCommand { id })),
"resume" => Some(Box::new(ResumeCommand {})), "resume" => Some(Box::new(ResumeCommand { id })),
"toggle_pause" => Some(Box::new(TogglePauseCommand {})), "toggle_pause" => Some(Box::new(TogglePauseCommand { id })),
"stop" => Some(Box::new(StopCommand {})), "stop" => Some(Box::new(StopCommand { id })),
"is_paused" => Some(Box::new(IsPausedCommand {})), "is_paused" => Some(Box::new(IsPausedCommand {})),
"get_state" => Some(Box::new(GetStateCommand {})), "get_state" => Some(Box::new(GetStateCommand {})),
"get_volume" => Some(Box::new(GetVolumeCommand {})), "get_volume" => Some(Box::new(GetVolumeCommand {})),
@@ -19,9 +21,9 @@ pub fn parse_command(request: &Request) -> Option<Box<dyn Executable + Send>> {
.unwrap_or(&String::new()) .unwrap_or(&String::new())
.parse::<f32>() .parse::<f32>()
.ok(); .ok();
Some(Box::new(SetVolumeCommand { volume })) Some(Box::new(SetVolumeCommand { volume, id }))
} }
"get_position" => Some(Box::new(GetPositionCommand {})), "get_position" => Some(Box::new(GetPositionCommand { id })),
"seek" => { "seek" => {
let position = request let position = request
.args .args
@@ -29,9 +31,9 @@ pub fn parse_command(request: &Request) -> Option<Box<dyn Executable + Send>> {
.unwrap_or(&String::new()) .unwrap_or(&String::new())
.parse::<f32>() .parse::<f32>()
.ok(); .ok();
Some(Box::new(SeekCommand { position })) Some(Box::new(SeekCommand { position, id }))
} }
"get_duration" => Some(Box::new(GetDurationCommand {})), "get_duration" => Some(Box::new(GetDurationCommand { id })),
"play" => { "play" => {
let file_path = request let file_path = request
.args .args
@@ -39,16 +41,24 @@ pub fn parse_command(request: &Request) -> Option<Box<dyn Executable + Send>> {
.unwrap_or(&String::new()) .unwrap_or(&String::new())
.parse::<PathBuf>() .parse::<PathBuf>()
.ok(); .ok();
Some(Box::new(PlayCommand { file_path })) let concurrent = request
.args
.get("concurrent")
.unwrap_or(&String::new())
.parse::<bool>()
.ok();
Some(Box::new(PlayCommand {
file_path,
concurrent,
}))
} }
"get_current_file_path" => Some(Box::new(GetCurrentFilePathCommand {})), "get_tracks" => Some(Box::new(GetTracksCommand {})),
"get_input" => Some(Box::new(GetCurrentInputCommand {})), "get_input" => Some(Box::new(GetCurrentInputCommand {})),
"get_inputs" => Some(Box::new(GetAllInputsCommand {})), "get_inputs" => Some(Box::new(GetAllInputsCommand {})),
"set_input" => { "set_input" => {
let name = Some(request.args.get("input_name").unwrap_or(&String::new())).cloned(); let name = Some(request.args.get("input_name").unwrap_or(&String::new())).cloned();
Some(Box::new(SetCurrentInputCommand { name })) Some(Box::new(SetCurrentInputCommand { name }))
} }
"get_loop" => Some(Box::new(GetLoopCommand {})),
"set_loop" => { "set_loop" => {
let enabled = request let enabled = request
.args .args
@@ -56,9 +66,10 @@ pub fn parse_command(request: &Request) -> Option<Box<dyn Executable + Send>> {
.unwrap_or(&String::new()) .unwrap_or(&String::new())
.parse::<bool>() .parse::<bool>()
.ok(); .ok();
Some(Box::new(SetLoopCommand { enabled })) Some(Box::new(SetLoopCommand { enabled, id }))
} }
"toggle_loop" => Some(Box::new(ToggleLoopCommand {})), "toggle_loop" => Some(Box::new(ToggleLoopCommand { id })),
"get_full_state" => Some(Box::new(GetFullStateCommand {})),
_ => None, _ => None,
} }
} }
+11 -27
View File
@@ -2,10 +2,9 @@ use crate::{
types::{ types::{
audio_player::AudioPlayer, audio_player::AudioPlayer,
config::DaemonConfig, config::DaemonConfig,
pipewire::AudioDevice,
socket::{Request, Response}, socket::{Request, Response},
}, },
utils::pipewire::{create_link, get_all_devices}, utils::pipewire::{create_link, get_device},
}; };
use std::path::PathBuf; use std::path::PathBuf;
use std::{error::Error, fs}; use std::{error::Error, fs};
@@ -36,37 +35,22 @@ pub fn get_daemon_config() -> DaemonConfig {
} }
pub async fn link_player_to_virtual_mic() -> Result<(), Box<dyn Error>> { pub async fn link_player_to_virtual_mic() -> Result<(), Box<dyn Error>> {
let (input_devices, output_devices) = get_all_devices().await?; let pwsp_daemon_output;
if let Ok(device) = get_device("alsa_playback.pwsp-daemon").await {
let mut pwsp_daemon_output: Option<AudioDevice> = None; pwsp_daemon_output = device;
for output_device in output_devices { } else {
if output_device.name == "alsa_playback.pwsp-daemon" { eprintln!("Could not find alsa_playback.pwsp-daemon device, skipping device linking");
pwsp_daemon_output = Some(output_device);
break;
}
}
if pwsp_daemon_output.is_none() {
println!("Could not find pwsp-daemon output device, skipping device linking");
return Ok(()); return Ok(());
} }
let mut pwsp_daemon_input: Option<AudioDevice> = None; let pwsp_daemon_input;
for input_device in input_devices { if let Ok(device) = get_device("pwsp-virtual-mic").await {
if input_device.name == "pwsp-virtual-mic" { pwsp_daemon_input = device;
pwsp_daemon_input = Some(input_device); } else {
break; eprintln!("Could not find pwsp-virtual-mic device, skipping device linking");
}
}
if pwsp_daemon_input.is_none() {
println!("Could not find pwsp-daemon input device, skipping device linking");
return Ok(()); return Ok(());
} }
let pwsp_daemon_output = pwsp_daemon_output.unwrap();
let pwsp_daemon_input = pwsp_daemon_input.unwrap();
let output_fl = pwsp_daemon_output let output_fl = pwsp_daemon_output
.clone() .clone()
.output_fl .output_fl
+33 -124
View File
@@ -1,16 +1,14 @@
use crate::{ use crate::{
types::{ types::{
audio_player::PlayerState, audio_player::FullState,
config::GuiConfig, config::GuiConfig,
gui::AudioPlayerState, gui::AudioPlayerState,
socket::{Request, Response}, socket::{Request, Response},
}, },
utils::daemon::{make_request, wait_for_daemon}, utils::daemon::{is_daemon_running, make_request},
}; };
use std::{ use std::{
collections::HashMap,
error::Error, error::Error,
path::PathBuf,
sync::{Arc, Mutex}, sync::{Arc, Mutex},
}; };
use tokio::time::{Duration, sleep}; use tokio::time::{Duration, sleep};
@@ -31,6 +29,12 @@ pub fn make_request_sync(request: Request) -> Result<Response, Box<dyn Error>> {
}) })
} }
pub fn make_request_async(request: Request) {
tokio::spawn(async move {
make_request(request).await.ok();
});
}
pub fn format_time_pair(position: f32, duration: f32) -> String { pub fn format_time_pair(position: f32, duration: f32) -> String {
fn format_time(seconds: f32) -> String { fn format_time(seconds: f32) -> String {
let total_seconds = seconds.round() as u32; let total_seconds = seconds.round() as u32;
@@ -47,109 +51,24 @@ pub fn start_app_state_thread(audio_player_state_shared: Arc<Mutex<AudioPlayerSt
let sleep_duration = Duration::from_secs_f32(1.0 / 60.0); let sleep_duration = Duration::from_secs_f32(1.0 / 60.0);
loop { loop {
wait_for_daemon().await.ok(); let is_running = is_daemon_running().unwrap_or(false);
let state_req = Request::get_state(); if !is_running {
let file_path_req = Request::get_current_file_path(); {
let is_paused_req = Request::get_is_paused(); let mut guard = audio_player_state_shared.lock().unwrap();
let volume_req = Request::get_volume(); guard.is_daemon_running = false;
let position_req = Request::get_position(); }
let duration_req = Request::get_duration(); sleep(Duration::from_millis(500)).await;
let current_input_req = Request::get_input(); continue;
let all_inputs_req = Request::get_inputs(); }
let looped_req = Request::get_loop();
let ( let full_state_req = Request::get_full_state();
state_res, let full_state_res = make_request(full_state_req).await.unwrap_or_default();
file_path_res,
is_paused_res,
volume_res,
position_res,
duration_res,
current_input_res,
all_inputs_res,
looped_res,
) = tokio::join!(
make_request(state_req),
make_request(file_path_req),
make_request(is_paused_req),
make_request(volume_req),
make_request(position_req),
make_request(duration_req),
make_request(current_input_req),
make_request(all_inputs_req),
make_request(looped_req),
);
let state_res = state_res.unwrap_or_default(); if full_state_res.status {
let file_path_res = file_path_res.unwrap_or_default(); let full_state: FullState =
let is_paused_res = is_paused_res.unwrap_or_default(); serde_json::from_str(&full_state_res.message).unwrap_or_default();
let volume_res = volume_res.unwrap_or_default();
let position_res = position_res.unwrap_or_default();
let duration_res = duration_res.unwrap_or_default();
let current_input_res = current_input_res.unwrap_or_default();
let all_inputs_res = all_inputs_res.unwrap_or_default();
let looped_res = looped_res.unwrap_or_default();
let state = match state_res.status {
true => serde_json::from_str::<PlayerState>(&state_res.message).unwrap(),
false => PlayerState::default(),
};
let file_path = match file_path_res.status {
true => PathBuf::from(file_path_res.message),
false => PathBuf::new(),
};
let is_paused = match is_paused_res.status {
true => is_paused_res.message == "true",
false => false,
};
let volume = match volume_res.status {
true => volume_res.message.parse::<f32>().unwrap(),
false => 0.0,
};
let position = match position_res.status {
true => position_res.message.parse::<f32>().unwrap(),
false => 0.0,
};
let duration = match duration_res.status {
true => duration_res.message.parse::<f32>().unwrap(),
false => 0.0,
};
let current_input = match current_input_res.status {
true => current_input_res
.message
.as_str()
.split(" - ")
.collect::<Vec<&str>>()
.first()
.unwrap()
.to_string(),
false => String::new(),
};
let all_inputs = match all_inputs_res.status {
true => all_inputs_res
.message
.as_str()
.split(';')
.filter_map(|entry| {
let entry = entry.trim();
if entry.is_empty() {
return None;
}
entry
.split_once(" - ")
.map(|(k, v)| (k.trim().to_string(), v.trim().to_string()))
})
.collect::<HashMap<String, String>>(),
false => HashMap::new(),
};
let looped = match looped_res.status {
true => looped_res.message.parse::<bool>().unwrap_or_default(),
false => false,
};
{
let mut guard = audio_player_state_shared.lock().unwrap(); let mut guard = audio_player_state_shared.lock().unwrap();
guard.state = match guard.new_state.clone() { guard.state = match guard.new_state.clone() {
@@ -157,28 +76,18 @@ pub fn start_app_state_thread(audio_player_state_shared: Arc<Mutex<AudioPlayerSt
guard.new_state = None; guard.new_state = None;
new_state new_state
} }
None => state, None => full_state.state,
}; };
guard.current_file_path = file_path; guard.tracks = full_state.tracks;
guard.is_paused = is_paused; guard.volume = full_state.volume;
guard.volume = match guard.new_volume { guard.current_input = full_state
Some(new_volume) => { .current_input
guard.new_volume = None; .split(" - ")
new_volume .next()
} .unwrap_or_default()
None => volume, .to_string();
}; guard.all_inputs = full_state.all_inputs;
guard.position = match guard.new_position { guard.is_daemon_running = true;
Some(new_position) => {
guard.new_position = None;
new_position
}
None => position,
};
guard.duration = if duration > 0.0 { duration } else { 1.0 };
guard.current_input = current_input;
guard.all_inputs = all_inputs;
guard.looped = looped;
} }
sleep(sleep_duration).await; sleep(sleep_duration).await;
+2 -2
View File
@@ -165,8 +165,8 @@ pub async fn get_all_devices() -> Result<(Vec<AudioDevice>, Vec<AudioDevice>), B
input_device.input_fr = Some(port) input_device.input_fr = Some(port)
} }
"capture_MONO" => { "capture_MONO" => {
input_device.input_fl = Some(port.clone()); input_device.output_fl = Some(port.clone());
input_device.input_fr = Some(port); input_device.output_fr = Some(port);
} }
_ => {} _ => {}
} }