musicus/src/home_page.rs

362 lines
12 KiB
Rust
Raw Normal View History

use std::cell::{OnceCell, RefCell};
use adw::subclass::{navigation_page::NavigationPageImpl, prelude::*};
use gtk::{
gio,
glib::{self, Properties},
prelude::*,
};
2023-09-30 18:26:11 +02:00
use crate::{
2024-06-05 13:26:47 +02:00
album_tile::MusicusAlbumTile,
config,
2024-03-23 18:06:46 +01:00
db::models::*,
editor::{person_editor::MusicusPersonEditor, work_editor::MusicusWorkEditor},
2024-03-23 18:06:46 +01:00
library::{LibraryQuery, MusicusLibrary},
2023-10-07 22:49:20 +02:00
player::MusicusPlayer,
2024-06-09 16:26:37 +02:00
program::Program,
program_tile::MusicusProgramTile,
2023-10-08 16:40:59 +02:00
recording_tile::MusicusRecordingTile,
2023-10-07 22:49:20 +02:00
search_entry::MusicusSearchEntry,
2023-10-08 00:16:41 +02:00
search_tag::Tag,
2023-10-08 16:40:59 +02:00
tag_tile::MusicusTagTile,
2023-09-30 18:26:11 +02:00
};
2024-06-05 13:26:47 +02:00
2023-09-20 13:49:02 +02:00
mod imp {
use super::*;
2023-09-29 21:18:28 +02:00
#[derive(Properties, Debug, Default, gtk::CompositeTemplate)]
#[properties(wrapper_type = super::MusicusHomePage)]
2023-09-24 13:58:05 +02:00
#[template(file = "data/ui/home_page.blp")]
2023-09-20 13:49:02 +02:00
pub struct MusicusHomePage {
#[property(get, construct_only)]
pub navigation: OnceCell<adw::NavigationView>,
2023-10-07 22:49:20 +02:00
#[property(get, construct_only)]
2023-09-30 18:26:11 +02:00
pub library: OnceCell<MusicusLibrary>,
2023-10-07 22:49:20 +02:00
#[property(get, construct_only)]
pub player: OnceCell<MusicusPlayer>,
2024-06-10 16:57:36 +02:00
pub programs: RefCell<Vec<Program>>,
2023-10-08 15:11:47 +02:00
pub composers: RefCell<Vec<Person>>,
pub performers: RefCell<Vec<Person>>,
2023-10-08 00:16:41 +02:00
pub ensembles: RefCell<Vec<Ensemble>>,
pub works: RefCell<Vec<Work>>,
pub recordings: RefCell<Vec<Recording>>,
2024-06-05 13:26:47 +02:00
pub albums: RefCell<Vec<Album>>,
2023-10-08 00:16:41 +02:00
2023-09-20 13:49:02 +02:00
#[template_child]
2023-09-30 00:22:33 +02:00
pub search_entry: TemplateChild<MusicusSearchEntry>,
2023-09-20 13:49:02 +02:00
#[template_child]
pub stack: TemplateChild<gtk::Stack>,
#[template_child]
pub header_box: TemplateChild<gtk::Box>,
#[template_child]
pub title_label: TemplateChild<gtk::Label>,
#[template_child]
pub subtitle_label: TemplateChild<gtk::Label>,
#[template_child]
2024-04-01 18:43:00 +02:00
pub programs_flow_box: TemplateChild<gtk::FlowBox>,
#[template_child]
2023-10-08 15:11:47 +02:00
pub composers_flow_box: TemplateChild<gtk::FlowBox>,
#[template_child]
pub performers_flow_box: TemplateChild<gtk::FlowBox>,
2023-09-20 13:49:02 +02:00
#[template_child]
pub ensembles_flow_box: TemplateChild<gtk::FlowBox>,
#[template_child]
2023-09-20 13:49:02 +02:00
pub works_flow_box: TemplateChild<gtk::FlowBox>,
#[template_child]
pub recordings_flow_box: TemplateChild<gtk::FlowBox>,
2023-09-21 17:19:31 +02:00
#[template_child]
2024-06-05 13:26:47 +02:00
pub albums_flow_box: TemplateChild<gtk::FlowBox>,
#[template_child]
2023-09-21 17:19:31 +02:00
pub play_button: TemplateChild<gtk::Button>,
2023-09-20 13:49:02 +02:00
}
#[glib::object_subclass]
impl ObjectSubclass for MusicusHomePage {
const NAME: &'static str = "MusicusHomePage";
type Type = super::MusicusHomePage;
type ParentType = adw::NavigationPage;
fn class_init(klass: &mut Self::Class) {
klass.bind_template();
2023-09-21 17:19:31 +02:00
klass.bind_template_instance_callbacks();
2023-09-20 13:49:02 +02:00
}
fn instance_init(obj: &glib::subclass::InitializingObject<Self>) {
obj.init_template();
}
}
2023-09-29 21:18:28 +02:00
#[glib::derived_properties]
2023-09-20 13:49:02 +02:00
impl ObjectImpl for MusicusHomePage {
fn constructed(&self) {
self.parent_constructed();
2023-09-30 00:22:33 +02:00
self.search_entry.set_key_capture_widget(&*self.obj());
2023-09-20 13:49:02 +02:00
let obj = self.obj().to_owned();
self.search_entry.connect_query_changed(move |entry| {
obj.query(&entry.query());
});
2023-09-30 19:10:32 +02:00
2025-01-17 18:16:01 +01:00
let obj = self.obj().to_owned();
self.library.get().unwrap().connect_changed(move |_| {
obj.imp().search_entry.reset();
});
2023-09-29 21:18:28 +02:00
self.player
2023-10-07 22:49:20 +02:00
.get()
.unwrap()
2023-09-29 21:18:28 +02:00
.bind_property("active", &self.play_button.get(), "visible")
.invert_boolean()
.sync_create()
.build();
let settings = gio::Settings::new(&config::APP_ID);
2024-06-09 16:26:37 +02:00
2024-06-10 16:57:36 +02:00
let programs = vec![
Program::deserialize(&settings.string("program1")).unwrap(),
Program::deserialize(&settings.string("program2")).unwrap(),
Program::deserialize(&settings.string("program3")).unwrap(),
];
2024-04-01 18:43:00 +02:00
2024-06-10 16:57:36 +02:00
for program in &programs {
self.programs_flow_box
.append(&MusicusProgramTile::new(program.to_owned()));
}
2024-04-01 18:43:00 +02:00
2024-06-10 16:57:36 +02:00
self.programs.replace(programs);
2024-04-01 18:43:00 +02:00
2023-10-07 22:49:20 +02:00
self.obj().query(&LibraryQuery::default());
2023-09-20 13:49:02 +02:00
}
}
impl WidgetImpl for MusicusHomePage {}
impl NavigationPageImpl for MusicusHomePage {}
}
glib::wrapper! {
pub struct MusicusHomePage(ObjectSubclass<imp::MusicusHomePage>)
@extends gtk::Widget, adw::NavigationPage;
}
2023-09-21 17:19:31 +02:00
#[gtk::template_callbacks]
2023-09-20 13:49:02 +02:00
impl MusicusHomePage {
pub fn new(
navigation: &adw::NavigationView,
library: &MusicusLibrary,
player: &MusicusPlayer,
) -> Self {
2023-10-07 22:49:20 +02:00
glib::Object::builder()
.property("navigation", navigation)
2023-10-07 22:49:20 +02:00
.property("library", library)
.property("player", player)
.build()
2023-09-20 13:49:02 +02:00
}
2023-09-21 17:19:31 +02:00
#[template_callback]
fn back_button_clicked(&self, _: &gtk::Button) {
self.imp().search_entry.reset();
}
#[template_callback]
fn edit_button_clicked(&self, _: &gtk::Button) {
if let Some(tag) = self.imp().search_entry.tags().first() {
match tag {
Tag::Composer(person) | Tag::Performer(person) => {
2024-06-06 15:17:56 +02:00
self.navigation().push(&MusicusPersonEditor::new(
&self.navigation(),
&self.library(),
Some(person),
));
}
Tag::Ensemble(_) => todo!(),
Tag::Work(work) => self.navigation().push(&MusicusWorkEditor::new(
&self.navigation(),
&self.library(),
Some(work),
false,
)),
}
}
}
2023-09-21 17:19:31 +02:00
#[template_callback]
fn play(&self, _: &gtk::Button) {
log::info!("Play button clicked");
2024-06-10 16:57:36 +02:00
let program = Program::from_query(self.imp().search_entry.query());
self.player().set_program(program);
2023-10-07 22:49:20 +02:00
self.player().play();
2023-09-21 17:19:31 +02:00
}
#[template_callback]
2023-09-30 00:22:33 +02:00
fn select(&self, search_entry: &MusicusSearchEntry) {
2023-10-08 00:16:41 +02:00
let imp = self.imp();
2024-04-01 18:43:00 +02:00
if imp.programs_flow_box.is_visible() {
2024-06-10 16:57:36 +02:00
if let Some(program) = imp.programs.borrow().first().cloned() {
self.player().set_program(program);
}
2024-04-01 18:43:00 +02:00
} else {
let (composer, performer, ensemble, work, recording, album) = {
(
imp.composers.borrow().first().cloned(),
imp.performers.borrow().first().cloned(),
imp.ensembles.borrow().first().cloned(),
imp.works.borrow().first().cloned(),
imp.recordings.borrow().first().cloned(),
imp.albums.borrow().first().cloned(),
)
};
if let Some(person) = composer {
search_entry.add_tag(Tag::Composer(person));
} else if let Some(person) = performer {
search_entry.add_tag(Tag::Performer(person));
} else if let Some(ensemble) = ensemble {
search_entry.add_tag(Tag::Ensemble(ensemble));
} else if let Some(work) = work {
search_entry.add_tag(Tag::Work(work));
} else if let Some(recording) = recording {
2024-06-10 16:57:36 +02:00
self.player().play_recording(&recording);
2024-04-01 18:43:00 +02:00
} else if let Some(album) = album {
self.show_album(&album);
}
2023-10-08 00:16:41 +02:00
}
2023-09-21 17:19:31 +02:00
}
2023-10-07 22:49:20 +02:00
2024-04-01 18:43:00 +02:00
#[template_callback]
fn program_selected(&self, tile: &gtk::FlowBoxChild, _: &gtk::FlowBox) {
2024-06-10 16:57:36 +02:00
self.player()
.set_program(tile.downcast_ref::<MusicusProgramTile>().unwrap().program());
2024-04-01 18:43:00 +02:00
}
2023-10-11 12:04:49 +02:00
#[template_callback]
fn tile_selected(&self, tile: &gtk::FlowBoxChild, _: &gtk::FlowBox) {
self.imp()
.search_entry
.add_tag(tile.downcast_ref::<MusicusTagTile>().unwrap().tag().clone())
}
#[template_callback]
fn recording_selected(&self, tile: &gtk::FlowBoxChild, _: &gtk::FlowBox) {
2024-06-10 16:57:36 +02:00
self.player().play_recording(
2023-10-11 12:04:49 +02:00
tile.downcast_ref::<MusicusRecordingTile>()
.unwrap()
.recording(),
);
}
2024-06-05 13:26:47 +02:00
#[template_callback]
fn album_selected(&self, tile: &gtk::FlowBoxChild, _: &gtk::FlowBox) {
self.show_album(tile.downcast_ref::<MusicusAlbumTile>().unwrap().album());
}
fn show_album(&self, _album: &Album) {
todo!("Show album");
}
2023-10-07 22:49:20 +02:00
fn query(&self, query: &LibraryQuery) {
let imp = self.imp();
2024-03-23 18:06:46 +01:00
let results = self.library().query(query).unwrap();
2023-10-07 22:49:20 +02:00
for flowbox in [
2023-10-08 15:11:47 +02:00
&imp.composers_flow_box,
&imp.performers_flow_box,
&imp.ensembles_flow_box,
&imp.works_flow_box,
&imp.recordings_flow_box,
2024-06-05 13:26:47 +02:00
&imp.albums_flow_box,
] {
while let Some(widget) = flowbox.first_child() {
flowbox.remove(&widget);
}
2023-10-07 22:49:20 +02:00
}
2024-04-01 18:43:00 +02:00
imp.programs_flow_box.set_visible(query.is_empty());
if let Some(tag) = imp.search_entry.tags().first() {
match tag {
Tag::Composer(person) | Tag::Performer(person) => {
imp.title_label.set_text(&person.name.get());
imp.subtitle_label.set_visible(false);
}
Tag::Ensemble(ensemble) => {
imp.title_label.set_text(&ensemble.name.get());
imp.subtitle_label.set_visible(false);
}
Tag::Work(work) => {
imp.title_label.set_text(&work.name.get());
imp.subtitle_label.set_text(&work.composers_string());
imp.subtitle_label.set_visible(true);
}
}
imp.header_box.set_visible(true);
} else {
imp.header_box.set_visible(false);
}
if results.is_empty() {
imp.stack.set_visible_child_name("empty");
} else {
imp.stack.set_visible_child_name("results");
2023-10-08 15:11:47 +02:00
imp.composers_flow_box
.set_visible(!results.composers.is_empty());
imp.performers_flow_box
.set_visible(!results.performers.is_empty());
imp.ensembles_flow_box
.set_visible(!results.ensembles.is_empty());
imp.works_flow_box.set_visible(!results.works.is_empty());
imp.recordings_flow_box
.set_visible(!results.recordings.is_empty());
2024-06-05 13:26:47 +02:00
imp.albums_flow_box.set_visible(!results.albums.is_empty());
2023-10-08 15:11:47 +02:00
for composer in &results.composers {
imp.composers_flow_box
2023-10-08 16:40:59 +02:00
.append(&MusicusTagTile::new(Tag::Composer(composer.clone())));
2023-10-08 15:11:47 +02:00
}
for performer in &results.performers {
imp.performers_flow_box
2023-10-08 16:40:59 +02:00
.append(&MusicusTagTile::new(Tag::Performer(performer.clone())));
}
2023-10-08 00:16:41 +02:00
for ensemble in &results.ensembles {
imp.ensembles_flow_box
2023-10-08 16:40:59 +02:00
.append(&MusicusTagTile::new(Tag::Ensemble(ensemble.clone())));
}
2023-10-08 00:16:41 +02:00
for work in &results.works {
2023-10-08 16:40:59 +02:00
imp.works_flow_box
.append(&MusicusTagTile::new(Tag::Work(work.clone())));
}
2023-10-08 00:16:41 +02:00
for recording in &results.recordings {
imp.recordings_flow_box.append(&MusicusRecordingTile::new(
&self.navigation(),
&self.library(),
recording,
));
}
2023-10-08 00:16:41 +02:00
2024-06-05 13:26:47 +02:00
for album in &results.albums {
imp.albums_flow_box.append(&MusicusAlbumTile::new(album));
}
2023-10-08 15:11:47 +02:00
imp.composers.replace(results.composers);
imp.performers.replace(results.performers);
2023-10-08 00:16:41 +02:00
imp.ensembles.replace(results.ensembles);
imp.works.replace(results.works);
imp.recordings.replace(results.recordings);
2024-06-05 13:26:47 +02:00
imp.albums.replace(results.albums);
}
2023-10-07 22:49:20 +02:00
}
2023-09-20 13:49:02 +02:00
}