2023-09-20 16:02:07 +02:00
|
|
|
use std::{
|
2023-09-22 21:25:43 +02:00
|
|
|
cmp::Ordering,
|
2023-10-04 16:16:05 +02:00
|
|
|
collections::HashSet,
|
2023-10-24 22:50:21 +02:00
|
|
|
sync::Arc,
|
2023-09-28 15:37:10 +02:00
|
|
|
sync::{
|
|
|
|
atomic::{AtomicBool, AtomicUsize},
|
2023-10-04 16:16:05 +02:00
|
|
|
mpsc, Mutex,
|
2023-09-28 15:37:10 +02:00
|
|
|
},
|
2023-09-20 16:02:07 +02:00
|
|
|
};
|
|
|
|
|
2023-09-22 21:25:43 +02:00
|
|
|
use musicdb_lib::data::{
|
2023-10-04 16:16:05 +02:00
|
|
|
album::Album,
|
|
|
|
artist::Artist,
|
|
|
|
database::Database,
|
|
|
|
queue::{Queue, QueueContent},
|
|
|
|
song::Song,
|
|
|
|
AlbumId, ArtistId, GeneralData, SongId,
|
2023-09-22 21:25:43 +02:00
|
|
|
};
|
2023-08-13 23:58:53 +02:00
|
|
|
use regex::{Regex, RegexBuilder};
|
|
|
|
use speedy2d::{
|
|
|
|
color::Color,
|
|
|
|
dimen::Vec2,
|
|
|
|
shape::Rectangle,
|
|
|
|
window::{MouseButton, VirtualKeyCode},
|
|
|
|
};
|
|
|
|
|
|
|
|
use crate::{
|
|
|
|
gui::{Dragging, DrawInfo, GuiAction, GuiElem, GuiElemCfg, GuiElemTrait},
|
2023-10-04 16:16:05 +02:00
|
|
|
gui_base::{Button, Panel, ScrollBox},
|
2023-08-13 23:58:53 +02:00
|
|
|
gui_text::{Label, TextField},
|
|
|
|
gui_wrappers::WithFocusHotkey,
|
|
|
|
};
|
|
|
|
|
2023-10-26 16:18:52 +02:00
|
|
|
use self::selected::Selected;
|
|
|
|
|
2023-08-14 00:58:31 +02:00
|
|
|
/*
|
|
|
|
|
|
|
|
This is responsible for showing the library,
|
|
|
|
with Regex search and drag-n-drop.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2023-08-13 23:58:53 +02:00
|
|
|
pub struct LibraryBrowser {
|
|
|
|
config: GuiElemCfg,
|
|
|
|
pub children: Vec<GuiElem>,
|
2023-09-22 21:25:43 +02:00
|
|
|
// - - -
|
|
|
|
library_sorted: Vec<(ArtistId, Vec<SongId>, Vec<(AlbumId, Vec<SongId>)>)>,
|
|
|
|
library_filtered: Vec<(
|
|
|
|
ArtistId,
|
|
|
|
Vec<(SongId, f32)>,
|
|
|
|
Vec<(AlbumId, Vec<(SongId, f32)>, f32)>,
|
|
|
|
f32,
|
|
|
|
)>,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected: Selected,
|
2023-09-22 21:25:43 +02:00
|
|
|
// - - -
|
2023-08-13 23:58:53 +02:00
|
|
|
search_artist: String,
|
|
|
|
search_artist_regex: Option<Regex>,
|
|
|
|
search_album: String,
|
|
|
|
search_album_regex: Option<Regex>,
|
|
|
|
search_song: String,
|
|
|
|
search_song_regex: Option<Regex>,
|
2023-10-24 22:50:21 +02:00
|
|
|
filter_target_state: Arc<AtomicBool>,
|
2023-09-20 16:02:07 +02:00
|
|
|
filter_state: f32,
|
2023-09-22 21:25:43 +02:00
|
|
|
library_updated: bool,
|
2023-10-24 22:50:21 +02:00
|
|
|
search_settings_changed: Arc<AtomicBool>,
|
|
|
|
search_is_case_sensitive: Arc<AtomicBool>,
|
2023-09-20 16:02:07 +02:00
|
|
|
search_was_case_sensitive: bool,
|
2023-10-24 22:50:21 +02:00
|
|
|
search_prefer_start_matches: Arc<AtomicBool>,
|
2023-09-22 21:25:43 +02:00
|
|
|
search_prefers_start_matches: bool,
|
2023-10-24 22:50:21 +02:00
|
|
|
filter_songs: Arc<Mutex<Filter>>,
|
|
|
|
filter_albums: Arc<Mutex<Filter>>,
|
|
|
|
filter_artists: Arc<Mutex<Filter>>,
|
2023-10-04 16:16:05 +02:00
|
|
|
do_something_receiver: mpsc::Receiver<Box<dyn FnOnce(&mut Self)>>,
|
|
|
|
}
|
|
|
|
impl Clone for LibraryBrowser {
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
Self::new(self.config.clone())
|
|
|
|
}
|
|
|
|
}
|
2023-10-26 16:18:52 +02:00
|
|
|
mod selected {
|
|
|
|
use super::*;
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct Selected(
|
|
|
|
// artist, album, songs
|
|
|
|
Arc<Mutex<(HashSet<ArtistId>, HashSet<AlbumId>, HashSet<SongId>)>>,
|
|
|
|
Arc<AtomicBool>,
|
|
|
|
);
|
|
|
|
impl Selected {
|
|
|
|
pub fn new(update: Arc<AtomicBool>) -> Self {
|
|
|
|
Self(Default::default(), update)
|
|
|
|
}
|
|
|
|
pub fn clear(&self) {
|
|
|
|
self.set_to(HashSet::new(), HashSet::new(), HashSet::new())
|
|
|
|
}
|
|
|
|
pub fn set_to(&self, artists: HashSet<u64>, albums: HashSet<u64>, songs: HashSet<u64>) {
|
|
|
|
let mut s = self.0.lock().unwrap();
|
|
|
|
s.0 = artists;
|
|
|
|
s.1 = albums;
|
|
|
|
s.2 = songs;
|
|
|
|
self.changed();
|
|
|
|
}
|
|
|
|
pub fn contains_artist(&self, id: &ArtistId) -> bool {
|
|
|
|
self.0.lock().unwrap().0.contains(id)
|
|
|
|
}
|
|
|
|
pub fn contains_album(&self, id: &AlbumId) -> bool {
|
|
|
|
self.0.lock().unwrap().1.contains(id)
|
|
|
|
}
|
|
|
|
pub fn contains_song(&self, id: &SongId) -> bool {
|
|
|
|
self.0.lock().unwrap().2.contains(id)
|
|
|
|
}
|
|
|
|
pub fn insert_artist(&self, id: ArtistId) -> bool {
|
|
|
|
self.changed();
|
|
|
|
self.0.lock().unwrap().0.insert(id)
|
|
|
|
}
|
|
|
|
pub fn insert_album(&self, id: AlbumId) -> bool {
|
|
|
|
self.changed();
|
|
|
|
self.0.lock().unwrap().1.insert(id)
|
|
|
|
}
|
|
|
|
pub fn insert_song(&self, id: SongId) -> bool {
|
|
|
|
self.changed();
|
|
|
|
self.0.lock().unwrap().2.insert(id)
|
|
|
|
}
|
|
|
|
pub fn remove_artist(&self, id: &ArtistId) -> bool {
|
|
|
|
self.changed();
|
|
|
|
self.0.lock().unwrap().0.remove(id)
|
|
|
|
}
|
|
|
|
pub fn remove_album(&self, id: &AlbumId) -> bool {
|
|
|
|
self.changed();
|
|
|
|
self.0.lock().unwrap().1.remove(id)
|
|
|
|
}
|
|
|
|
pub fn remove_song(&self, id: &SongId) -> bool {
|
|
|
|
self.changed();
|
|
|
|
self.0.lock().unwrap().2.remove(id)
|
|
|
|
}
|
|
|
|
pub fn view<T>(
|
|
|
|
&self,
|
|
|
|
f: impl FnOnce(&(HashSet<ArtistId>, HashSet<AlbumId>, HashSet<SongId>)) -> T,
|
|
|
|
) -> T {
|
|
|
|
f(&self.0.lock().unwrap())
|
|
|
|
}
|
|
|
|
pub fn view_mut<T>(
|
|
|
|
&self,
|
|
|
|
f: impl FnOnce(&mut (HashSet<ArtistId>, HashSet<AlbumId>, HashSet<SongId>)) -> T,
|
|
|
|
) -> T {
|
|
|
|
let v = f(&mut self.0.lock().unwrap());
|
|
|
|
self.changed();
|
|
|
|
v
|
|
|
|
}
|
|
|
|
fn changed(&self) {
|
|
|
|
self.1.store(true, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
}
|
|
|
|
pub fn as_queue(&self, lb: &LibraryBrowser, db: &Database) -> Vec<Queue> {
|
|
|
|
let lock = self.0.lock().unwrap();
|
|
|
|
let (sel_artists, sel_albums, sel_songs) = &*lock;
|
|
|
|
let mut out = vec![];
|
|
|
|
for (artist, singles, albums, _) in &lb.library_filtered {
|
|
|
|
let artist_selected = sel_artists.contains(artist);
|
|
|
|
let mut local_artist_owned = vec![];
|
|
|
|
let mut local_artist = if artist_selected {
|
|
|
|
&mut local_artist_owned
|
2023-10-04 16:16:05 +02:00
|
|
|
} else {
|
2023-10-26 16:18:52 +02:00
|
|
|
&mut out
|
2023-10-04 16:16:05 +02:00
|
|
|
};
|
2023-10-26 16:18:52 +02:00
|
|
|
for (song, _) in singles {
|
2023-10-04 16:16:05 +02:00
|
|
|
let song_selected = sel_songs.contains(song);
|
|
|
|
if song_selected {
|
2023-10-26 16:18:52 +02:00
|
|
|
local_artist.push(QueueContent::Song(*song).into());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (album, songs, _) in albums {
|
|
|
|
let album_selected = sel_albums.contains(album);
|
|
|
|
let mut local_album_owned = vec![];
|
|
|
|
let local_album = if album_selected {
|
|
|
|
&mut local_album_owned
|
|
|
|
} else {
|
|
|
|
&mut local_artist
|
|
|
|
};
|
|
|
|
for (song, _) in songs {
|
|
|
|
let song_selected = sel_songs.contains(song);
|
|
|
|
if song_selected {
|
|
|
|
local_album.push(QueueContent::Song(*song).into());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if album_selected {
|
|
|
|
local_artist.push(
|
|
|
|
QueueContent::Folder(
|
|
|
|
0,
|
|
|
|
local_album_owned,
|
|
|
|
match db.albums().get(album) {
|
|
|
|
Some(v) => v.name.clone(),
|
|
|
|
None => "< unknown album >".to_owned(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.into(),
|
|
|
|
);
|
2023-10-04 16:16:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-10-26 16:18:52 +02:00
|
|
|
if artist_selected {
|
|
|
|
out.push(
|
2023-10-04 16:16:05 +02:00
|
|
|
QueueContent::Folder(
|
|
|
|
0,
|
2023-10-26 16:18:52 +02:00
|
|
|
local_artist_owned,
|
|
|
|
match db.artists().get(artist) {
|
|
|
|
Some(v) => v.name.to_owned(),
|
|
|
|
None => "< unknown artist >".to_owned(),
|
2023-10-04 16:16:05 +02:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.into(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2023-10-26 16:18:52 +02:00
|
|
|
out
|
2023-10-04 16:16:05 +02:00
|
|
|
}
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
2023-09-22 21:25:43 +02:00
|
|
|
fn search_regex_new(pat: &str, case_insensitive: bool) -> Result<Option<Regex>, regex::Error> {
|
|
|
|
if pat.is_empty() {
|
|
|
|
Ok(None)
|
|
|
|
} else {
|
|
|
|
Ok(Some(
|
|
|
|
RegexBuilder::new(pat)
|
|
|
|
.unicode(true)
|
|
|
|
.case_insensitive(case_insensitive)
|
|
|
|
.build()?,
|
|
|
|
))
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
2023-09-20 16:02:07 +02:00
|
|
|
const LP_LIB1: f32 = 0.1;
|
|
|
|
const LP_LIB2: f32 = 1.0;
|
|
|
|
const LP_LIB1S: f32 = 0.4;
|
2023-08-13 23:58:53 +02:00
|
|
|
impl LibraryBrowser {
|
|
|
|
pub fn new(config: GuiElemCfg) -> Self {
|
|
|
|
let search_artist = TextField::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.01, 0.01), (0.45, 0.05))),
|
|
|
|
"artist".to_string(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
);
|
|
|
|
let search_album = TextField::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.55, 0.01), (0.99, 0.05))),
|
|
|
|
"album".to_string(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
);
|
|
|
|
let search_song = WithFocusHotkey::new_ctrl(
|
|
|
|
VirtualKeyCode::F,
|
|
|
|
TextField::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.01, 0.06), (0.99, 0.1))),
|
|
|
|
"song".to_string(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
let library_scroll_box = ScrollBox::new(
|
2023-09-20 16:02:07 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, LP_LIB1), (1.0, LP_LIB2))),
|
2023-08-13 23:58:53 +02:00
|
|
|
crate::gui_base::ScrollBoxSizeUnit::Pixels,
|
|
|
|
vec![],
|
|
|
|
);
|
2023-10-04 16:16:05 +02:00
|
|
|
let (do_something_sender, do_something_receiver) = mpsc::channel();
|
2023-10-24 22:50:21 +02:00
|
|
|
let search_settings_changed = Arc::new(AtomicBool::new(false));
|
2023-09-22 21:25:43 +02:00
|
|
|
let search_was_case_sensitive = false;
|
2023-10-24 22:50:21 +02:00
|
|
|
let search_is_case_sensitive = Arc::new(AtomicBool::new(search_was_case_sensitive));
|
2023-09-22 21:25:43 +02:00
|
|
|
let search_prefers_start_matches = true;
|
2023-10-24 22:50:21 +02:00
|
|
|
let search_prefer_start_matches = Arc::new(AtomicBool::new(search_prefers_start_matches));
|
|
|
|
let filter_target_state = Arc::new(AtomicBool::new(false));
|
|
|
|
let fts = Arc::clone(&filter_target_state);
|
2023-09-20 16:02:07 +02:00
|
|
|
let filter_button = Button::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.46, 0.01), (0.54, 0.05))),
|
|
|
|
move |_| {
|
|
|
|
fts.store(
|
|
|
|
!fts.load(std::sync::atomic::Ordering::Relaxed),
|
|
|
|
std::sync::atomic::Ordering::Relaxed,
|
|
|
|
);
|
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"more".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
);
|
2023-10-24 22:50:21 +02:00
|
|
|
let filter_songs = Arc::new(Mutex::new(Filter {
|
2023-09-28 15:37:10 +02:00
|
|
|
and: true,
|
|
|
|
filters: vec![],
|
|
|
|
}));
|
2023-10-24 22:50:21 +02:00
|
|
|
let filter_albums = Arc::new(Mutex::new(Filter {
|
2023-09-28 15:37:10 +02:00
|
|
|
and: true,
|
|
|
|
filters: vec![],
|
|
|
|
}));
|
2023-10-24 22:50:21 +02:00
|
|
|
let filter_artists = Arc::new(Mutex::new(Filter {
|
2023-09-28 15:37:10 +02:00
|
|
|
and: true,
|
|
|
|
filters: vec![],
|
|
|
|
}));
|
2023-10-26 16:18:52 +02:00
|
|
|
let selected = Selected::new(Arc::clone(&search_settings_changed));
|
2023-08-13 23:58:53 +02:00
|
|
|
Self {
|
2023-10-26 16:18:52 +02:00
|
|
|
config: config.w_keyboard_watch(),
|
2023-08-13 23:58:53 +02:00
|
|
|
children: vec![
|
|
|
|
GuiElem::new(search_artist),
|
|
|
|
GuiElem::new(search_album),
|
|
|
|
GuiElem::new(search_song),
|
|
|
|
GuiElem::new(library_scroll_box),
|
2023-09-20 16:02:07 +02:00
|
|
|
GuiElem::new(filter_button),
|
2023-09-22 21:25:43 +02:00
|
|
|
GuiElem::new(FilterPanel::new(
|
2023-10-24 22:50:21 +02:00
|
|
|
Arc::clone(&search_settings_changed),
|
|
|
|
Arc::clone(&search_is_case_sensitive),
|
|
|
|
Arc::clone(&search_prefer_start_matches),
|
|
|
|
Arc::clone(&filter_songs),
|
|
|
|
Arc::clone(&filter_albums),
|
|
|
|
Arc::clone(&filter_artists),
|
2023-10-04 16:16:05 +02:00
|
|
|
selected.clone(),
|
|
|
|
do_something_sender.clone(),
|
2023-09-22 21:25:43 +02:00
|
|
|
)),
|
2023-08-13 23:58:53 +02:00
|
|
|
],
|
2023-09-22 21:25:43 +02:00
|
|
|
// - - -
|
|
|
|
library_sorted: vec![],
|
|
|
|
library_filtered: vec![],
|
2023-10-04 16:16:05 +02:00
|
|
|
selected,
|
2023-09-22 21:25:43 +02:00
|
|
|
// - - -
|
2023-08-13 23:58:53 +02:00
|
|
|
search_artist: String::new(),
|
|
|
|
search_artist_regex: None,
|
|
|
|
search_album: String::new(),
|
|
|
|
search_album_regex: None,
|
|
|
|
search_song: String::new(),
|
|
|
|
search_song_regex: None,
|
2023-09-20 16:02:07 +02:00
|
|
|
filter_target_state,
|
|
|
|
filter_state: 0.0,
|
2023-09-22 21:25:43 +02:00
|
|
|
library_updated: true,
|
|
|
|
search_settings_changed,
|
2023-09-20 16:02:07 +02:00
|
|
|
search_is_case_sensitive,
|
2023-09-22 21:25:43 +02:00
|
|
|
search_was_case_sensitive,
|
|
|
|
search_prefer_start_matches,
|
|
|
|
search_prefers_start_matches,
|
2023-09-28 15:37:10 +02:00
|
|
|
filter_songs,
|
|
|
|
filter_albums,
|
|
|
|
filter_artists,
|
2023-10-04 16:16:05 +02:00
|
|
|
do_something_receiver,
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
2023-10-26 16:18:52 +02:00
|
|
|
pub fn selected_add_all(&self) {
|
|
|
|
self.selected.view_mut(|sel| {
|
|
|
|
for (id, singles, albums, _) in &self.library_filtered {
|
|
|
|
sel.0.insert(*id);
|
|
|
|
for (s, _) in singles {
|
|
|
|
sel.2.insert(*s);
|
|
|
|
}
|
|
|
|
for (id, album, _) in albums {
|
|
|
|
sel.1.insert(*id);
|
|
|
|
for (s, _) in album {
|
|
|
|
sel.2.insert(*s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
pub fn selected_add_songs(&self) {
|
|
|
|
self.selected.view_mut(|sel| {
|
|
|
|
for (_, singles, albums, _) in &self.library_filtered {
|
|
|
|
for (s, _) in singles {
|
|
|
|
sel.2.insert(*s);
|
|
|
|
}
|
|
|
|
for (_, album, _) in albums {
|
|
|
|
for (s, _) in album {
|
|
|
|
sel.2.insert(*s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
pub fn selected_add_albums(&self) {
|
|
|
|
self.selected.view_mut(|sel| {
|
|
|
|
for (_, _, albums, _) in &self.library_filtered {
|
|
|
|
for (id, album, _) in albums {
|
|
|
|
sel.1.insert(*id);
|
|
|
|
for (s, _) in album {
|
|
|
|
sel.2.insert(*s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
impl GuiElemTrait for LibraryBrowser {
|
|
|
|
fn config(&self) -> &GuiElemCfg {
|
|
|
|
&self.config
|
|
|
|
}
|
|
|
|
fn config_mut(&mut self) -> &mut GuiElemCfg {
|
|
|
|
&mut self.config
|
|
|
|
}
|
|
|
|
fn children(&mut self) -> Box<dyn Iterator<Item = &mut GuiElem> + '_> {
|
|
|
|
Box::new(self.children.iter_mut())
|
|
|
|
}
|
|
|
|
fn any(&self) -> &dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn any_mut(&mut self) -> &mut dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn clone_gui(&self) -> Box<dyn GuiElemTrait> {
|
|
|
|
Box::new(self.clone())
|
|
|
|
}
|
|
|
|
fn draw(&mut self, info: &mut DrawInfo, _g: &mut speedy2d::Graphics2D) {
|
2023-10-04 16:16:05 +02:00
|
|
|
loop {
|
|
|
|
if let Ok(action) = self.do_something_receiver.try_recv() {
|
|
|
|
action(self);
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-09-20 16:02:07 +02:00
|
|
|
// search
|
2023-08-13 23:58:53 +02:00
|
|
|
let mut search_changed = false;
|
2023-09-20 16:02:07 +02:00
|
|
|
let mut rebuild_regex = false;
|
2023-09-22 21:25:43 +02:00
|
|
|
if self
|
|
|
|
.search_settings_changed
|
|
|
|
.load(std::sync::atomic::Ordering::Relaxed)
|
|
|
|
{
|
|
|
|
search_changed = true;
|
|
|
|
self.search_settings_changed
|
|
|
|
.store(false, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
let case_sensitive = self
|
|
|
|
.search_is_case_sensitive
|
|
|
|
.load(std::sync::atomic::Ordering::Relaxed);
|
|
|
|
if self.search_was_case_sensitive != case_sensitive {
|
|
|
|
self.search_was_case_sensitive = case_sensitive;
|
|
|
|
rebuild_regex = true;
|
|
|
|
}
|
|
|
|
let pref_start = self
|
|
|
|
.search_prefer_start_matches
|
|
|
|
.load(std::sync::atomic::Ordering::Relaxed);
|
|
|
|
if self.search_prefers_start_matches != pref_start {
|
|
|
|
self.search_prefers_start_matches = pref_start;
|
|
|
|
}
|
2023-09-20 16:02:07 +02:00
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
{
|
|
|
|
let v = &mut self.children[0].try_as_mut::<TextField>().unwrap().children[0]
|
|
|
|
.try_as_mut::<Label>()
|
|
|
|
.unwrap()
|
|
|
|
.content;
|
2023-09-20 16:02:07 +02:00
|
|
|
if rebuild_regex || v.will_redraw() && self.search_artist != *v.get_text() {
|
2023-08-13 23:58:53 +02:00
|
|
|
search_changed = true;
|
|
|
|
self.search_artist = v.get_text().clone();
|
2023-09-20 16:02:07 +02:00
|
|
|
self.search_artist_regex =
|
2023-09-22 21:25:43 +02:00
|
|
|
search_regex_new(&self.search_artist, !self.search_was_case_sensitive)
|
|
|
|
.ok()
|
|
|
|
.flatten();
|
2023-08-13 23:58:53 +02:00
|
|
|
*v.color() = if self.search_artist_regex.is_some() {
|
|
|
|
Color::WHITE
|
|
|
|
} else {
|
|
|
|
Color::RED
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
|
|
|
let v = &mut self.children[1].try_as_mut::<TextField>().unwrap().children[0]
|
|
|
|
.try_as_mut::<Label>()
|
|
|
|
.unwrap()
|
|
|
|
.content;
|
2023-09-20 16:02:07 +02:00
|
|
|
if rebuild_regex || v.will_redraw() && self.search_album != *v.get_text() {
|
2023-08-13 23:58:53 +02:00
|
|
|
search_changed = true;
|
|
|
|
self.search_album = v.get_text().clone();
|
2023-09-20 16:02:07 +02:00
|
|
|
self.search_album_regex =
|
2023-09-22 21:25:43 +02:00
|
|
|
search_regex_new(&self.search_album, !self.search_was_case_sensitive)
|
|
|
|
.ok()
|
|
|
|
.flatten();
|
2023-08-13 23:58:53 +02:00
|
|
|
*v.color() = if self.search_album_regex.is_some() {
|
|
|
|
Color::WHITE
|
|
|
|
} else {
|
|
|
|
Color::RED
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
|
|
|
let v = &mut self.children[2]
|
|
|
|
.try_as_mut::<WithFocusHotkey<TextField>>()
|
|
|
|
.unwrap()
|
|
|
|
.inner
|
|
|
|
.children[0]
|
|
|
|
.try_as_mut::<Label>()
|
|
|
|
.unwrap()
|
|
|
|
.content;
|
2023-09-20 16:02:07 +02:00
|
|
|
if rebuild_regex || v.will_redraw() && self.search_song != *v.get_text() {
|
2023-08-13 23:58:53 +02:00
|
|
|
search_changed = true;
|
|
|
|
self.search_song = v.get_text().clone();
|
2023-09-22 21:25:43 +02:00
|
|
|
self.search_song_regex =
|
|
|
|
search_regex_new(&self.search_song, !self.search_was_case_sensitive)
|
|
|
|
.ok()
|
|
|
|
.flatten();
|
2023-08-13 23:58:53 +02:00
|
|
|
*v.color() = if self.search_song_regex.is_some() {
|
|
|
|
Color::WHITE
|
|
|
|
} else {
|
|
|
|
Color::RED
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
2023-09-22 21:25:43 +02:00
|
|
|
// filter panel
|
2023-09-20 16:02:07 +02:00
|
|
|
let filter_target_state = self
|
|
|
|
.filter_target_state
|
|
|
|
.load(std::sync::atomic::Ordering::Relaxed);
|
|
|
|
let draw_filter = if filter_target_state && self.filter_state != 1.0 {
|
|
|
|
if let Some(h) = &info.helper {
|
|
|
|
h.request_redraw();
|
|
|
|
}
|
|
|
|
self.filter_state += (1.0 - self.filter_state) * 0.2;
|
|
|
|
if self.filter_state > 0.999 {
|
|
|
|
self.filter_state = 1.0;
|
|
|
|
}
|
|
|
|
true
|
|
|
|
} else if !filter_target_state && self.filter_state != 0.0 {
|
|
|
|
if let Some(h) = &info.helper {
|
|
|
|
h.request_redraw();
|
|
|
|
}
|
|
|
|
self.filter_state *= 0.8;
|
|
|
|
if self.filter_state < 0.001 {
|
|
|
|
self.filter_state = 0.0;
|
|
|
|
}
|
|
|
|
true
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
};
|
|
|
|
if draw_filter {
|
|
|
|
let y = LP_LIB1 + (LP_LIB1S - LP_LIB1) * self.filter_state;
|
|
|
|
self.children[3]
|
|
|
|
.try_as_mut::<ScrollBox>()
|
|
|
|
.unwrap()
|
|
|
|
.config_mut()
|
|
|
|
.pos = Rectangle::new(Vec2::new(0.0, y), Vec2::new(1.0, LP_LIB2));
|
|
|
|
let filter_panel = self.children[5].try_as_mut::<FilterPanel>().unwrap();
|
|
|
|
filter_panel.config_mut().pos =
|
|
|
|
Rectangle::new(Vec2::new(0.0, LP_LIB1), Vec2::new(1.0, y));
|
|
|
|
filter_panel.config.enabled = self.filter_state > 0.0;
|
|
|
|
}
|
|
|
|
// -
|
2023-09-22 21:25:43 +02:00
|
|
|
if self.library_updated {
|
|
|
|
self.library_updated = false;
|
|
|
|
self.update_local_library(&info.database, |(_, a), (_, b)| a.name.cmp(&b.name));
|
|
|
|
search_changed = true;
|
|
|
|
}
|
|
|
|
if search_changed {
|
|
|
|
fn filter(
|
|
|
|
s: &LibraryBrowser,
|
|
|
|
pat: &str,
|
|
|
|
regex: &Option<Regex>,
|
|
|
|
search_text: &String,
|
2023-09-28 15:37:10 +02:00
|
|
|
filter: &Filter,
|
|
|
|
search_gd: &GeneralData,
|
2023-09-22 21:25:43 +02:00
|
|
|
) -> f32 {
|
2023-09-28 15:37:10 +02:00
|
|
|
if !filter.passes(search_gd) {
|
|
|
|
return 0.0;
|
|
|
|
};
|
2023-09-22 21:25:43 +02:00
|
|
|
if let Some(r) = regex {
|
|
|
|
if s.search_prefers_start_matches {
|
|
|
|
r.find_iter(pat)
|
|
|
|
.map(|m| match pat[0..m.start()].chars().rev().next() {
|
|
|
|
// found at the start of h, reaches to the end (whole pattern is part of the match)
|
2023-09-25 17:54:32 +02:00
|
|
|
None if m.end() == pat.len() => 6.0,
|
2023-09-22 21:25:43 +02:00
|
|
|
// found at start of h
|
|
|
|
None => 4.0,
|
2023-09-28 15:37:10 +02:00
|
|
|
Some(ch) if ch.is_whitespace() => {
|
|
|
|
match pat[m.end()..].chars().next() {
|
|
|
|
// whole word matches
|
|
|
|
None => 5.0,
|
|
|
|
Some(ch) if ch.is_whitespace() => 5.0,
|
|
|
|
// found after whitespace in h
|
|
|
|
Some(_) => 3.0,
|
|
|
|
}
|
|
|
|
}
|
2023-09-22 21:25:43 +02:00
|
|
|
// found somewhere else in h
|
|
|
|
_ => 2.0,
|
|
|
|
})
|
|
|
|
.fold(0.0, f32::max)
|
|
|
|
} else {
|
|
|
|
if r.is_match(pat) {
|
|
|
|
2.0
|
|
|
|
} else {
|
|
|
|
0.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if search_text.is_empty() {
|
|
|
|
1.0
|
|
|
|
} else {
|
|
|
|
0.0
|
|
|
|
}
|
|
|
|
}
|
2023-09-28 15:37:10 +02:00
|
|
|
let allow_singles = self.search_album.is_empty()
|
|
|
|
&& self.filter_albums.lock().unwrap().filters.is_empty();
|
2023-09-22 21:25:43 +02:00
|
|
|
self.filter_local_library(
|
|
|
|
&info.database,
|
2023-09-28 15:37:10 +02:00
|
|
|
|s, artist| {
|
|
|
|
filter(
|
|
|
|
s,
|
|
|
|
&artist.name,
|
|
|
|
&s.search_artist_regex,
|
|
|
|
&s.search_artist,
|
|
|
|
&s.filter_artists.lock().unwrap(),
|
|
|
|
&artist.general,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
|s, album| {
|
|
|
|
filter(
|
|
|
|
s,
|
|
|
|
&album.name,
|
|
|
|
&s.search_album_regex,
|
|
|
|
&s.search_album,
|
|
|
|
&s.filter_albums.lock().unwrap(),
|
|
|
|
&album.general,
|
|
|
|
)
|
|
|
|
},
|
2023-09-22 21:25:43 +02:00
|
|
|
|s, song| {
|
2023-09-28 15:37:10 +02:00
|
|
|
if song.album.is_some() || allow_singles {
|
|
|
|
filter(
|
|
|
|
s,
|
|
|
|
&song.title,
|
|
|
|
&s.search_song_regex,
|
|
|
|
&s.search_song,
|
|
|
|
&s.filter_songs.lock().unwrap(),
|
|
|
|
&song.general,
|
|
|
|
)
|
2023-09-22 21:25:43 +02:00
|
|
|
} else {
|
|
|
|
0.0
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
self.config.redraw = true;
|
|
|
|
}
|
|
|
|
if self.config.redraw || info.pos.size() != self.config.pixel_pos.size() {
|
2023-08-13 23:58:53 +02:00
|
|
|
self.config.redraw = false;
|
2023-09-22 21:25:43 +02:00
|
|
|
self.update_ui(&info.database, info.line_height);
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fn updated_library(&mut self) {
|
2023-09-22 21:25:43 +02:00
|
|
|
self.library_updated = true;
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
2023-10-26 16:18:52 +02:00
|
|
|
fn key_watch(
|
|
|
|
&mut self,
|
|
|
|
modifiers: speedy2d::window::ModifiersState,
|
|
|
|
down: bool,
|
|
|
|
key: Option<VirtualKeyCode>,
|
|
|
|
scan: speedy2d::window::KeyScancode,
|
|
|
|
) -> Vec<GuiAction> {
|
|
|
|
if down && crate::gui::hotkey_deselect_all(&modifiers, key) {
|
|
|
|
self.selected.clear();
|
|
|
|
}
|
|
|
|
if down && crate::gui::hotkey_select_all(&modifiers, key) {
|
|
|
|
self.selected_add_all();
|
|
|
|
}
|
|
|
|
if down && crate::gui::hotkey_select_albums(&modifiers, key) {
|
|
|
|
self.selected_add_albums();
|
|
|
|
}
|
|
|
|
if down && crate::gui::hotkey_select_songs(&modifiers, key) {
|
|
|
|
self.selected_add_songs();
|
|
|
|
}
|
|
|
|
vec![]
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
impl LibraryBrowser {
|
2023-09-22 21:25:43 +02:00
|
|
|
/// Sets `self.library_sorted` based on the contents of the `Database`.
|
|
|
|
fn update_local_library(
|
|
|
|
&mut self,
|
|
|
|
db: &Database,
|
|
|
|
sort_artists: impl FnMut(&(&ArtistId, &Artist), &(&ArtistId, &Artist)) -> Ordering,
|
|
|
|
) {
|
2023-08-13 23:58:53 +02:00
|
|
|
let mut artists = db.artists().iter().collect::<Vec<_>>();
|
2023-09-22 21:25:43 +02:00
|
|
|
artists.sort_unstable_by(sort_artists);
|
|
|
|
self.library_sorted = artists
|
|
|
|
.into_iter()
|
|
|
|
.map(|(ar_id, artist)| {
|
|
|
|
let singles = artist.singles.iter().map(|id| *id).collect();
|
|
|
|
let albums = artist
|
|
|
|
.albums
|
|
|
|
.iter()
|
|
|
|
.map(|id| {
|
|
|
|
let songs = if let Some(album) = db.albums().get(id) {
|
|
|
|
album.songs.iter().map(|id| *id).collect()
|
|
|
|
} else {
|
|
|
|
eprintln!("[warn] No album with id {id} found in db!");
|
|
|
|
vec![]
|
|
|
|
};
|
|
|
|
(*id, songs)
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
(*ar_id, singles, albums)
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
}
|
|
|
|
/// Sets `self.library_filtered` using the value of `self.library_sorted` and filter functions.
|
|
|
|
/// Return values of the filter functions:
|
|
|
|
/// 0.0 -> don't show
|
|
|
|
/// 1.0 -> neutral
|
|
|
|
/// anything else -> priority (determines how things will be sorted)
|
|
|
|
/// Album Value = max(Song Values) * AlbumFilterVal
|
|
|
|
/// Artist Value = max(Album Values) * ArtistFilterVal
|
|
|
|
fn filter_local_library(
|
|
|
|
&mut self,
|
|
|
|
db: &Database,
|
|
|
|
filter_artist: impl Fn(&Self, &Artist) -> f32,
|
|
|
|
filter_album: impl Fn(&Self, &Album) -> f32,
|
|
|
|
filter_song: impl Fn(&Self, &Song) -> f32,
|
|
|
|
) {
|
|
|
|
let mut a = vec![];
|
|
|
|
for (artist_id, singles, albums) in self.library_sorted.iter() {
|
|
|
|
if let Some(artist) = db.artists().get(artist_id) {
|
|
|
|
let mut filterscore_artist = filter_artist(self, artist);
|
|
|
|
if filterscore_artist > 0.0 {
|
|
|
|
let mut max_score_in_artist = 0.0;
|
|
|
|
if filterscore_artist > 0.0 {
|
|
|
|
let mut s = singles
|
|
|
|
.iter()
|
|
|
|
.filter_map(|song_id| {
|
2023-08-13 23:58:53 +02:00
|
|
|
if let Some(song) = db.songs().get(song_id) {
|
2023-09-22 21:25:43 +02:00
|
|
|
let filterscore_song = filter_song(self, song);
|
|
|
|
if filterscore_song > 0.0 {
|
|
|
|
if filterscore_song > max_score_in_artist {
|
|
|
|
max_score_in_artist = filterscore_song;
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
2023-09-22 21:25:43 +02:00
|
|
|
return Some((*song_id, filterscore_song));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
s.sort_by(|(.., a), (.., b)| b.partial_cmp(a).unwrap_or(Ordering::Equal));
|
|
|
|
let mut al = albums
|
|
|
|
.iter()
|
|
|
|
.filter_map(|(album_id, songs)| {
|
|
|
|
if let Some(album) = db.albums().get(album_id) {
|
|
|
|
let mut filterscore_album = filter_album(self, album);
|
|
|
|
if filterscore_album > 0.0 {
|
|
|
|
let mut max_score_in_album = 0.0;
|
|
|
|
let mut s = songs
|
|
|
|
.iter()
|
|
|
|
.filter_map(|song_id| {
|
|
|
|
if let Some(song) = db.songs().get(song_id) {
|
|
|
|
let filterscore_song = filter_song(self, song);
|
|
|
|
if filterscore_song > 0.0 {
|
|
|
|
if filterscore_song > max_score_in_album {
|
|
|
|
max_score_in_album = filterscore_song;
|
|
|
|
}
|
|
|
|
return Some((*song_id, filterscore_song));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
s.sort_by(|(.., a), (.., b)| {
|
|
|
|
b.partial_cmp(a).unwrap_or(Ordering::Equal)
|
|
|
|
});
|
|
|
|
filterscore_album *= max_score_in_album;
|
|
|
|
if filterscore_album > 0.0 {
|
|
|
|
if filterscore_album > max_score_in_artist {
|
|
|
|
max_score_in_artist = filterscore_album;
|
|
|
|
}
|
|
|
|
return Some((*album_id, s, filterscore_album));
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-09-22 21:25:43 +02:00
|
|
|
None
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
al.sort_by(|(.., a), (.., b)| b.partial_cmp(a).unwrap_or(Ordering::Equal));
|
|
|
|
filterscore_artist *= max_score_in_artist;
|
|
|
|
if filterscore_artist > 0.0 {
|
|
|
|
a.push((*artist_id, s, al, filterscore_artist));
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-09-22 21:25:43 +02:00
|
|
|
a.sort_by(|(.., a), (.., b)| b.partial_cmp(a).unwrap_or(Ordering::Equal));
|
|
|
|
self.library_filtered = a;
|
|
|
|
}
|
|
|
|
/// Sets the contents of the `ScrollBox` based on `self.library_filtered`.
|
|
|
|
fn update_ui(&mut self, db: &Database, line_height: f32) {
|
|
|
|
let mut elems = vec![];
|
|
|
|
for (artist_id, singles, albums, _artist_filterscore) in self.library_filtered.iter() {
|
|
|
|
elems.push(self.build_ui_element_artist(*artist_id, db, line_height));
|
|
|
|
for (song_id, _song_filterscore) in singles {
|
|
|
|
elems.push(self.build_ui_element_song(*song_id, db, line_height));
|
|
|
|
}
|
|
|
|
for (album_id, songs, _album_filterscore) in albums {
|
|
|
|
elems.push(self.build_ui_element_album(*album_id, db, line_height));
|
|
|
|
for (song_id, _song_filterscore) in songs {
|
|
|
|
elems.push(self.build_ui_element_song(*song_id, db, line_height));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let library_scroll_box = self.children[3].try_as_mut::<ScrollBox>().unwrap();
|
|
|
|
library_scroll_box.children = elems;
|
|
|
|
library_scroll_box.config_mut().redraw = true;
|
|
|
|
}
|
|
|
|
fn build_ui_element_artist(&self, id: ArtistId, db: &Database, h: f32) -> (GuiElem, f32) {
|
|
|
|
(
|
|
|
|
GuiElem::new(ListArtist::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
id,
|
|
|
|
if let Some(v) = db.artists().get(&id) {
|
|
|
|
v.name.to_owned()
|
|
|
|
} else {
|
|
|
|
format!("[ Artist #{id} ]")
|
|
|
|
},
|
2023-10-04 16:16:05 +02:00
|
|
|
self.selected.clone(),
|
2023-09-22 21:25:43 +02:00
|
|
|
)),
|
|
|
|
h * 2.5,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
fn build_ui_element_album(&self, id: ArtistId, db: &Database, h: f32) -> (GuiElem, f32) {
|
|
|
|
(
|
|
|
|
GuiElem::new(ListAlbum::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
id,
|
|
|
|
if let Some(v) = db.albums().get(&id) {
|
|
|
|
v.name.to_owned()
|
|
|
|
} else {
|
|
|
|
format!("[ Album #{id} ]")
|
|
|
|
},
|
2023-10-04 16:16:05 +02:00
|
|
|
self.selected.clone(),
|
2023-09-22 21:25:43 +02:00
|
|
|
)),
|
|
|
|
h * 1.5,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
fn build_ui_element_song(&self, id: ArtistId, db: &Database, h: f32) -> (GuiElem, f32) {
|
|
|
|
(
|
|
|
|
GuiElem::new(ListSong::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
id,
|
|
|
|
if let Some(v) = db.songs().get(&id) {
|
|
|
|
v.title.to_owned()
|
|
|
|
} else {
|
|
|
|
format!("[ Song #{id} ]")
|
|
|
|
},
|
2023-10-04 16:16:05 +02:00
|
|
|
self.selected.clone(),
|
2023-09-22 21:25:43 +02:00
|
|
|
)),
|
|
|
|
h,
|
|
|
|
)
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
struct ListArtist {
|
|
|
|
config: GuiElemCfg,
|
|
|
|
id: ArtistId,
|
|
|
|
children: Vec<GuiElem>,
|
2023-08-27 21:53:00 +02:00
|
|
|
mouse: bool,
|
2023-08-13 23:58:53 +02:00
|
|
|
mouse_pos: Vec2,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected: Selected,
|
|
|
|
sel: bool,
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
impl ListArtist {
|
2023-10-04 16:16:05 +02:00
|
|
|
pub fn new(mut config: GuiElemCfg, id: ArtistId, name: String, selected: Selected) -> Self {
|
2023-08-13 23:58:53 +02:00
|
|
|
let label = Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
name,
|
|
|
|
Color::from_int_rgb(81, 24, 125),
|
|
|
|
None,
|
|
|
|
Vec2::new(0.0, 0.5),
|
|
|
|
);
|
2023-10-04 16:16:05 +02:00
|
|
|
config.redraw = true;
|
2023-08-13 23:58:53 +02:00
|
|
|
Self {
|
|
|
|
config: config.w_mouse(),
|
|
|
|
id,
|
|
|
|
children: vec![GuiElem::new(label)],
|
2023-08-27 21:53:00 +02:00
|
|
|
mouse: false,
|
2023-08-13 23:58:53 +02:00
|
|
|
mouse_pos: Vec2::ZERO,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected,
|
|
|
|
sel: false,
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl GuiElemTrait for ListArtist {
|
|
|
|
fn config(&self) -> &GuiElemCfg {
|
|
|
|
&self.config
|
|
|
|
}
|
|
|
|
fn config_mut(&mut self) -> &mut GuiElemCfg {
|
|
|
|
&mut self.config
|
|
|
|
}
|
|
|
|
fn children(&mut self) -> Box<dyn Iterator<Item = &mut GuiElem> + '_> {
|
|
|
|
Box::new(self.children.iter_mut())
|
|
|
|
}
|
|
|
|
fn any(&self) -> &dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn any_mut(&mut self) -> &mut dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn clone_gui(&self) -> Box<dyn GuiElemTrait> {
|
|
|
|
Box::new(self.clone())
|
|
|
|
}
|
|
|
|
fn draw(&mut self, info: &mut DrawInfo, _g: &mut speedy2d::Graphics2D) {
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.config.redraw {
|
|
|
|
self.config.redraw = false;
|
2023-10-26 16:18:52 +02:00
|
|
|
let sel = self.selected.contains_artist(&self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
if sel != self.sel {
|
|
|
|
self.sel = sel;
|
|
|
|
if sel {
|
|
|
|
self.children.push(GuiElem::new(Panel::with_background(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
vec![],
|
|
|
|
Color::from_rgba(1.0, 1.0, 1.0, 0.2),
|
|
|
|
)));
|
|
|
|
} else {
|
|
|
|
self.children.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
if self.mouse {
|
|
|
|
if info.pos.contains(info.mouse_pos) {
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
self.mouse = false;
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.sel {
|
|
|
|
let selected = self.selected.clone();
|
|
|
|
info.actions.push(GuiAction::Do(Box::new(move |gui| {
|
|
|
|
let q = selected.as_queue(
|
|
|
|
gui.gui
|
|
|
|
.inner
|
|
|
|
.children()
|
2023-10-26 16:18:52 +02:00
|
|
|
.nth(3)
|
2023-10-04 16:16:05 +02:00
|
|
|
.unwrap()
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(2)
|
|
|
|
.unwrap()
|
|
|
|
.try_as()
|
|
|
|
.unwrap(),
|
|
|
|
&gui.database.lock().unwrap(),
|
|
|
|
);
|
|
|
|
gui.exec_gui_action(GuiAction::SetDragging(Some((
|
|
|
|
Dragging::Queues(q),
|
|
|
|
None,
|
|
|
|
))));
|
|
|
|
})));
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
self.mouse_pos = Vec2::new(
|
2023-09-22 21:25:43 +02:00
|
|
|
info.mouse_pos.x - info.pos.top_left().x,
|
|
|
|
info.mouse_pos.y - info.pos.top_left().y,
|
2023-08-13 23:58:53 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
fn mouse_down(&mut self, button: MouseButton) -> Vec<GuiAction> {
|
|
|
|
if button == MouseButton::Left {
|
2023-08-27 21:53:00 +02:00
|
|
|
self.mouse = true;
|
2023-08-13 23:58:53 +02:00
|
|
|
let mouse_pos = self.mouse_pos;
|
|
|
|
let w = self.config.pixel_pos.width();
|
|
|
|
let h = self.config.pixel_pos.height();
|
|
|
|
let mut el = GuiElem::new(self.clone());
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.sel {
|
|
|
|
vec![]
|
|
|
|
} else {
|
|
|
|
vec![GuiAction::SetDragging(Some((
|
|
|
|
Dragging::Artist(self.id),
|
|
|
|
Some(Box::new(move |i, g| {
|
|
|
|
let sw = i.pos.width();
|
|
|
|
let sh = i.pos.height();
|
|
|
|
let x = (i.mouse_pos.x - mouse_pos.x) / sw;
|
|
|
|
let y = (i.mouse_pos.y - mouse_pos.y) / sh;
|
|
|
|
el.inner.config_mut().pos =
|
|
|
|
Rectangle::from_tuples((x, y), (x + w / sw, y + h / sh));
|
|
|
|
el.draw(i, g)
|
|
|
|
})),
|
|
|
|
)))]
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
} else {
|
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
fn mouse_up(&mut self, button: MouseButton) -> Vec<GuiAction> {
|
|
|
|
if self.mouse && button == MouseButton::Left {
|
|
|
|
self.mouse = false;
|
2023-10-04 16:16:05 +02:00
|
|
|
self.config.redraw = true;
|
|
|
|
if !self.sel {
|
2023-10-26 16:18:52 +02:00
|
|
|
self.selected.insert_artist(self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
} else {
|
2023-10-26 16:18:52 +02:00
|
|
|
self.selected.remove_artist(&self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
2023-10-04 16:16:05 +02:00
|
|
|
vec![]
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
struct ListAlbum {
|
|
|
|
config: GuiElemCfg,
|
|
|
|
id: AlbumId,
|
|
|
|
children: Vec<GuiElem>,
|
2023-08-27 21:53:00 +02:00
|
|
|
mouse: bool,
|
2023-08-13 23:58:53 +02:00
|
|
|
mouse_pos: Vec2,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected: Selected,
|
|
|
|
sel: bool,
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
impl ListAlbum {
|
2023-10-04 16:16:05 +02:00
|
|
|
pub fn new(mut config: GuiElemCfg, id: AlbumId, name: String, selected: Selected) -> Self {
|
2023-08-13 23:58:53 +02:00
|
|
|
let label = Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
name,
|
|
|
|
Color::from_int_rgb(8, 61, 47),
|
|
|
|
None,
|
|
|
|
Vec2::new(0.0, 0.5),
|
|
|
|
);
|
2023-10-04 16:16:05 +02:00
|
|
|
config.redraw = true;
|
2023-08-13 23:58:53 +02:00
|
|
|
Self {
|
|
|
|
config: config.w_mouse(),
|
|
|
|
id,
|
|
|
|
children: vec![GuiElem::new(label)],
|
2023-08-27 21:53:00 +02:00
|
|
|
mouse: false,
|
2023-08-13 23:58:53 +02:00
|
|
|
mouse_pos: Vec2::ZERO,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected,
|
|
|
|
sel: false,
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl GuiElemTrait for ListAlbum {
|
|
|
|
fn config(&self) -> &GuiElemCfg {
|
|
|
|
&self.config
|
|
|
|
}
|
|
|
|
fn config_mut(&mut self) -> &mut GuiElemCfg {
|
|
|
|
&mut self.config
|
|
|
|
}
|
|
|
|
fn children(&mut self) -> Box<dyn Iterator<Item = &mut GuiElem> + '_> {
|
|
|
|
Box::new(self.children.iter_mut())
|
|
|
|
}
|
|
|
|
fn any(&self) -> &dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn any_mut(&mut self) -> &mut dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn clone_gui(&self) -> Box<dyn GuiElemTrait> {
|
|
|
|
Box::new(self.clone())
|
|
|
|
}
|
|
|
|
fn draw(&mut self, info: &mut DrawInfo, _g: &mut speedy2d::Graphics2D) {
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.config.redraw {
|
|
|
|
self.config.redraw = false;
|
2023-10-26 16:18:52 +02:00
|
|
|
let sel = self.selected.contains_album(&self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
if sel != self.sel {
|
|
|
|
self.sel = sel;
|
|
|
|
if sel {
|
|
|
|
self.children.push(GuiElem::new(Panel::with_background(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
vec![],
|
|
|
|
Color::from_rgba(1.0, 1.0, 1.0, 0.2),
|
|
|
|
)));
|
|
|
|
} else {
|
|
|
|
self.children.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
if self.mouse {
|
|
|
|
if info.pos.contains(info.mouse_pos) {
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
self.mouse = false;
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.sel {
|
|
|
|
let selected = self.selected.clone();
|
|
|
|
info.actions.push(GuiAction::Do(Box::new(move |gui| {
|
|
|
|
let q = selected.as_queue(
|
|
|
|
gui.gui
|
|
|
|
.inner
|
|
|
|
.children()
|
2023-10-26 16:18:52 +02:00
|
|
|
.nth(3)
|
2023-10-04 16:16:05 +02:00
|
|
|
.unwrap()
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(2)
|
|
|
|
.unwrap()
|
|
|
|
.try_as()
|
|
|
|
.unwrap(),
|
|
|
|
&gui.database.lock().unwrap(),
|
|
|
|
);
|
|
|
|
gui.exec_gui_action(GuiAction::SetDragging(Some((
|
|
|
|
Dragging::Queues(q),
|
|
|
|
None,
|
|
|
|
))));
|
|
|
|
})));
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
self.mouse_pos = Vec2::new(
|
2023-09-22 21:25:43 +02:00
|
|
|
info.mouse_pos.x - info.pos.top_left().x,
|
|
|
|
info.mouse_pos.y - info.pos.top_left().y,
|
2023-08-13 23:58:53 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
fn mouse_down(&mut self, button: MouseButton) -> Vec<GuiAction> {
|
|
|
|
if button == MouseButton::Left {
|
2023-08-27 21:53:00 +02:00
|
|
|
self.mouse = true;
|
2023-08-13 23:58:53 +02:00
|
|
|
let mouse_pos = self.mouse_pos;
|
|
|
|
let w = self.config.pixel_pos.width();
|
|
|
|
let h = self.config.pixel_pos.height();
|
|
|
|
let mut el = GuiElem::new(self.clone());
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.sel {
|
|
|
|
vec![]
|
|
|
|
} else {
|
|
|
|
vec![GuiAction::SetDragging(Some((
|
|
|
|
Dragging::Album(self.id),
|
|
|
|
Some(Box::new(move |i, g| {
|
|
|
|
let sw = i.pos.width();
|
|
|
|
let sh = i.pos.height();
|
|
|
|
let x = (i.mouse_pos.x - mouse_pos.x) / sw;
|
|
|
|
let y = (i.mouse_pos.y - mouse_pos.y) / sh;
|
|
|
|
el.inner.config_mut().pos =
|
|
|
|
Rectangle::from_tuples((x, y), (x + w / sw, y + h / sh));
|
|
|
|
el.draw(i, g)
|
|
|
|
})),
|
|
|
|
)))]
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
} else {
|
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
fn mouse_up(&mut self, button: MouseButton) -> Vec<GuiAction> {
|
|
|
|
if self.mouse && button == MouseButton::Left {
|
|
|
|
self.mouse = false;
|
2023-10-04 16:16:05 +02:00
|
|
|
self.config.redraw = true;
|
|
|
|
if !self.sel {
|
2023-10-26 16:18:52 +02:00
|
|
|
self.selected.insert_album(self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
} else {
|
2023-10-26 16:18:52 +02:00
|
|
|
self.selected.remove_album(&self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
2023-10-04 16:16:05 +02:00
|
|
|
vec![]
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
struct ListSong {
|
|
|
|
config: GuiElemCfg,
|
|
|
|
id: SongId,
|
|
|
|
children: Vec<GuiElem>,
|
2023-08-27 21:53:00 +02:00
|
|
|
mouse: bool,
|
2023-08-13 23:58:53 +02:00
|
|
|
mouse_pos: Vec2,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected: Selected,
|
|
|
|
sel: bool,
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
impl ListSong {
|
2023-10-04 16:16:05 +02:00
|
|
|
pub fn new(mut config: GuiElemCfg, id: SongId, name: String, selected: Selected) -> Self {
|
2023-08-13 23:58:53 +02:00
|
|
|
let label = Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
name,
|
|
|
|
Color::from_int_rgb(175, 175, 175),
|
|
|
|
None,
|
|
|
|
Vec2::new(0.0, 0.5),
|
|
|
|
);
|
2023-10-04 16:16:05 +02:00
|
|
|
config.redraw = true;
|
2023-08-13 23:58:53 +02:00
|
|
|
Self {
|
|
|
|
config: config.w_mouse(),
|
|
|
|
id,
|
|
|
|
children: vec![GuiElem::new(label)],
|
2023-08-27 21:53:00 +02:00
|
|
|
mouse: false,
|
2023-08-13 23:58:53 +02:00
|
|
|
mouse_pos: Vec2::ZERO,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected,
|
|
|
|
sel: false,
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl GuiElemTrait for ListSong {
|
|
|
|
fn config(&self) -> &GuiElemCfg {
|
|
|
|
&self.config
|
|
|
|
}
|
|
|
|
fn config_mut(&mut self) -> &mut GuiElemCfg {
|
|
|
|
&mut self.config
|
|
|
|
}
|
|
|
|
fn children(&mut self) -> Box<dyn Iterator<Item = &mut GuiElem> + '_> {
|
|
|
|
Box::new(self.children.iter_mut())
|
|
|
|
}
|
|
|
|
fn any(&self) -> &dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn any_mut(&mut self) -> &mut dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn clone_gui(&self) -> Box<dyn GuiElemTrait> {
|
|
|
|
Box::new(self.clone())
|
|
|
|
}
|
|
|
|
fn draw(&mut self, info: &mut DrawInfo, _g: &mut speedy2d::Graphics2D) {
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.config.redraw {
|
|
|
|
self.config.redraw = false;
|
2023-10-26 16:18:52 +02:00
|
|
|
let sel = self.selected.contains_song(&self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
if sel != self.sel {
|
|
|
|
self.sel = sel;
|
|
|
|
if sel {
|
|
|
|
self.children.push(GuiElem::new(Panel::with_background(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
vec![],
|
|
|
|
Color::from_rgba(1.0, 1.0, 1.0, 0.2),
|
|
|
|
)));
|
|
|
|
} else {
|
|
|
|
self.children.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
if self.mouse {
|
|
|
|
if info.pos.contains(info.mouse_pos) {
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
self.mouse = false;
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.sel {
|
|
|
|
let selected = self.selected.clone();
|
|
|
|
info.actions.push(GuiAction::Do(Box::new(move |gui| {
|
|
|
|
let q = selected.as_queue(
|
|
|
|
gui.gui
|
|
|
|
.inner
|
|
|
|
.children()
|
2023-10-26 16:18:52 +02:00
|
|
|
.nth(3)
|
2023-10-04 16:16:05 +02:00
|
|
|
.unwrap()
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(2)
|
|
|
|
.unwrap()
|
|
|
|
.try_as()
|
|
|
|
.unwrap(),
|
|
|
|
&gui.database.lock().unwrap(),
|
|
|
|
);
|
|
|
|
gui.exec_gui_action(GuiAction::SetDragging(Some((
|
|
|
|
Dragging::Queues(q),
|
|
|
|
None,
|
|
|
|
))));
|
|
|
|
})));
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
self.mouse_pos = Vec2::new(
|
2023-09-22 21:25:43 +02:00
|
|
|
info.mouse_pos.x - info.pos.top_left().x,
|
|
|
|
info.mouse_pos.y - info.pos.top_left().y,
|
2023-08-13 23:58:53 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
fn mouse_down(&mut self, button: MouseButton) -> Vec<GuiAction> {
|
|
|
|
if button == MouseButton::Left {
|
2023-08-27 21:53:00 +02:00
|
|
|
self.mouse = true;
|
2023-08-13 23:58:53 +02:00
|
|
|
let mouse_pos = self.mouse_pos;
|
|
|
|
let w = self.config.pixel_pos.width();
|
|
|
|
let h = self.config.pixel_pos.height();
|
|
|
|
let mut el = GuiElem::new(self.clone());
|
2023-10-04 16:16:05 +02:00
|
|
|
if self.sel {
|
|
|
|
vec![]
|
|
|
|
} else {
|
|
|
|
vec![GuiAction::SetDragging(Some((
|
|
|
|
Dragging::Song(self.id),
|
|
|
|
Some(Box::new(move |i, g| {
|
|
|
|
let sw = i.pos.width();
|
|
|
|
let sh = i.pos.height();
|
|
|
|
let x = (i.mouse_pos.x - mouse_pos.x) / sw;
|
|
|
|
let y = (i.mouse_pos.y - mouse_pos.y) / sh;
|
|
|
|
el.inner.config_mut().pos =
|
|
|
|
Rectangle::from_tuples((x, y), (x + w / sw, y + h / sh));
|
|
|
|
el.draw(i, g)
|
|
|
|
})),
|
|
|
|
)))]
|
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
} else {
|
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
fn mouse_up(&mut self, button: MouseButton) -> Vec<GuiAction> {
|
|
|
|
if self.mouse && button == MouseButton::Left {
|
|
|
|
self.mouse = false;
|
2023-10-04 16:16:05 +02:00
|
|
|
self.config.redraw = true;
|
|
|
|
if !self.sel {
|
2023-10-26 16:18:52 +02:00
|
|
|
self.selected.insert_song(self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
} else {
|
2023-10-26 16:18:52 +02:00
|
|
|
self.selected.remove_song(&self.id);
|
2023-10-04 16:16:05 +02:00
|
|
|
}
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
2023-10-04 16:16:05 +02:00
|
|
|
vec![]
|
2023-08-27 21:53:00 +02:00
|
|
|
}
|
2023-08-13 23:58:53 +02:00
|
|
|
}
|
2023-09-20 16:02:07 +02:00
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
struct FilterPanel {
|
|
|
|
config: GuiElemCfg,
|
|
|
|
children: Vec<GuiElem>,
|
2023-10-24 22:50:21 +02:00
|
|
|
search_settings_changed: Arc<AtomicBool>,
|
2023-09-28 15:37:10 +02:00
|
|
|
tab: usize,
|
2023-10-24 22:50:21 +02:00
|
|
|
new_tab: Arc<AtomicUsize>,
|
2023-09-20 16:02:07 +02:00
|
|
|
line_height: f32,
|
2023-10-24 22:50:21 +02:00
|
|
|
filter_songs: Arc<Mutex<Filter>>,
|
|
|
|
filter_albums: Arc<Mutex<Filter>>,
|
|
|
|
filter_artists: Arc<Mutex<Filter>>,
|
2023-09-20 16:02:07 +02:00
|
|
|
}
|
2023-09-22 21:25:43 +02:00
|
|
|
const FP_CASESENS_N: &'static str = "search is case-insensitive";
|
|
|
|
const FP_CASESENS_Y: &'static str = "search is case-sensitive!";
|
|
|
|
const FP_PREFSTART_N: &'static str = "simple search";
|
|
|
|
const FP_PREFSTART_Y: &'static str = "will prefer matches at the start of a word";
|
2023-09-20 16:02:07 +02:00
|
|
|
impl FilterPanel {
|
2023-09-22 21:25:43 +02:00
|
|
|
pub fn new(
|
2023-10-24 22:50:21 +02:00
|
|
|
search_settings_changed: Arc<AtomicBool>,
|
|
|
|
search_is_case_sensitive: Arc<AtomicBool>,
|
|
|
|
search_prefer_start_matches: Arc<AtomicBool>,
|
|
|
|
filter_songs: Arc<Mutex<Filter>>,
|
|
|
|
filter_albums: Arc<Mutex<Filter>>,
|
|
|
|
filter_artists: Arc<Mutex<Filter>>,
|
2023-10-04 16:16:05 +02:00
|
|
|
selected: Selected,
|
|
|
|
do_something_sender: mpsc::Sender<Box<dyn FnOnce(&mut LibraryBrowser)>>,
|
2023-09-22 21:25:43 +02:00
|
|
|
) -> Self {
|
2023-09-20 16:02:07 +02:00
|
|
|
let is_case_sensitive = search_is_case_sensitive.load(std::sync::atomic::Ordering::Relaxed);
|
2023-09-22 21:25:43 +02:00
|
|
|
let prefer_start_matches =
|
|
|
|
search_prefer_start_matches.load(std::sync::atomic::Ordering::Relaxed);
|
2023-10-24 22:50:21 +02:00
|
|
|
let ssc1 = Arc::clone(&search_settings_changed);
|
|
|
|
let ssc2 = Arc::clone(&search_settings_changed);
|
2023-10-04 16:16:05 +02:00
|
|
|
let sel3 = selected.clone();
|
2023-10-04 19:53:58 +02:00
|
|
|
const VSPLIT: f32 = 0.4;
|
2023-10-04 16:16:05 +02:00
|
|
|
let tab_main = GuiElem::new(ScrollBox::new(
|
2023-10-04 19:53:58 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, 0.0), (VSPLIT, 1.0))),
|
2023-09-28 15:37:10 +02:00
|
|
|
crate::gui_base::ScrollBoxSizeUnit::Pixels,
|
|
|
|
vec![
|
|
|
|
(
|
|
|
|
GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
move |button| {
|
|
|
|
let v = !search_is_case_sensitive
|
|
|
|
.load(std::sync::atomic::Ordering::Relaxed);
|
|
|
|
search_is_case_sensitive.store(v, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
ssc1.store(true, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
*button
|
|
|
|
.children()
|
|
|
|
.next()
|
|
|
|
.unwrap()
|
|
|
|
.try_as_mut::<Label>()
|
|
|
|
.unwrap()
|
|
|
|
.content
|
|
|
|
.text() = if v {
|
|
|
|
FP_CASESENS_Y.to_owned()
|
|
|
|
} else {
|
|
|
|
FP_CASESENS_N.to_owned()
|
|
|
|
};
|
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
if is_case_sensitive {
|
|
|
|
FP_CASESENS_Y.to_owned()
|
|
|
|
} else {
|
|
|
|
FP_CASESENS_N.to_owned()
|
|
|
|
},
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
1.0,
|
|
|
|
),
|
|
|
|
(
|
|
|
|
GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
move |button| {
|
|
|
|
let v = !search_prefer_start_matches
|
|
|
|
.load(std::sync::atomic::Ordering::Relaxed);
|
|
|
|
search_prefer_start_matches
|
|
|
|
.store(v, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
ssc2.store(true, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
*button
|
|
|
|
.children()
|
|
|
|
.next()
|
|
|
|
.unwrap()
|
|
|
|
.try_as_mut::<Label>()
|
|
|
|
.unwrap()
|
|
|
|
.content
|
|
|
|
.text() = if v {
|
|
|
|
FP_PREFSTART_Y.to_owned()
|
|
|
|
} else {
|
|
|
|
FP_PREFSTART_N.to_owned()
|
|
|
|
};
|
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
if prefer_start_matches {
|
|
|
|
FP_PREFSTART_Y.to_owned()
|
|
|
|
} else {
|
|
|
|
FP_PREFSTART_N.to_owned()
|
|
|
|
},
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
1.0,
|
|
|
|
),
|
2023-10-04 16:16:05 +02:00
|
|
|
(
|
|
|
|
GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
move |_| {
|
2023-10-26 16:18:52 +02:00
|
|
|
let mut sel = sel3.clear();
|
2023-10-04 16:16:05 +02:00
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"deselect all".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
1.0,
|
|
|
|
),
|
|
|
|
(
|
|
|
|
GuiElem::new(Panel::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
vec![
|
|
|
|
GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, 0.0), (0.5, 1.0))),
|
|
|
|
{
|
|
|
|
let dss = do_something_sender.clone();
|
|
|
|
move |_| {
|
2023-10-26 16:18:52 +02:00
|
|
|
dss.send(Box::new(|s| s.selected_add_all())).unwrap();
|
2023-10-04 16:16:05 +02:00
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"select all".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
GuiElem::new(Button::new(
|
2023-10-04 19:53:58 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.55, 0.0), (0.75, 1.0))),
|
2023-10-04 16:16:05 +02:00
|
|
|
{
|
|
|
|
let dss = do_something_sender.clone();
|
|
|
|
move |_| {
|
2023-10-26 16:18:52 +02:00
|
|
|
dss.send(Box::new(|s| s.selected_add_songs())).unwrap();
|
2023-10-04 16:16:05 +02:00
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"songs".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
GuiElem::new(Button::new(
|
2023-10-04 19:53:58 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.8, 0.0), (1.0, 1.0))),
|
2023-10-04 16:16:05 +02:00
|
|
|
{
|
|
|
|
let dss = do_something_sender.clone();
|
|
|
|
move |_| {
|
2023-10-26 16:18:52 +02:00
|
|
|
dss.send(Box::new(|s| s.selected_add_albums())).unwrap();
|
2023-10-04 16:16:05 +02:00
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"albums".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
],
|
|
|
|
)),
|
|
|
|
1.0,
|
|
|
|
),
|
2023-09-28 15:37:10 +02:00
|
|
|
],
|
|
|
|
));
|
|
|
|
let tab_filters_songs = GuiElem::new(ScrollBox::new(
|
2023-10-04 19:53:58 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((VSPLIT, 0.0), (1.0, 1.0))),
|
2023-09-28 15:37:10 +02:00
|
|
|
crate::gui_base::ScrollBoxSizeUnit::Pixels,
|
|
|
|
vec![],
|
|
|
|
));
|
|
|
|
let tab_filters_albums = GuiElem::new(ScrollBox::new(
|
2023-10-04 19:53:58 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((VSPLIT, 0.0), (1.0, 1.0))).disabled(),
|
2023-09-28 15:37:10 +02:00
|
|
|
crate::gui_base::ScrollBoxSizeUnit::Pixels,
|
|
|
|
vec![],
|
|
|
|
));
|
|
|
|
let tab_filters_artists = GuiElem::new(ScrollBox::new(
|
2023-10-04 19:53:58 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((VSPLIT, 0.0), (1.0, 1.0))).disabled(),
|
2023-09-28 15:37:10 +02:00
|
|
|
crate::gui_base::ScrollBoxSizeUnit::Pixels,
|
|
|
|
vec![],
|
|
|
|
));
|
2023-10-24 22:50:21 +02:00
|
|
|
let new_tab = Arc::new(AtomicUsize::new(0));
|
|
|
|
let set_tab_1 = Arc::clone(&new_tab);
|
|
|
|
let set_tab_2 = Arc::clone(&new_tab);
|
|
|
|
let set_tab_3 = Arc::clone(&new_tab);
|
2023-09-28 15:37:10 +02:00
|
|
|
const HEIGHT: f32 = 0.1;
|
2023-09-20 16:02:07 +02:00
|
|
|
Self {
|
|
|
|
config: GuiElemCfg::default().disabled(),
|
2023-09-28 15:37:10 +02:00
|
|
|
children: vec![
|
|
|
|
GuiElem::new(Panel::new(
|
2023-10-04 19:53:58 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((VSPLIT, 0.0), (1.0, HEIGHT))),
|
2023-09-28 15:37:10 +02:00
|
|
|
vec![
|
2023-09-22 21:25:43 +02:00
|
|
|
GuiElem::new(Button::new(
|
2023-10-04 16:16:05 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, 0.0), (0.33, 1.0))),
|
2023-09-28 15:37:10 +02:00
|
|
|
move |_| {
|
2023-10-04 19:53:58 +02:00
|
|
|
set_tab_1.store(0, std::sync::atomic::Ordering::Relaxed);
|
2023-09-28 15:37:10 +02:00
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
2023-10-04 16:16:05 +02:00
|
|
|
"Filter Songs".to_owned(),
|
2023-09-22 21:25:43 +02:00
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
GuiElem::new(Button::new(
|
2023-10-04 16:16:05 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.33, 0.0), (0.67, 1.0))),
|
2023-09-28 15:37:10 +02:00
|
|
|
move |_| {
|
2023-10-04 19:53:58 +02:00
|
|
|
set_tab_2.store(1, std::sync::atomic::Ordering::Relaxed);
|
2023-09-22 21:25:43 +02:00
|
|
|
vec![]
|
2023-09-20 16:02:07 +02:00
|
|
|
},
|
2023-09-22 21:25:43 +02:00
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
2023-10-04 16:16:05 +02:00
|
|
|
"Filter Albums".to_owned(),
|
2023-09-22 21:25:43 +02:00
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
2023-09-28 15:37:10 +02:00
|
|
|
GuiElem::new(Button::new(
|
2023-10-04 16:16:05 +02:00
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.67, 0.0), (1.0, 1.0))),
|
2023-09-28 15:37:10 +02:00
|
|
|
move |_| {
|
2023-10-04 19:53:58 +02:00
|
|
|
set_tab_3.store(2, std::sync::atomic::Ordering::Relaxed);
|
2023-09-28 15:37:10 +02:00
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
2023-10-04 16:16:05 +02:00
|
|
|
"Filter Artists".to_owned(),
|
2023-09-28 15:37:10 +02:00
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
],
|
|
|
|
)),
|
|
|
|
GuiElem::new(Panel::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, HEIGHT), (1.0, 1.0))),
|
2023-10-04 19:53:58 +02:00
|
|
|
vec![tab_filters_songs, tab_filters_albums, tab_filters_artists],
|
2023-09-28 15:37:10 +02:00
|
|
|
)),
|
2023-10-04 19:53:58 +02:00
|
|
|
tab_main,
|
2023-09-28 15:37:10 +02:00
|
|
|
],
|
2023-09-20 16:02:07 +02:00
|
|
|
line_height: 0.0,
|
2023-09-28 15:37:10 +02:00
|
|
|
search_settings_changed,
|
|
|
|
tab: 0,
|
|
|
|
new_tab,
|
|
|
|
filter_songs,
|
|
|
|
filter_albums,
|
|
|
|
filter_artists,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fn build_filter(
|
2023-10-24 22:50:21 +02:00
|
|
|
filter: &Arc<Mutex<Filter>>,
|
2023-09-28 15:37:10 +02:00
|
|
|
line_height: f32,
|
2023-10-24 22:50:21 +02:00
|
|
|
on_change: &Arc<impl Fn(bool) + 'static>,
|
2023-09-28 15:37:10 +02:00
|
|
|
path: Vec<usize>,
|
|
|
|
) -> Vec<(GuiElem, f32)> {
|
2023-10-24 22:50:21 +02:00
|
|
|
let f0 = Arc::clone(filter);
|
|
|
|
let oc0 = Arc::clone(on_change);
|
|
|
|
let f1 = Arc::clone(filter);
|
|
|
|
let f2 = Arc::clone(filter);
|
|
|
|
let oc1 = Arc::clone(on_change);
|
|
|
|
let oc2 = Arc::clone(on_change);
|
2023-09-28 15:37:10 +02:00
|
|
|
let mut children = vec![
|
|
|
|
GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
move |_| {
|
|
|
|
f0.lock().unwrap().filters.clear();
|
|
|
|
oc0(true);
|
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"clear filters".to_owned(),
|
|
|
|
Color::LIGHT_GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
move |_| {
|
|
|
|
f1.lock()
|
|
|
|
.unwrap()
|
|
|
|
.filters
|
|
|
|
.push(FilterType::TagEq("Fav".to_owned()));
|
|
|
|
oc1(true);
|
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"must have tag".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
move |_| {
|
|
|
|
f2.lock().unwrap().filters.push(FilterType::TagWithValueInt(
|
|
|
|
"Year=".to_owned(),
|
|
|
|
1990,
|
|
|
|
2000,
|
|
|
|
));
|
|
|
|
oc2(true);
|
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
"tag with integer value between (min) and (max)".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)),
|
|
|
|
];
|
|
|
|
Self::build_filter_editor(
|
|
|
|
&filter.lock().unwrap(),
|
|
|
|
filter,
|
|
|
|
&mut children,
|
|
|
|
0.0,
|
|
|
|
0.05,
|
|
|
|
on_change,
|
|
|
|
path,
|
|
|
|
);
|
|
|
|
children.into_iter().map(|v| (v, line_height)).collect()
|
|
|
|
}
|
|
|
|
fn build_filter_editor(
|
|
|
|
filter: &Filter,
|
2023-10-24 22:50:21 +02:00
|
|
|
mutex: &Arc<Mutex<Filter>>,
|
2023-09-28 15:37:10 +02:00
|
|
|
children: &mut Vec<GuiElem>,
|
|
|
|
mut indent: f32,
|
|
|
|
indent_by: f32,
|
2023-10-24 22:50:21 +02:00
|
|
|
on_change: &Arc<impl Fn(bool) + 'static>,
|
2023-09-28 15:37:10 +02:00
|
|
|
path: Vec<usize>,
|
|
|
|
) {
|
|
|
|
if filter.filters.len() > 1 {
|
2023-10-24 22:50:21 +02:00
|
|
|
let mx = Arc::clone(mutex);
|
|
|
|
let oc = Arc::clone(on_change);
|
2023-09-28 15:37:10 +02:00
|
|
|
let p = path.clone();
|
|
|
|
children.push(GuiElem::new(Button::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((indent, 0.0), (1.0, 1.0))),
|
|
|
|
move |_| {
|
|
|
|
if let Some(f) = match mx.lock().unwrap().get_mut(&p) {
|
|
|
|
Some(Ok(f)) => f.inner_filter(),
|
|
|
|
Some(Err(f)) => Some(f),
|
|
|
|
None => None,
|
|
|
|
} {
|
|
|
|
f.and = !f.and;
|
|
|
|
oc(true);
|
|
|
|
}
|
|
|
|
vec![]
|
|
|
|
},
|
|
|
|
vec![GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::default(),
|
|
|
|
if filter.and { "AND" } else { "OR" }.to_owned(),
|
|
|
|
Color::WHITE,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
))],
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
indent += indent_by;
|
|
|
|
for (i, f) in filter.filters.iter().enumerate() {
|
|
|
|
let mut path = path.clone();
|
|
|
|
path.push(i);
|
|
|
|
match f {
|
|
|
|
FilterType::Nested(f) => Self::build_filter_editor(
|
|
|
|
f, mutex, children, indent, indent_by, on_change, path,
|
|
|
|
),
|
|
|
|
FilterType::Not(f) => {
|
|
|
|
children.push(GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((indent, 0.0), (1.0, 1.0))),
|
|
|
|
"NOT".to_owned(),
|
|
|
|
Color::WHITE,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.0, 0.5),
|
|
|
|
)));
|
|
|
|
Self::build_filter_editor(
|
|
|
|
f, mutex, children, indent, indent_by, on_change, path,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
FilterType::TagEq(v) => {
|
|
|
|
let mut tf = TextField::new_adv(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.1, 0.0), (1.0, 1.0))),
|
|
|
|
v.to_owned(),
|
|
|
|
"tag value".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
);
|
2023-10-24 22:50:21 +02:00
|
|
|
let mx = Arc::clone(mutex);
|
|
|
|
let oc = Arc::clone(on_change);
|
2023-09-28 15:37:10 +02:00
|
|
|
tf.on_changed = Some(Box::new(move |text| {
|
|
|
|
if let Some(Ok(FilterType::TagEq(v))) = mx.lock().unwrap().get_mut(&path) {
|
|
|
|
*v = text.to_owned();
|
|
|
|
oc(false);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
children.push(GuiElem::new(Panel::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((indent, 0.0), (1.0, 1.0))),
|
|
|
|
vec![
|
|
|
|
GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, 0.0), (0.1, 1.0))),
|
|
|
|
"=".to_owned(),
|
|
|
|
Color::WHITE,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
)),
|
|
|
|
GuiElem::new(tf),
|
|
|
|
],
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
FilterType::TagStartsWith(v) => {
|
|
|
|
let mut tf = TextField::new_adv(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.1, 0.0), (1.0, 1.0))),
|
|
|
|
v.to_owned(),
|
|
|
|
"tag value".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
);
|
2023-10-24 22:50:21 +02:00
|
|
|
let mx = Arc::clone(mutex);
|
|
|
|
let oc = Arc::clone(on_change);
|
2023-09-28 15:37:10 +02:00
|
|
|
tf.on_changed = Some(Box::new(move |text| {
|
|
|
|
if let Some(Ok(FilterType::TagStartsWith(v))) =
|
|
|
|
mx.lock().unwrap().get_mut(&path)
|
|
|
|
{
|
|
|
|
*v = text.to_owned();
|
|
|
|
oc(false);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
children.push(GuiElem::new(Panel::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((indent, 0.0), (1.0, 1.0))),
|
|
|
|
vec![
|
|
|
|
GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, 0.0), (0.1, 1.0))),
|
|
|
|
">".to_owned(),
|
|
|
|
Color::WHITE,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
)),
|
|
|
|
GuiElem::new(tf),
|
|
|
|
],
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
FilterType::TagWithValueInt(v, min, max) => {
|
|
|
|
let mut tf = TextField::new_adv(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.1, 0.0), (0.6, 1.0))),
|
|
|
|
v.to_owned(),
|
|
|
|
"tag value".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
);
|
2023-10-24 22:50:21 +02:00
|
|
|
let mx = Arc::clone(mutex);
|
|
|
|
let oc = Arc::clone(on_change);
|
2023-09-28 15:37:10 +02:00
|
|
|
let p = path.clone();
|
|
|
|
tf.on_changed = Some(Box::new(move |text| {
|
|
|
|
if let Some(Ok(FilterType::TagWithValueInt(v, _, _))) =
|
|
|
|
mx.lock().unwrap().get_mut(&p)
|
|
|
|
{
|
|
|
|
*v = text.to_owned();
|
|
|
|
oc(false);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
let mut tf1 = TextField::new_adv(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.6, 0.0), (0.8, 1.0))),
|
|
|
|
min.to_string(),
|
|
|
|
"min".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
);
|
|
|
|
let mut tf2 = TextField::new_adv(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.8, 0.0), (1.0, 1.0))),
|
|
|
|
max.to_string(),
|
|
|
|
"max".to_owned(),
|
|
|
|
Color::GRAY,
|
|
|
|
Color::WHITE,
|
|
|
|
);
|
2023-10-24 22:50:21 +02:00
|
|
|
let mx = Arc::clone(mutex);
|
|
|
|
let oc = Arc::clone(on_change);
|
2023-09-28 15:37:10 +02:00
|
|
|
let p = path.clone();
|
|
|
|
tf1.on_changed = Some(Box::new(move |text| {
|
|
|
|
if let Ok(n) = text.parse() {
|
|
|
|
if let Some(Ok(FilterType::TagWithValueInt(_, v, _))) =
|
|
|
|
mx.lock().unwrap().get_mut(&p)
|
|
|
|
{
|
|
|
|
*v = n;
|
|
|
|
oc(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}));
|
2023-10-24 22:50:21 +02:00
|
|
|
let mx = Arc::clone(mutex);
|
|
|
|
let oc = Arc::clone(on_change);
|
2023-09-28 15:37:10 +02:00
|
|
|
let p = path.clone();
|
|
|
|
tf2.on_changed = Some(Box::new(move |text| {
|
|
|
|
if let Ok(n) = text.parse() {
|
|
|
|
if let Some(Ok(FilterType::TagWithValueInt(_, _, v))) =
|
|
|
|
mx.lock().unwrap().get_mut(&p)
|
|
|
|
{
|
|
|
|
*v = n;
|
|
|
|
oc(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
children.push(GuiElem::new(Panel::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((indent, 0.0), (1.0, 1.0))),
|
|
|
|
vec![
|
|
|
|
GuiElem::new(Label::new(
|
|
|
|
GuiElemCfg::at(Rectangle::from_tuples((0.0, 0.0), (0.1, 1.0))),
|
|
|
|
"..".to_owned(),
|
|
|
|
Color::WHITE,
|
|
|
|
None,
|
|
|
|
Vec2::new(0.5, 0.5),
|
|
|
|
)),
|
|
|
|
GuiElem::new(tf),
|
|
|
|
GuiElem::new(tf1),
|
|
|
|
GuiElem::new(tf2),
|
|
|
|
],
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
}
|
2023-09-20 16:02:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl GuiElemTrait for FilterPanel {
|
|
|
|
fn draw(&mut self, info: &mut DrawInfo, _g: &mut speedy2d::Graphics2D) {
|
2023-09-28 15:37:10 +02:00
|
|
|
// set line height
|
2023-09-20 16:02:07 +02:00
|
|
|
if info.line_height != self.line_height {
|
2023-10-04 19:53:58 +02:00
|
|
|
let sb = self.children[2].try_as_mut::<ScrollBox>().unwrap();
|
|
|
|
for (_, h) in &mut sb.children {
|
|
|
|
*h = info.line_height;
|
|
|
|
}
|
2023-09-28 15:37:10 +02:00
|
|
|
for c in &mut self.children[1].inner.children() {
|
|
|
|
if let Some(sb) = c.try_as_mut::<ScrollBox>() {
|
|
|
|
for (_, h) in &mut sb.children {
|
|
|
|
*h = info.line_height;
|
|
|
|
}
|
|
|
|
}
|
2023-09-20 16:02:07 +02:00
|
|
|
}
|
|
|
|
self.line_height = info.line_height;
|
|
|
|
}
|
2023-09-28 15:37:10 +02:00
|
|
|
// maybe switch tabs
|
2023-10-04 16:16:05 +02:00
|
|
|
let mut new_tab = self.new_tab.load(std::sync::atomic::Ordering::Relaxed);
|
2023-09-28 15:37:10 +02:00
|
|
|
let mut load_tab = false;
|
2023-10-04 19:53:58 +02:00
|
|
|
if new_tab != usize::MAX {
|
|
|
|
self.new_tab
|
|
|
|
.store(usize::MAX, std::sync::atomic::Ordering::Relaxed);
|
2023-09-28 15:37:10 +02:00
|
|
|
load_tab = true;
|
2023-10-04 16:16:05 +02:00
|
|
|
if new_tab == usize::MAX {
|
|
|
|
self.new_tab
|
|
|
|
.store(self.tab, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
new_tab = self.tab;
|
|
|
|
} else {
|
|
|
|
self.children[1]
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(self.tab)
|
|
|
|
.unwrap()
|
|
|
|
.inner
|
|
|
|
.config_mut()
|
|
|
|
.enabled = false;
|
|
|
|
self.children[1]
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(new_tab)
|
|
|
|
.unwrap()
|
|
|
|
.inner
|
|
|
|
.config_mut()
|
|
|
|
.enabled = true;
|
2023-10-04 19:53:58 +02:00
|
|
|
*self.children[0]
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(self.tab)
|
|
|
|
.unwrap()
|
|
|
|
.try_as_mut::<Button>()
|
|
|
|
.unwrap()
|
|
|
|
.children[0]
|
|
|
|
.try_as_mut::<Label>()
|
|
|
|
.unwrap()
|
|
|
|
.content
|
|
|
|
.color() = Color::GRAY;
|
|
|
|
*self.children[0]
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(new_tab)
|
|
|
|
.unwrap()
|
|
|
|
.try_as_mut::<Button>()
|
|
|
|
.unwrap()
|
|
|
|
.children[0]
|
|
|
|
.try_as_mut::<Label>()
|
|
|
|
.unwrap()
|
|
|
|
.content
|
|
|
|
.color() = Color::WHITE;
|
2023-10-04 16:16:05 +02:00
|
|
|
self.tab = new_tab;
|
|
|
|
}
|
2023-09-28 15:37:10 +02:00
|
|
|
}
|
|
|
|
// load tab
|
|
|
|
if load_tab {
|
|
|
|
match new_tab {
|
2023-10-04 19:53:58 +02:00
|
|
|
0 | 1 | 2 => {
|
2023-09-28 15:37:10 +02:00
|
|
|
let sb = self.children[1]
|
|
|
|
.inner
|
|
|
|
.children()
|
|
|
|
.nth(new_tab)
|
|
|
|
.unwrap()
|
|
|
|
.try_as_mut::<ScrollBox>()
|
|
|
|
.unwrap();
|
2023-10-24 22:50:21 +02:00
|
|
|
let ssc = Arc::clone(&self.search_settings_changed);
|
2023-10-04 19:53:58 +02:00
|
|
|
let my_tab = new_tab;
|
2023-10-24 22:50:21 +02:00
|
|
|
let ntab = Arc::clone(&self.new_tab);
|
2023-09-28 15:37:10 +02:00
|
|
|
sb.children = Self::build_filter(
|
|
|
|
match new_tab {
|
2023-10-04 19:53:58 +02:00
|
|
|
0 => &self.filter_songs,
|
|
|
|
1 => &self.filter_albums,
|
|
|
|
2 => &self.filter_artists,
|
2023-09-28 15:37:10 +02:00
|
|
|
_ => unreachable!(),
|
|
|
|
},
|
|
|
|
info.line_height,
|
2023-10-24 22:50:21 +02:00
|
|
|
&Arc::new(move |update_ui| {
|
2023-09-28 15:37:10 +02:00
|
|
|
if update_ui {
|
2023-10-04 19:53:58 +02:00
|
|
|
ntab.store(my_tab, std::sync::atomic::Ordering::Relaxed);
|
2023-09-28 15:37:10 +02:00
|
|
|
}
|
|
|
|
ssc.store(true, std::sync::atomic::Ordering::Relaxed);
|
|
|
|
}),
|
|
|
|
vec![],
|
|
|
|
);
|
|
|
|
sb.config_mut().redraw = true;
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
2023-09-20 16:02:07 +02:00
|
|
|
}
|
|
|
|
fn config(&self) -> &GuiElemCfg {
|
|
|
|
&self.config
|
|
|
|
}
|
|
|
|
fn config_mut(&mut self) -> &mut GuiElemCfg {
|
|
|
|
&mut self.config
|
|
|
|
}
|
|
|
|
fn children(&mut self) -> Box<dyn Iterator<Item = &mut GuiElem> + '_> {
|
|
|
|
Box::new(self.children.iter_mut())
|
|
|
|
}
|
|
|
|
fn any(&self) -> &dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn any_mut(&mut self) -> &mut dyn std::any::Any {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
fn clone_gui(&self) -> Box<dyn GuiElemTrait> {
|
|
|
|
Box::new(self.clone())
|
|
|
|
}
|
|
|
|
}
|
2023-09-28 15:37:10 +02:00
|
|
|
struct Filter {
|
|
|
|
and: bool,
|
|
|
|
filters: Vec<FilterType>,
|
|
|
|
}
|
|
|
|
enum FilterType {
|
|
|
|
Nested(Filter),
|
|
|
|
Not(Filter),
|
|
|
|
TagEq(String),
|
|
|
|
TagStartsWith(String),
|
|
|
|
/// true if the tag is '<String><Integer>' and Integer is between min and max (both inclusive)
|
|
|
|
/// note: <String> usually ends with '='.
|
|
|
|
TagWithValueInt(String, i32, i32),
|
|
|
|
}
|
|
|
|
impl Filter {
|
|
|
|
pub fn passes(&self, gd: &GeneralData) -> bool {
|
|
|
|
if self.filters.is_empty() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
let mut iter = self.filters.iter().map(|v| v.passes(gd));
|
|
|
|
if self.and {
|
|
|
|
iter.all(|v| v)
|
|
|
|
} else {
|
|
|
|
iter.any(|v| v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub fn get_mut(&mut self, path: &[usize]) -> Option<Result<&mut FilterType, &mut Self>> {
|
|
|
|
if let Some(i) = path.first() {
|
|
|
|
let p = &path[1..];
|
|
|
|
if let Some(f) = self.filters.get_mut(*i) {
|
|
|
|
f.get_mut(p)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Some(Err(self))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl FilterType {
|
|
|
|
pub fn passes(&self, gd: &GeneralData) -> bool {
|
|
|
|
match self {
|
|
|
|
Self::Nested(f) => f.passes(gd),
|
|
|
|
Self::Not(f) => !f.passes(gd),
|
|
|
|
Self::TagEq(v) => gd.tags.iter().any(|t| t == v),
|
|
|
|
Self::TagStartsWith(v) => gd.tags.iter().any(|t| t.starts_with(v)),
|
|
|
|
Self::TagWithValueInt(v, min, max) => gd.tags.iter().any(|t| {
|
|
|
|
if t.starts_with(v) {
|
|
|
|
if let Ok(val) = t[v.len()..].parse() {
|
|
|
|
*min <= val && val <= *max
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub fn get_mut(&mut self, path: &[usize]) -> Option<Result<&mut Self, &mut Filter>> {
|
|
|
|
if path.is_empty() {
|
|
|
|
Some(Ok(self))
|
|
|
|
} else {
|
|
|
|
if let Some(f) = self.inner_filter() {
|
|
|
|
f.get_mut(path)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pub fn inner_filter(&mut self) -> Option<&mut Filter> {
|
|
|
|
match self {
|
|
|
|
Self::Nested(f) | Self::Not(f) => Some(f),
|
|
|
|
Self::TagEq(_) | Self::TagStartsWith(_) | Self::TagWithValueInt(..) => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|