use crate::clients::music::{self, MusicClient, PlayerState, PlayerUpdate, Status, Track}; use crate::config::CommonConfig; use crate::modules::{Module, ModuleInfo, ModuleUpdateEvent, ModuleWidget, WidgetContext}; use crate::popup::Popup; use crate::{send_async, try_send}; use color_eyre::Result; use dirs::{audio_dir, home_dir}; use glib::Continue; use gtk::gdk_pixbuf::Pixbuf; use gtk::pango::EllipsizeMode as GtkEllipsizeMode; use gtk::prelude::*; use gtk::{Button, Image, Label, Orientation, Scale}; use regex::Regex; use serde::Deserialize; use std::path::PathBuf; use std::sync::Arc; use std::time::Duration; use tokio::spawn; use tokio::sync::mpsc::{Receiver, Sender}; use tracing::error; #[derive(Debug)] pub enum PlayerCommand { Previous, Play, Pause, Next, Volume(u8), } #[derive(Debug, Deserialize, Clone)] pub struct Icons { /// Icon to display when playing. #[serde(default = "default_icon_play")] play: String, /// Icon to display when paused. #[serde(default = "default_icon_pause")] pause: String, /// Icon to display under volume slider #[serde(default = "default_icon_volume")] volume: String, } impl Default for Icons { fn default() -> Self { Self { pause: default_icon_pause(), play: default_icon_play(), volume: default_icon_volume(), } } } #[derive(Debug, Deserialize, Clone, Copy)] #[serde(rename_all = "snake_case")] pub enum PlayerType { Mpd, Mpris, } impl Default for PlayerType { fn default() -> Self { Self::Mpris } } #[derive(Debug, Deserialize, Clone, Copy)] #[serde(rename_all = "snake_case")] pub enum EllipsizeMode { Start, Middle, End, } impl From for GtkEllipsizeMode { fn from(value: EllipsizeMode) -> Self { match value { EllipsizeMode::Start => Self::Start, EllipsizeMode::Middle => Self::Middle, EllipsizeMode::End => Self::End, } } } #[derive(Debug, Deserialize, Clone, Copy)] #[serde(untagged)] enum TruncateMode { Auto(EllipsizeMode), MaxLength { mode: EllipsizeMode, length: Option, }, } impl TruncateMode { fn mode(&self) -> EllipsizeMode { match self { TruncateMode::Auto(mode) => *mode, TruncateMode::MaxLength { mode, .. } => *mode, } } fn length(&self) -> Option { match self { TruncateMode::Auto(_) => None, TruncateMode::MaxLength { length, .. } => *length, } } } #[derive(Debug, Deserialize, Clone)] pub struct MusicModule { /// Type of player to connect to #[serde(default)] player_type: PlayerType, /// Format of current song info to display on the bar. #[serde(default = "default_format")] format: String, /// Player state icons #[serde(default)] icons: Icons, truncate: Option, // -- MPD -- /// TCP or Unix socket address. #[serde(default = "default_socket")] host: String, /// Path to root of music directory. #[serde(default = "default_music_dir")] music_dir: PathBuf, #[serde(flatten)] pub common: Option, } fn default_socket() -> String { String::from("localhost:6600") } fn default_format() -> String { String::from("{icon} {title} / {artist}") } fn default_icon_play() -> String { String::from("") } fn default_icon_pause() -> String { String::from("") } fn default_icon_volume() -> String { String::from("墳") } fn default_music_dir() -> PathBuf { audio_dir().unwrap_or_else(|| home_dir().map(|dir| dir.join("Music")).unwrap_or_default()) } /// Formats a duration given in seconds /// in hh:mm format fn format_time(duration: Duration) -> String { let time = duration.as_secs(); let minutes = (time / 60) % 60; let seconds = time % 60; format!("{minutes:0>2}:{seconds:0>2}") } /// Extracts the formatting tokens from a formatting string fn get_tokens(re: &Regex, format_string: &str) -> Vec { re.captures_iter(format_string) .map(|caps| caps[1].to_string()) .collect::>() } #[derive(Clone, Debug)] pub struct SongUpdate { song: Track, status: Status, display_string: String, } async fn get_client( player_type: PlayerType, host: &str, music_dir: PathBuf, ) -> Box> { match player_type { PlayerType::Mpd => music::get_client(music::ClientType::Mpd { host, music_dir }), PlayerType::Mpris => music::get_client(music::ClientType::Mpris {}), } .await } impl Module