diff --git a/xmpd-gui/src/components/left_nav/mod.rs b/xmpd-gui/src/components/left_nav/mod.rs index f9774c5..1ed0e9b 100644 --- a/xmpd-gui/src/components/left_nav/mod.rs +++ b/xmpd-gui/src/components/left_nav/mod.rs @@ -1,5 +1,5 @@ use egui::{CursorIcon, RichText, Sense, TextBuffer}; -use xmpd_manifest::store::BaseStore; +use xmpd_manifest::store::{BaseStore, StoreExtras}; use crate::utils::SearchType; use super::{CompGetter, CompUi}; @@ -23,12 +23,7 @@ impl CompUi for LeftNav { ui.vertical(|ui| { let len = state.manifest.store().get_songs().len(); add_playlist_tab(ui, &None, "All Songs", None, len, w); - let mut playlists: Vec<_> = state.manifest.store().get_playlists().into_iter().collect(); - playlists.sort_by(|a, b| { - let a = a.1.name().to_lowercase(); - let b = b.1.name().to_lowercase(); - a.cmp(&b) - }); + let playlists = state.manifest.store().get_playlists_sorted(); let search_text = handle_error_ui!(header::Header::get()).search_text.clone(); let (qtyp, qtxt) = crate::utils::SearchType::from_str(&search_text); for (pid, playlist) in playlists.iter() { diff --git a/xmpd-gui/src/components/song_list/mod.rs b/xmpd-gui/src/components/song_list/mod.rs index bd9395c..3d29cc3 100644 --- a/xmpd-gui/src/components/song_list/mod.rs +++ b/xmpd-gui/src/components/song_list/mod.rs @@ -1,6 +1,6 @@ use egui::{Color32, CursorIcon, ImageSource, RichText, Sense, Vec2}; use xmpd_cache::DlStatus; -use xmpd_manifest::{query, song::Song, store::BaseStore}; +use xmpd_manifest::{query, song::Song, store::{BaseStore, StoreExtras}}; use crate::utils::SearchType; use super::{CompGetter, CompUi}; @@ -24,7 +24,7 @@ impl CompUi for SongList { sl.playable_songs = Self::get_playable_songs(&songs)?; if let Some((sid, _)) = songs.first() { if sl.selected_sid == Default::default() { - sl.selected_sid = sid.clone(); + sl.selected_sid = (*sid).clone(); } } } @@ -34,8 +34,8 @@ impl CompUi for SongList { .show(ui, |ui| { ui.vertical(|ui| { ui.add_space(3.0); - for (sid, song) in disp_songs { - handle_error_ui!(Self::display_song_tab(ui, state, &sid, &song)); + for sid in disp_songs { + handle_error_ui!(Self::display_song_tab(ui, state, &sid)); } }); }); @@ -44,40 +44,19 @@ impl CompUi for SongList { } impl SongList { - fn get_and_sort_songs(state: &mut crate::GuiState) -> crate::Result> { + fn get_and_sort_songs(state: &mut crate::GuiState) -> crate::Result> { let pid = super::left_nav::LeftNav::get()?.selected_playlist_id.clone(); match pid { None => { - let songs = state.manifest.store().get_songs().clone().into_iter(); - let mut songs: Vec<_> = songs.collect(); - songs.sort_by(|a, b| { - let a = a.1.name().to_lowercase(); - let b = b.1.name().to_lowercase(); - a.cmp(&b) - }); - Ok(songs) + Ok(state.manifest.store().get_songs_sorted()) } Some(pid) => { - let Some(playlist) = state.manifest.store().get_playlist(&pid) else { - anyhow::bail!("Couldnt find playlist (corruption?)"); - }; - let mut songs = Vec::new(); - for sid in playlist.songs() { - if let Some(song) = state.manifest.store().get_song(&sid) { - songs.push((sid.clone(), song.clone())); - } - } - songs.sort_by(|a, b| { - let a = a.1.name().to_lowercase(); - let b = b.1.name().to_lowercase(); - a.cmp(&b) - }); - Ok(songs) + Ok(state.manifest.store().get_playlist_songs_sorted(pid).expect("Invalid pid")) } } } - fn get_songs_to_display(songs: &Vec<(uuid::Uuid, Song)>) -> crate::Result>{ + fn get_songs_to_display<'a>(songs: &'a [(&uuid::Uuid, &Song)]) -> crate::Result>{ let mut to_display = Vec::new(); let query = {header::Header::get()?.search_text.clone()}; let (query_type, query_text) = crate::utils::SearchType::from_str(&query); @@ -105,15 +84,16 @@ impl SongList { }; if should_display { - to_display.push((sid.clone(), song.clone())); + to_display.push((*sid).clone()); } } Ok(to_display) } - fn display_song_tab(ui: &mut egui::Ui, state: &mut crate::GuiState, sid: &uuid::Uuid, song: &Song) -> crate::Result<()> { + fn display_song_tab(ui: &mut egui::Ui, state: &mut crate::GuiState, sid: &uuid::Uuid) -> crate::Result<()> { let mut clicked = false; ui.horizontal(|ui| { + let song = handle_option!("(internal)", state.manifest.store().get_song(sid)); let theme = handle_error_ui!(xmpd_settings::Settings::get()).theme.clone(); // let icon_status = handle_error_ui!(xmpd_cache::Cache::get()).get_cached_icon_status(&sid).clone(); let img = ui.add( @@ -200,7 +180,7 @@ impl SongList { .size(16.0); ui.add(spinner); } - Some(DlStatus::Error(e, ..)) => { + Some(DlStatus::Error(_, _, e)) => { let img = egui::Image::new(crate::data::WARN_ICON) .tint(Color32::LIGHT_YELLOW) .sense(Sense::hover()) @@ -274,12 +254,12 @@ impl SongList { self.play_song(next, state)?; Ok(()) } - fn get_playable_songs(songs: &Vec<(uuid::Uuid, Song)>) -> crate::Result> { + fn get_playable_songs(songs: &[(&uuid::Uuid, &Song)]) -> crate::Result> { let mut playable_songs = Vec::new(); for (sid, _) in songs { if let Some(DlStatus::Done(_)) = xmpd_cache::Cache::get()?.get_cached_song_status(&sid) { - playable_songs.push(sid.clone()); + playable_songs.push((*sid).clone()); } } Ok(playable_songs) diff --git a/xmpd-gui/src/components/top_nav.rs b/xmpd-gui/src/components/top_nav.rs index c2d7988..2a98429 100644 --- a/xmpd-gui/src/components/top_nav.rs +++ b/xmpd-gui/src/components/top_nav.rs @@ -1,6 +1,6 @@ use std::path::PathBuf; -use egui::TextBuffer; +use egui::{Layout, TextBuffer}; use xmpd_manifest::store::{JsonStore, TomlStore}; use crate::windows::WindowId; @@ -65,7 +65,10 @@ impl CompUi for TopNav { } }); - + #[cfg(debug_assertions)] + ui.with_layout(Layout::right_to_left(egui::Align::Max), |ui| { + ui.label(format!("ft: {} ms", state.debug_info.last_frame_time.as_millis())); + }); }); }); let mut used = false; diff --git a/xmpd-gui/src/lib.rs b/xmpd-gui/src/lib.rs index 8604124..79dfc74 100644 --- a/xmpd-gui/src/lib.rs +++ b/xmpd-gui/src/lib.rs @@ -1,6 +1,4 @@ -#![feature(async_closure)] - -use std::time::Duration; +use std::time::{Duration, Instant}; use xmpd_manifest::{store::JsonStore, Manifest}; #[macro_use] @@ -20,10 +18,19 @@ pub fn start() -> Result<()> { let options = eframe::NativeOptions::default(); let mut state = GuiState::new()?; let res = eframe::run_simple_native(W_NAME, options, move |ctx, _frame| { + #[cfg(debug_assertions)] + let f_start = Instant::now(); + egui_extras::install_image_loaders(ctx); - state.windows.clone().draw_all(ctx, &mut state); + windows::Windows::draw_all(ctx, &mut state); handle_error_ui!(main_window::draw(ctx, &mut state, &cache_rx)); ctx.request_repaint_after(Duration::from_millis(500)); + + #[cfg(debug_assertions)] + { + let f_end = Instant::now(); + state.debug_info.last_frame_time = f_end.duration_since(f_start); + } }); if let Err(e) = res { // dumb err value by eframe anyhow::bail!(e.to_string()); @@ -31,15 +38,34 @@ pub fn start() -> Result<()> { Ok(()) } - +#[cfg(debug_assertions)] +#[derive(Debug, Default)] +pub struct DebugInfo { + pub last_frame_time: Duration, +} pub struct GuiState { + #[cfg(debug_assertions)] + pub debug_info: DebugInfo, pub manifest: Manifest, pub windows: windows::Windows, pub player: xmpd_player::Player, } impl GuiState { + #[cfg(debug_assertions)] + pub fn new() -> Result { + Ok(Self { + debug_info: DebugInfo { + last_frame_time: Default::default() + }, + player: xmpd_player::Player::new(), + manifest: Manifest::new(&xmpd_cliargs::CLIARGS.manifest_path())?, + windows: windows::Windows::new(), + }) + } + + #[cfg(not(debug_assertions))] pub fn new() -> Result { Ok(Self { player: xmpd_player::Player::new(), diff --git a/xmpd-gui/src/main_window.rs b/xmpd-gui/src/main_window.rs index 885b0ce..ea4dc7e 100644 --- a/xmpd-gui/src/main_window.rs +++ b/xmpd-gui/src/main_window.rs @@ -30,13 +30,15 @@ pub fn draw(ctx: &egui::Context, state: &mut GuiState, cache_rx: &Receiver &'static str where Self: Sized { "Add Song to Playlist" } - fn draw(&mut self, ui: &mut egui::Ui, _: &mut crate::GuiState) -> crate::Result<()> { - ui.label("Hello from other window!"); + fn draw(&mut self, ui: &mut egui::Ui, state: &mut crate::GuiState) -> crate::Result<()> { + let theme = xmpd_settings::Settings::get()?.theme.clone(); + + let songs: Vec<_> = state.manifest.store().get_songs_sorted(); + if self.sid.is_nil() { + if let Some(sid) = songs.first() { + self.sid = sid.0.clone(); + } + } + let avail = ui.available_size(); + ui.horizontal(|ui| { + ui.vertical(|ui| { + ui.group(|ui| { + ui.set_height(avail.y); + let img = egui::Image::new(crate::data::NOTE_ICON) + .tint(theme.accent_color) + .fit_to_exact_size(egui::Vec2::new(64.0, 64.0)); + ui.add(img); + let mut name = String::new(); + let mut author = String::new(); + if let Some(song) = state.manifest.store().get_song(&self.sid) { + name = song.name().to_string(); + author = song.author().to_string(); + } + ui.horizontal(|ui| { + ui.style_mut().spacing.text_edit_width = 150.0; + ui.label("Name: "); + ui.add_enabled(false, TextEdit::singleline(&mut name)); + }); + ui.horizontal(|ui| { + ui.style_mut().spacing.text_edit_width = 150.0; + ui.label("Author: "); + ui.add_enabled(false, TextEdit::singleline(&mut author)); + }); + }); + }); + ui.vertical(|ui| { + ui.group(|ui| { + egui::ScrollArea::vertical() + .id_source("song_list_song_add") + .drag_to_scroll(false) + .show(ui, |ui| { + ui.vertical(|ui| { + for (sid, song) in songs { + ui.group(|ui| { + let avail = ui.available_size(); + ui.horizontal(|ui| { + ui.set_width(avail.x); + let img = egui::Image::new(crate::data::NOTE_ICON) + .tint(theme.accent_color) + .fit_to_exact_size(egui::Vec2::new(32.0, 32.0)); + ui.add(img); + ui.vertical(|ui| { + let status = { + handle_error_ui!(xmpd_cache::Cache::get()).get_cached_song_status(&sid).clone() + }; + let label = if self.sid == *sid { + RichText::new(song.name()) + .color(theme.accent_color) + } else if matches!(status, Some(DlStatus::Done(_))) { + RichText::new(song.name()) + .color(theme.text_color) + } else { + RichText::new(song.name()) + .color(theme.dim_text_color) + }; + ui.label(label); + }); + }); + }); + } + }); + } + ); + }); + }) + }); Ok(()) } } diff --git a/xmpd-gui/src/windows/mod.rs b/xmpd-gui/src/windows/mod.rs index 82a1a80..66b51ab 100644 --- a/xmpd-gui/src/windows/mod.rs +++ b/xmpd-gui/src/windows/mod.rs @@ -67,13 +67,13 @@ impl Windows { WINDOWS.lock().unwrap().insert(WT::id(), Box::::default()); } - pub fn draw_all(&mut self, ctx: &egui::Context, state: &mut GuiState) { + pub fn draw_all(ctx: &egui::Context, state: &mut GuiState) { let theme = handle_error_ui!(xmpd_settings::Settings::get()).theme.clone(); - for (win_id, (vp_id, builder)) in &self.windows { - if self.is_open(&win_id) { + for (win_id, (vp_id, builder)) in state.windows.windows.clone().into_iter() { + if state.windows.is_open(&win_id) { ctx.show_viewport_immediate(vp_id.clone(), builder.clone(), |ctx, _vp_class| { ctx.input(|inp| { - self.toggle(win_id, !inp.viewport().close_requested()); + state.windows.toggle(&win_id, !inp.viewport().close_requested()); }); egui::CentralPanel::default() .frame( diff --git a/xmpd-manifest/src/store/json.rs b/xmpd-manifest/src/store/json.rs index 8e879bf..1bd6322 100644 --- a/xmpd-manifest/src/store/json.rs +++ b/xmpd-manifest/src/store/json.rs @@ -15,6 +15,8 @@ pub struct JsonStore { playlists: HashMap } +impl super::StoreExtras for JsonStore {} + impl super::BaseStore for JsonStore { fn get_default_file_contents() -> &'static str { &DEFAULT_TEXT diff --git a/xmpd-manifest/src/store/mod.rs b/xmpd-manifest/src/store/mod.rs index a7fbe8c..d62e289 100644 --- a/xmpd-manifest/src/store/mod.rs +++ b/xmpd-manifest/src/store/mod.rs @@ -40,3 +40,55 @@ pub trait BaseStore { fn get_original_path(&self) -> &Path; } +pub trait StoreExtras: BaseStore { + fn get_songs_sorted(&self) -> Vec<(&Uuid, &Song)> { + let mut songs: Vec<_> = self.get_songs().iter().collect(); + songs.sort_by(|a, b| { + let a = a.1.name().to_lowercase(); + let b = b.1.name().to_lowercase(); + a.cmp(&b) + }); + songs + } + + fn get_songs_sorted_cloned(&self) -> Vec<(Uuid, Song)> { + let mut songs = Vec::new(); + for song in self.get_songs_sorted() { + songs.push((song.0.clone(), song.1.clone())); + } + songs + } + + fn get_playlists_sorted(&self) -> Vec<(&Uuid, &Playlist)> { + let mut songs: Vec<_> = self.get_playlists().iter().collect(); + songs.sort_by(|a, b| { + let a = a.1.name().to_lowercase(); + let b = b.1.name().to_lowercase(); + a.cmp(&b) + }); + songs + } + fn get_playlist_songs_sorted(&self, pid: uuid::Uuid) -> Option> { + let songs_ids: Vec<_> = self.get_playlist(&pid)?.songs().iter().collect(); + let mut songs = Vec::new(); + + for sid in songs_ids { + songs.push((sid, self.get_song(sid)?)); + } + + songs.sort_by(|a, b| { + let a = a.1.name().to_lowercase(); + let b = b.1.name().to_lowercase(); + a.cmp(&b) + }); + Some(songs) + } + fn get_playlist_songs_sorted_cloned(&self, pid: uuid::Uuid) -> Option> { + let mut songs = Vec::new(); + for song in self.get_playlist_songs_sorted(pid)? { + songs.push((song.0.clone(), song.1.clone())); + } + Some(songs) + } +} + diff --git a/xmpd-manifest/src/store/toml.rs b/xmpd-manifest/src/store/toml.rs index be4ec0b..95d82a2 100644 --- a/xmpd-manifest/src/store/toml.rs +++ b/xmpd-manifest/src/store/toml.rs @@ -15,6 +15,8 @@ pub struct TomlStore { playlists: HashMap } +impl super::StoreExtras for TomlStore {} + impl super::BaseStore for TomlStore { fn get_default_file_contents() -> &'static str { &DEFAULT_TEXT