2020-12-02 15:41:19 +01:00
|
|
|
use super::selector::Selector;
|
2021-05-07 23:49:05 +02:00
|
|
|
use crate::editors::{PersonEditor, RecordingEditor, WorkEditor};
|
2021-02-03 15:46:48 +01:00
|
|
|
use crate::navigator::{NavigationHandle, Screen};
|
|
|
|
|
use crate::widgets::Widget;
|
2021-05-09 11:12:19 +02:00
|
|
|
use adw::prelude::*;
|
2020-12-02 15:41:19 +01:00
|
|
|
use gettextrs::gettext;
|
|
|
|
|
use glib::clone;
|
2021-05-07 23:49:05 +02:00
|
|
|
use musicus_backend::db::{Person, Recording, Work};
|
2020-12-02 15:41:19 +01:00
|
|
|
use std::rc::Rc;
|
|
|
|
|
|
|
|
|
|
/// A screen for selecting a recording.
|
|
|
|
|
pub struct RecordingSelector {
|
2021-02-03 15:46:48 +01:00
|
|
|
handle: NavigationHandle<Recording>,
|
|
|
|
|
selector: Rc<Selector<Person>>,
|
2020-12-02 15:41:19 +01:00
|
|
|
}
|
|
|
|
|
|
2021-02-03 15:46:48 +01:00
|
|
|
impl Screen<(), Recording> for RecordingSelector {
|
|
|
|
|
fn new(_: (), handle: NavigationHandle<Recording>) -> Rc<Self> {
|
2020-12-02 15:41:19 +01:00
|
|
|
// Create UI
|
|
|
|
|
|
2021-05-07 20:44:27 +02:00
|
|
|
let selector = Selector::<Person>::new(Rc::clone(&handle.backend));
|
2021-02-03 15:46:48 +01:00
|
|
|
selector.set_title(&gettext("Select composer"));
|
2020-12-02 15:41:19 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
let this = Rc::new(Self { handle, selector });
|
2020-12-02 15:41:19 +01:00
|
|
|
|
|
|
|
|
// Connect signals and callbacks
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_back_cb(clone!(@weak this => move || {
|
2021-02-03 15:46:48 +01:00
|
|
|
this.handle.pop(None);
|
2020-12-02 15:41:19 +01:00
|
|
|
}));
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_add_cb(clone!(@weak this => move || {
|
2021-02-03 15:46:48 +01:00
|
|
|
spawn!(@clone this, async move {
|
|
|
|
|
if let Some(person) = push!(this.handle, PersonEditor, None).await {
|
|
|
|
|
// We can assume that there are no existing works of this composer and
|
|
|
|
|
// immediately show the work editor. Going back from the work editor will
|
|
|
|
|
// correctly show the person selector again.
|
|
|
|
|
|
|
|
|
|
let work = Work::new(person);
|
|
|
|
|
if let Some(work) = push!(this.handle, WorkEditor, Some(work)).await {
|
|
|
|
|
// There will also be no existing recordings, so we show the recording
|
|
|
|
|
// editor next.
|
|
|
|
|
|
|
|
|
|
let recording = Recording::new(work);
|
|
|
|
|
if let Some(recording) = push!(this.handle, RecordingEditor, Some(recording)).await {
|
|
|
|
|
this.handle.pop(Some(recording));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}));
|
2020-12-17 21:23:16 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector
|
|
|
|
|
.set_load_online(clone!(@weak this => @default-panic, move || {
|
|
|
|
|
async move { Ok(this.handle.backend.cl().get_persons().await?) }
|
|
|
|
|
}));
|
2020-12-02 15:41:19 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector
|
|
|
|
|
.set_load_local(clone!(@weak this => @default-panic, move || {
|
|
|
|
|
async move { this.handle.backend.db().get_persons().await.unwrap() }
|
|
|
|
|
}));
|
2021-02-03 15:46:48 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_make_widget(clone!(@weak this => @default-panic, move |person| {
|
2021-05-09 11:12:19 +02:00
|
|
|
let row = adw::ActionRow::new();
|
2021-02-03 15:46:48 +01:00
|
|
|
row.set_activatable(true);
|
2021-10-10 10:24:18 +02:00
|
|
|
row.set_title(&person.name_lf());
|
2021-02-03 15:46:48 +01:00
|
|
|
|
|
|
|
|
let person = person.to_owned();
|
2021-05-07 23:49:05 +02:00
|
|
|
row.connect_activated(clone!(@weak this => move |_| {
|
2021-02-03 15:46:48 +01:00
|
|
|
// Instead of returning the person from here, like the person selector does, we
|
|
|
|
|
// show a second selector for choosing the work.
|
|
|
|
|
|
|
|
|
|
let person = person.clone();
|
|
|
|
|
spawn!(@clone this, async move {
|
|
|
|
|
if let Some(work) = push!(this.handle, RecordingSelectorWorkScreen, person).await {
|
|
|
|
|
// Now the user can select a recording for that work.
|
|
|
|
|
|
|
|
|
|
if let Some(recording) = push!(this.handle, RecordingSelectorRecordingScreen, work).await {
|
|
|
|
|
this.handle.pop(Some(recording));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
2020-12-02 15:41:19 +01:00
|
|
|
}));
|
|
|
|
|
|
2021-02-03 15:46:48 +01:00
|
|
|
row.upcast()
|
|
|
|
|
}));
|
|
|
|
|
|
2020-12-02 15:41:19 +01:00
|
|
|
this.selector
|
2021-02-03 15:46:48 +01:00
|
|
|
.set_filter(|search, person| person.name_fl().to_lowercase().contains(search));
|
2020-12-02 15:41:19 +01:00
|
|
|
|
2021-02-03 15:46:48 +01:00
|
|
|
this
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Widget for RecordingSelector {
|
|
|
|
|
fn get_widget(&self) -> gtk::Widget {
|
|
|
|
|
self.selector.widget.clone().upcast()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The work selector within the recording selector.
|
|
|
|
|
struct RecordingSelectorWorkScreen {
|
|
|
|
|
handle: NavigationHandle<Work>,
|
|
|
|
|
person: Person,
|
|
|
|
|
selector: Rc<Selector<Work>>,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Screen<Person, Work> for RecordingSelectorWorkScreen {
|
|
|
|
|
fn new(person: Person, handle: NavigationHandle<Work>) -> Rc<Self> {
|
2021-05-07 20:44:27 +02:00
|
|
|
let selector = Selector::<Work>::new(Rc::clone(&handle.backend));
|
2021-02-03 15:46:48 +01:00
|
|
|
selector.set_title(&gettext("Select work"));
|
|
|
|
|
selector.set_subtitle(&person.name_fl());
|
|
|
|
|
|
|
|
|
|
let this = Rc::new(Self {
|
|
|
|
|
handle,
|
|
|
|
|
person,
|
|
|
|
|
selector,
|
|
|
|
|
});
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_back_cb(clone!(@weak this => move || {
|
2021-02-03 15:46:48 +01:00
|
|
|
this.handle.pop(None);
|
|
|
|
|
}));
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_add_cb(clone!(@weak this => move || {
|
2021-02-03 15:46:48 +01:00
|
|
|
spawn!(@clone this, async move {
|
|
|
|
|
let work = Work::new(this.person.clone());
|
|
|
|
|
if let Some(work) = push!(this.handle, WorkEditor, Some(work)).await {
|
|
|
|
|
this.handle.pop(Some(work));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}));
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector
|
|
|
|
|
.set_load_online(clone!(@weak this => @default-panic, move || {
|
|
|
|
|
async move { Ok(this.handle.backend.cl().get_works(&this.person.id).await?) }
|
|
|
|
|
}));
|
2021-02-03 15:46:48 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector
|
|
|
|
|
.set_load_local(clone!(@weak this => @default-panic, move || {
|
|
|
|
|
async move { this.handle.backend.db().get_works(&this.person.id).await.unwrap() }
|
|
|
|
|
}));
|
2021-02-03 15:46:48 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector
|
|
|
|
|
.set_make_widget(clone!(@weak this => @default-panic, move |work| {
|
2021-05-09 11:12:19 +02:00
|
|
|
let row = adw::ActionRow::new();
|
2021-05-07 23:49:05 +02:00
|
|
|
row.set_activatable(true);
|
2021-10-10 10:24:18 +02:00
|
|
|
row.set_title(&work.title);
|
2021-01-25 14:00:57 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
let work = work.to_owned();
|
|
|
|
|
row.connect_activated(clone!(@weak this => move |_| {
|
|
|
|
|
this.handle.pop(Some(work.clone()));
|
|
|
|
|
}));
|
2021-01-25 14:00:57 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
row.upcast()
|
|
|
|
|
}));
|
2020-12-02 15:41:19 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector
|
|
|
|
|
.set_filter(|search, work| work.title.to_lowercase().contains(search));
|
2020-12-02 15:41:19 +01:00
|
|
|
|
|
|
|
|
this
|
|
|
|
|
}
|
2021-02-03 15:46:48 +01:00
|
|
|
}
|
2020-12-02 15:41:19 +01:00
|
|
|
|
2021-02-03 15:46:48 +01:00
|
|
|
impl Widget for RecordingSelectorWorkScreen {
|
|
|
|
|
fn get_widget(&self) -> gtk::Widget {
|
|
|
|
|
self.selector.widget.clone().upcast()
|
2020-12-02 15:41:19 +01:00
|
|
|
}
|
2021-02-03 15:46:48 +01:00
|
|
|
}
|
2020-12-02 15:41:19 +01:00
|
|
|
|
2021-02-03 15:46:48 +01:00
|
|
|
/// The actual recording selector within the recording selector.
|
|
|
|
|
struct RecordingSelectorRecordingScreen {
|
|
|
|
|
handle: NavigationHandle<Recording>,
|
|
|
|
|
work: Work,
|
|
|
|
|
selector: Rc<Selector<Recording>>,
|
2020-12-02 15:41:19 +01:00
|
|
|
}
|
|
|
|
|
|
2021-02-03 15:46:48 +01:00
|
|
|
impl Screen<Work, Recording> for RecordingSelectorRecordingScreen {
|
|
|
|
|
fn new(work: Work, handle: NavigationHandle<Recording>) -> Rc<Self> {
|
2021-05-07 20:44:27 +02:00
|
|
|
let selector = Selector::<Recording>::new(Rc::clone(&handle.backend));
|
2021-02-03 15:46:48 +01:00
|
|
|
selector.set_title(&gettext("Select recording"));
|
|
|
|
|
selector.set_subtitle(&work.get_title());
|
|
|
|
|
|
|
|
|
|
let this = Rc::new(Self {
|
|
|
|
|
handle,
|
|
|
|
|
work,
|
|
|
|
|
selector,
|
|
|
|
|
});
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_back_cb(clone!(@weak this => move || {
|
2021-02-03 15:46:48 +01:00
|
|
|
this.handle.pop(None);
|
|
|
|
|
}));
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_add_cb(clone!(@weak this => move || {
|
2021-02-03 15:46:48 +01:00
|
|
|
spawn!(@clone this, async move {
|
|
|
|
|
let recording = Recording::new(this.work.clone());
|
|
|
|
|
if let Some(recording) = push!(this.handle, RecordingEditor, Some(recording)).await {
|
|
|
|
|
this.handle.pop(Some(recording));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}));
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_load_online(clone!(@weak this => @default-panic, move || {
|
2021-02-04 21:47:22 +01:00
|
|
|
async move { Ok(this.handle.backend.cl().get_recordings_for_work(&this.work.id).await?) }
|
2021-02-03 15:46:48 +01:00
|
|
|
}));
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_load_local(clone!(@weak this => @default-panic, move || {
|
2021-02-03 15:46:48 +01:00
|
|
|
async move { this.handle.backend.db().get_recordings_for_work(&this.work.id).await.unwrap() }
|
|
|
|
|
}));
|
|
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector
|
|
|
|
|
.set_make_widget(clone!(@weak this => @default-panic, move |recording| {
|
2021-05-09 11:12:19 +02:00
|
|
|
let row = adw::ActionRow::new();
|
2021-05-07 23:49:05 +02:00
|
|
|
row.set_activatable(true);
|
2021-10-10 10:24:18 +02:00
|
|
|
row.set_title(&recording.get_performers());
|
2021-02-03 15:46:48 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
let recording = recording.to_owned();
|
|
|
|
|
row.connect_activated(clone!(@weak this => move |_| {
|
|
|
|
|
this.handle.pop(Some(recording.clone()));
|
|
|
|
|
}));
|
2021-02-03 15:46:48 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
row.upcast()
|
|
|
|
|
}));
|
2021-02-03 15:46:48 +01:00
|
|
|
|
2021-05-07 23:49:05 +02:00
|
|
|
this.selector.set_filter(|search, recording| {
|
|
|
|
|
recording.get_performers().to_lowercase().contains(search)
|
|
|
|
|
});
|
2021-02-03 15:46:48 +01:00
|
|
|
|
|
|
|
|
this
|
2020-12-02 15:41:19 +01:00
|
|
|
}
|
2021-02-03 15:46:48 +01:00
|
|
|
}
|
2020-12-02 15:41:19 +01:00
|
|
|
|
2021-02-03 15:46:48 +01:00
|
|
|
impl Widget for RecordingSelectorRecordingScreen {
|
2020-12-02 15:41:19 +01:00
|
|
|
fn get_widget(&self) -> gtk::Widget {
|
|
|
|
|
self.selector.widget.clone().upcast()
|
|
|
|
|
}
|
|
|
|
|
}
|