mirror of
https://github.com/johrpan/musicus.git
synced 2025-10-26 03:47:23 +01:00
Simplify library manager
This commit is contained in:
parent
610a3d8ff9
commit
d49b9a9efe
5 changed files with 95 additions and 643 deletions
|
|
@ -23,7 +23,7 @@ template $MusicusLibraryManager: Adw.NavigationPage {
|
||||||
margin-top: 24;
|
margin-top: 24;
|
||||||
|
|
||||||
styles [
|
styles [
|
||||||
"heading"
|
"heading",
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -32,7 +32,7 @@ template $MusicusLibraryManager: Adw.NavigationPage {
|
||||||
margin-top: 12;
|
margin-top: 12;
|
||||||
|
|
||||||
styles [
|
styles [
|
||||||
"boxed-list-separate"
|
"boxed-list-separate",
|
||||||
]
|
]
|
||||||
|
|
||||||
Adw.ActionRow library_path_row {
|
Adw.ActionRow library_path_row {
|
||||||
|
|
@ -41,7 +41,7 @@ template $MusicusLibraryManager: Adw.NavigationPage {
|
||||||
activated => $open_library() swapped;
|
activated => $open_library() swapped;
|
||||||
|
|
||||||
styles [
|
styles [
|
||||||
"property"
|
"property",
|
||||||
]
|
]
|
||||||
|
|
||||||
[suffix]
|
[suffix]
|
||||||
|
|
@ -62,232 +62,6 @@ template $MusicusLibraryManager: Adw.NavigationPage {
|
||||||
activated => $export_archive() swapped;
|
activated => $export_archive() swapped;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Gtk.Label {
|
|
||||||
label: _("Contents");
|
|
||||||
xalign: 0;
|
|
||||||
margin-top: 24;
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"heading"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.ListBox {
|
|
||||||
selection-mode: none;
|
|
||||||
margin-top: 12;
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"boxed-list"
|
|
||||||
]
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Persons");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_persons() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_persons_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Roles");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_roles() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_roles_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Instruments");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_instruments() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_instruments_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Works");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_works() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_works_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Ensembles");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_ensembles() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_ensembles_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Recordings");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_recordings() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_recordings_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Tracks");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_tracks() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_tracks_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Mediums");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_mediums() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_mediums_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.ActionRow {
|
|
||||||
title: _("Albums");
|
|
||||||
activatable: true;
|
|
||||||
activated => $show_albums() swapped;
|
|
||||||
|
|
||||||
[suffix]
|
|
||||||
Gtk.Box {
|
|
||||||
spacing: 6;
|
|
||||||
|
|
||||||
Gtk.Label n_albums_label {
|
|
||||||
label: "0";
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"numeric"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.Image {
|
|
||||||
icon-name: "go-next-symbolic";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,42 +0,0 @@
|
||||||
using Gtk 4.0;
|
|
||||||
using Adw 1;
|
|
||||||
|
|
||||||
template $MusicusLibraryManagerAlbumsPage: Adw.NavigationPage {
|
|
||||||
title: _("Albums");
|
|
||||||
|
|
||||||
Adw.ToolbarView {
|
|
||||||
[top]
|
|
||||||
Gtk.Box {
|
|
||||||
orientation: vertical;
|
|
||||||
|
|
||||||
Adw.HeaderBar {
|
|
||||||
[end]
|
|
||||||
Gtk.Button {
|
|
||||||
icon-name: "list-add-symbolic";
|
|
||||||
clicked => $create() swapped;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Adw.Clamp {
|
|
||||||
Gtk.SearchEntry search_entry {
|
|
||||||
placeholder-text: _("Search albums…");
|
|
||||||
search-changed => $search_changed() swapped;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Gtk.ScrolledWindow {
|
|
||||||
Adw.Clamp {
|
|
||||||
Gtk.ListBox list {
|
|
||||||
selection-mode: none;
|
|
||||||
margin-top: 12;
|
|
||||||
valign: start;
|
|
||||||
|
|
||||||
styles [
|
|
||||||
"boxed-list"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
92
src/library_manager.rs
Normal file
92
src/library_manager.rs
Normal file
|
|
@ -0,0 +1,92 @@
|
||||||
|
use std::{cell::OnceCell, ffi::OsStr, path::Path};
|
||||||
|
|
||||||
|
use adw::{prelude::*, subclass::prelude::*};
|
||||||
|
use gettextrs::gettext;
|
||||||
|
use gtk::glib;
|
||||||
|
|
||||||
|
use crate::{library::Library, window::Window};
|
||||||
|
|
||||||
|
mod imp {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[derive(Debug, Default, gtk::CompositeTemplate)]
|
||||||
|
#[template(file = "data/ui/library_manager.blp")]
|
||||||
|
pub struct LibraryManager {
|
||||||
|
pub navigation: OnceCell<adw::NavigationView>,
|
||||||
|
pub library: OnceCell<Library>,
|
||||||
|
|
||||||
|
#[template_child]
|
||||||
|
pub library_path_row: TemplateChild<adw::ActionRow>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[glib::object_subclass]
|
||||||
|
impl ObjectSubclass for LibraryManager {
|
||||||
|
const NAME: &'static str = "MusicusLibraryManager";
|
||||||
|
type Type = super::LibraryManager;
|
||||||
|
type ParentType = adw::NavigationPage;
|
||||||
|
|
||||||
|
fn class_init(klass: &mut Self::Class) {
|
||||||
|
klass.bind_template();
|
||||||
|
klass.bind_template_instance_callbacks();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn instance_init(obj: &glib::subclass::InitializingObject<Self>) {
|
||||||
|
obj.init_template();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ObjectImpl for LibraryManager {}
|
||||||
|
impl WidgetImpl for LibraryManager {}
|
||||||
|
impl NavigationPageImpl for LibraryManager {}
|
||||||
|
}
|
||||||
|
|
||||||
|
glib::wrapper! {
|
||||||
|
pub struct LibraryManager(ObjectSubclass<imp::LibraryManager>)
|
||||||
|
@extends gtk::Widget, adw::NavigationPage;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[gtk::template_callbacks]
|
||||||
|
impl LibraryManager {
|
||||||
|
pub fn new(navigation: &adw::NavigationView, library: &Library) -> Self {
|
||||||
|
let obj: Self = glib::Object::new();
|
||||||
|
|
||||||
|
obj.imp().navigation.set(navigation.to_owned()).unwrap();
|
||||||
|
obj.imp().library.set(library.to_owned()).unwrap();
|
||||||
|
|
||||||
|
if let Some(Some(filename)) = Path::new(&library.folder()).file_name().map(OsStr::to_str) {
|
||||||
|
obj.imp().library_path_row.set_subtitle(filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
obj
|
||||||
|
}
|
||||||
|
|
||||||
|
#[template_callback]
|
||||||
|
async fn open_library(&self) {
|
||||||
|
let dialog = gtk::FileDialog::builder()
|
||||||
|
.title(gettext("Select music library folder"))
|
||||||
|
.modal(true)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
let root = self.root();
|
||||||
|
let window = root
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|r| r.downcast_ref::<gtk::Window>())
|
||||||
|
.and_then(|w| w.downcast_ref::<Window>())
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
match dialog.select_folder_future(Some(window)).await {
|
||||||
|
Err(err) => {
|
||||||
|
if !err.matches(gtk::DialogError::Dismissed) {
|
||||||
|
log::error!("Folder selection failed: {err}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(folder) => window.set_library_folder(&folder),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[template_callback]
|
||||||
|
fn import_archive(&self) {}
|
||||||
|
|
||||||
|
#[template_callback]
|
||||||
|
fn export_archive(&self) {}
|
||||||
|
}
|
||||||
|
|
@ -1,145 +0,0 @@
|
||||||
use std::cell::{OnceCell, RefCell};
|
|
||||||
|
|
||||||
use adw::{prelude::*, subclass::prelude::*};
|
|
||||||
use gettextrs::gettext;
|
|
||||||
use gtk::glib::{self, clone};
|
|
||||||
|
|
||||||
use crate::{db::models::Album, editor::album::AlbumEditor, library::Library};
|
|
||||||
|
|
||||||
mod imp {
|
|
||||||
use super::*;
|
|
||||||
|
|
||||||
#[derive(Debug, Default, gtk::CompositeTemplate)]
|
|
||||||
#[template(file = "data/ui/library_manager_albums_page.blp")]
|
|
||||||
pub struct AlbumsPage {
|
|
||||||
pub navigation: OnceCell<adw::NavigationView>,
|
|
||||||
pub library: OnceCell<Library>,
|
|
||||||
pub albums: RefCell<Vec<Album>>,
|
|
||||||
pub albums_filtered: RefCell<Vec<Album>>,
|
|
||||||
|
|
||||||
#[template_child]
|
|
||||||
pub search_entry: TemplateChild<gtk::SearchEntry>,
|
|
||||||
#[template_child]
|
|
||||||
pub list: TemplateChild<gtk::ListBox>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[glib::object_subclass]
|
|
||||||
impl ObjectSubclass for AlbumsPage {
|
|
||||||
const NAME: &'static str = "MusicusLibraryManagerAlbumsPage";
|
|
||||||
type Type = super::AlbumsPage;
|
|
||||||
type ParentType = adw::NavigationPage;
|
|
||||||
|
|
||||||
fn class_init(klass: &mut Self::Class) {
|
|
||||||
klass.bind_template();
|
|
||||||
klass.bind_template_instance_callbacks();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn instance_init(obj: &glib::subclass::InitializingObject<Self>) {
|
|
||||||
obj.init_template();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ObjectImpl for AlbumsPage {}
|
|
||||||
impl WidgetImpl for AlbumsPage {}
|
|
||||||
|
|
||||||
impl NavigationPageImpl for AlbumsPage {
|
|
||||||
fn showing(&self) {
|
|
||||||
self.parent_showing();
|
|
||||||
self.obj().update();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
glib::wrapper! {
|
|
||||||
pub struct AlbumsPage(ObjectSubclass<imp::AlbumsPage>)
|
|
||||||
@extends gtk::Widget, adw::NavigationPage;
|
|
||||||
}
|
|
||||||
|
|
||||||
#[gtk::template_callbacks]
|
|
||||||
impl AlbumsPage {
|
|
||||||
pub fn new(navigation: &adw::NavigationView, library: &Library) -> Self {
|
|
||||||
let obj: Self = glib::Object::new();
|
|
||||||
let imp = obj.imp();
|
|
||||||
|
|
||||||
imp.navigation.set(navigation.to_owned()).unwrap();
|
|
||||||
imp.library.set(library.to_owned()).unwrap();
|
|
||||||
|
|
||||||
obj
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update(&self) {
|
|
||||||
let albums = self.imp().library.get().unwrap().all_albums().unwrap();
|
|
||||||
self.imp().albums.replace(albums);
|
|
||||||
self.search_changed();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn search_changed(&self) {
|
|
||||||
let albums_filtered = self
|
|
||||||
.imp()
|
|
||||||
.albums
|
|
||||||
.borrow()
|
|
||||||
.iter()
|
|
||||||
.filter(|a| {
|
|
||||||
a.name
|
|
||||||
.get()
|
|
||||||
.contains(&self.imp().search_entry.text().to_string())
|
|
||||||
})
|
|
||||||
.cloned()
|
|
||||||
.collect::<Vec<Album>>();
|
|
||||||
|
|
||||||
self.imp().list.remove_all();
|
|
||||||
|
|
||||||
for album in albums_filtered {
|
|
||||||
let row = adw::ActionRow::builder()
|
|
||||||
.title(album.name.get())
|
|
||||||
.activatable(true)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
row.connect_activated(clone!(
|
|
||||||
#[weak(rename_to = obj)]
|
|
||||||
self,
|
|
||||||
#[strong]
|
|
||||||
album,
|
|
||||||
move |_| {
|
|
||||||
obj.imp().navigation.get().unwrap().push(&AlbumEditor::new(
|
|
||||||
&obj.imp().navigation.get().unwrap(),
|
|
||||||
&obj.imp().library.get().unwrap(),
|
|
||||||
Some(&album),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
));
|
|
||||||
|
|
||||||
let delete_button = gtk::Button::builder()
|
|
||||||
.icon_name("user-trash-symbolic")
|
|
||||||
.tooltip_text(gettext("Delete album"))
|
|
||||||
.valign(gtk::Align::Center)
|
|
||||||
.css_classes(["flat"])
|
|
||||||
.build();
|
|
||||||
|
|
||||||
// TODO:
|
|
||||||
// delete_button.connect_clicked(clone!(
|
|
||||||
// #[weak(rename_to = obj)]
|
|
||||||
// self,
|
|
||||||
// #[strong]
|
|
||||||
// album,
|
|
||||||
// move |_| {
|
|
||||||
// obj.imp().library.delete_album(&album.album_id).unwrap();
|
|
||||||
// }
|
|
||||||
// ));
|
|
||||||
|
|
||||||
row.add_suffix(&delete_button);
|
|
||||||
|
|
||||||
self.imp().list.append(&row);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn create(&self) {
|
|
||||||
self.imp().navigation.get().unwrap().push(&AlbumEditor::new(
|
|
||||||
&self.imp().navigation.get().unwrap(),
|
|
||||||
&self.imp().library.get().unwrap(),
|
|
||||||
None,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,227 +0,0 @@
|
||||||
pub mod albums_page;
|
|
||||||
|
|
||||||
use std::{
|
|
||||||
cell::{OnceCell, RefCell},
|
|
||||||
ffi::OsStr,
|
|
||||||
path::Path,
|
|
||||||
};
|
|
||||||
|
|
||||||
use adw::{prelude::*, subclass::prelude::*};
|
|
||||||
use albums_page::AlbumsPage;
|
|
||||||
use gettextrs::gettext;
|
|
||||||
use gtk::glib;
|
|
||||||
|
|
||||||
use crate::{
|
|
||||||
db::{
|
|
||||||
models::{Album, Ensemble, Instrument, Person, Recording, Role, Track, Work},
|
|
||||||
tables::Medium,
|
|
||||||
},
|
|
||||||
library::Library,
|
|
||||||
window::Window,
|
|
||||||
};
|
|
||||||
|
|
||||||
mod imp {
|
|
||||||
use super::*;
|
|
||||||
|
|
||||||
#[derive(Debug, Default, gtk::CompositeTemplate)]
|
|
||||||
#[template(file = "data/ui/library_manager.blp")]
|
|
||||||
pub struct LibraryManager {
|
|
||||||
pub navigation: OnceCell<adw::NavigationView>,
|
|
||||||
pub library: OnceCell<Library>,
|
|
||||||
|
|
||||||
pub persons: RefCell<Vec<Person>>,
|
|
||||||
pub roles: RefCell<Vec<Role>>,
|
|
||||||
pub instruments: RefCell<Vec<Instrument>>,
|
|
||||||
pub works: RefCell<Vec<Work>>,
|
|
||||||
pub ensembles: RefCell<Vec<Ensemble>>,
|
|
||||||
pub recordings: RefCell<Vec<Recording>>,
|
|
||||||
pub tracks: RefCell<Vec<Track>>,
|
|
||||||
pub mediums: RefCell<Vec<Medium>>,
|
|
||||||
pub albums: RefCell<Vec<Album>>,
|
|
||||||
|
|
||||||
#[template_child]
|
|
||||||
pub library_path_row: TemplateChild<adw::ActionRow>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_persons_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_roles_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_instruments_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_works_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_ensembles_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_recordings_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_tracks_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_mediums_label: TemplateChild<gtk::Label>,
|
|
||||||
#[template_child]
|
|
||||||
pub n_albums_label: TemplateChild<gtk::Label>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[glib::object_subclass]
|
|
||||||
impl ObjectSubclass for LibraryManager {
|
|
||||||
const NAME: &'static str = "MusicusLibraryManager";
|
|
||||||
type Type = super::LibraryManager;
|
|
||||||
type ParentType = adw::NavigationPage;
|
|
||||||
|
|
||||||
fn class_init(klass: &mut Self::Class) {
|
|
||||||
klass.bind_template();
|
|
||||||
klass.bind_template_instance_callbacks();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn instance_init(obj: &glib::subclass::InitializingObject<Self>) {
|
|
||||||
obj.init_template();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ObjectImpl for LibraryManager {}
|
|
||||||
impl WidgetImpl for LibraryManager {}
|
|
||||||
|
|
||||||
impl NavigationPageImpl for LibraryManager {
|
|
||||||
fn showing(&self) {
|
|
||||||
self.parent_showing();
|
|
||||||
self.obj().update();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
glib::wrapper! {
|
|
||||||
pub struct LibraryManager(ObjectSubclass<imp::LibraryManager>)
|
|
||||||
@extends gtk::Widget, adw::NavigationPage;
|
|
||||||
}
|
|
||||||
|
|
||||||
#[gtk::template_callbacks]
|
|
||||||
impl LibraryManager {
|
|
||||||
pub fn new(navigation: &adw::NavigationView, library: &Library) -> Self {
|
|
||||||
let obj: Self = glib::Object::new();
|
|
||||||
let imp = obj.imp();
|
|
||||||
|
|
||||||
imp.navigation.set(navigation.to_owned()).unwrap();
|
|
||||||
imp.library.set(library.to_owned()).unwrap();
|
|
||||||
|
|
||||||
obj
|
|
||||||
}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
async fn open_library(&self) {
|
|
||||||
let dialog = gtk::FileDialog::builder()
|
|
||||||
.title(gettext("Select music library folder"))
|
|
||||||
.modal(true)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
let root = self.root();
|
|
||||||
let window = root
|
|
||||||
.as_ref()
|
|
||||||
.and_then(|r| r.downcast_ref::<gtk::Window>())
|
|
||||||
.and_then(|w| w.downcast_ref::<Window>())
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
match dialog.select_folder_future(Some(window)).await {
|
|
||||||
Err(err) => {
|
|
||||||
if !err.matches(gtk::DialogError::Dismissed) {
|
|
||||||
log::error!("Folder selection failed: {err}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(folder) => window.set_library_folder(&folder),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn import_archive(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn export_archive(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_persons(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_roles(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_instruments(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_works(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_ensembles(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_recordings(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_tracks(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_mediums(&self) {}
|
|
||||||
|
|
||||||
#[template_callback]
|
|
||||||
fn show_albums(&self) {
|
|
||||||
let navigation = self.imp().navigation.get().unwrap();
|
|
||||||
let library = self.imp().library.get().unwrap();
|
|
||||||
navigation.push(&AlbumsPage::new(navigation, library));
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: Make this async.
|
|
||||||
fn update(&self) {
|
|
||||||
let library = self.imp().library.get().unwrap();
|
|
||||||
|
|
||||||
if let Some(Some(filename)) = Path::new(&library.folder()).file_name().map(OsStr::to_str) {
|
|
||||||
self.imp().library_path_row.set_subtitle(filename);
|
|
||||||
}
|
|
||||||
|
|
||||||
let persons = library.all_persons().unwrap();
|
|
||||||
self.imp()
|
|
||||||
.n_persons_label
|
|
||||||
.set_label(&persons.len().to_string());
|
|
||||||
self.imp().persons.replace(persons);
|
|
||||||
|
|
||||||
let roles = library.all_roles().unwrap();
|
|
||||||
self.imp().n_roles_label.set_label(&roles.len().to_string());
|
|
||||||
self.imp().roles.replace(roles);
|
|
||||||
|
|
||||||
let instruments = library.all_instruments().unwrap();
|
|
||||||
self.imp()
|
|
||||||
.n_instruments_label
|
|
||||||
.set_label(&instruments.len().to_string());
|
|
||||||
self.imp().instruments.replace(instruments);
|
|
||||||
|
|
||||||
let works = library.all_works().unwrap();
|
|
||||||
self.imp().n_works_label.set_label(&works.len().to_string());
|
|
||||||
self.imp().works.replace(works);
|
|
||||||
|
|
||||||
let ensembles = library.all_ensembles().unwrap();
|
|
||||||
self.imp()
|
|
||||||
.n_ensembles_label
|
|
||||||
.set_label(&ensembles.len().to_string());
|
|
||||||
self.imp().ensembles.replace(ensembles);
|
|
||||||
|
|
||||||
let recordings = library.all_recordings().unwrap();
|
|
||||||
self.imp()
|
|
||||||
.n_recordings_label
|
|
||||||
.set_label(&recordings.len().to_string());
|
|
||||||
self.imp().recordings.replace(recordings);
|
|
||||||
|
|
||||||
let tracks = library.all_tracks().unwrap();
|
|
||||||
self.imp()
|
|
||||||
.n_tracks_label
|
|
||||||
.set_label(&tracks.len().to_string());
|
|
||||||
self.imp().tracks.replace(tracks);
|
|
||||||
|
|
||||||
let mediums = library.all_mediums().unwrap();
|
|
||||||
self.imp()
|
|
||||||
.n_mediums_label
|
|
||||||
.set_label(&mediums.len().to_string());
|
|
||||||
self.imp().mediums.replace(mediums);
|
|
||||||
|
|
||||||
let albums = library.all_albums().unwrap();
|
|
||||||
self.imp()
|
|
||||||
.n_albums_label
|
|
||||||
.set_label(&albums.len().to_string());
|
|
||||||
self.imp().albums.replace(albums);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue