musicus/src/selectors/recording.rs

234 lines
7.8 KiB
Rust
Raw Normal View History

2020-12-02 15:41:19 +01:00
use super::selector::Selector;
2021-02-03 23:03:47 +01:00
use crate::backend::{Backend, Person, Work, Recording};
2021-02-03 15:46:48 +01:00
use crate::editors::{PersonEditor, WorkEditor, RecordingEditor};
use crate::navigator::{NavigationHandle, Screen};
use crate::widgets::Widget;
2020-12-02 15:41:19 +01:00
use gettextrs::gettext;
use glib::clone;
use gtk::prelude::*;
2021-01-29 15:17:27 +01:00
use libadwaita::prelude::*;
2020-12-02 15:41:19 +01:00
use std::cell::RefCell;
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-02-03 15:46:48 +01:00
let selector = Selector::<Person>::new();
selector.set_title(&gettext("Select composer"));
2020-12-02 15:41:19 +01:00
let this = Rc::new(Self {
2021-02-03 15:46:48 +01:00
handle,
2020-12-02 15:41:19 +01:00
selector,
});
// Connect signals and callbacks
2021-02-03 15:46:48 +01:00
this.selector.set_back_cb(clone!(@weak this => move || {
this.handle.pop(None);
2020-12-02 15:41:19 +01:00
}));
2021-02-03 15:46:48 +01:00
this.selector.set_add_cb(clone!(@weak this => move || {
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-02-03 15:46:48 +01:00
this.selector.set_load_online(clone!(@weak this => move || {
async move { this.handle.backend.get_persons().await }
2020-12-02 15:41:19 +01:00
}));
2021-02-03 15:46:48 +01:00
this.selector.set_load_local(clone!(@weak this => move || {
async move { this.handle.backend.db().get_persons().await.unwrap() }
}));
this.selector.set_make_widget(clone!(@weak this => move |person| {
let row = libadwaita::ActionRow::new();
row.set_activatable(true);
row.set_title(Some(&person.name_lf()));
let person = person.to_owned();
row.connect_activated(clone!(@weak this => move |_| {
// 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> {
let selector = Selector::<Work>::new();
selector.set_title(&gettext("Select work"));
selector.set_subtitle(&person.name_fl());
let this = Rc::new(Self {
handle,
person,
selector,
});
this.selector.set_back_cb(clone!(@weak this => move || {
this.handle.pop(None);
}));
this.selector.set_add_cb(clone!(@weak this => move || {
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));
}
});
}));
this.selector.set_load_online(clone!(@weak this => move || {
async move { this.handle.backend.get_works(&this.person.id).await }
}));
this.selector.set_load_local(clone!(@weak this => move || {
async move { this.handle.backend.db().get_works(&this.person.id).await.unwrap() }
}));
this.selector.set_make_widget(clone!(@weak this => move |work| {
2021-01-29 15:17:27 +01:00
let row = libadwaita::ActionRow::new();
2021-01-25 14:00:57 +01:00
row.set_activatable(true);
2021-02-03 15:46:48 +01:00
row.set_title(Some(&work.title));
2021-01-25 14:00:57 +01:00
2021-02-03 15:46:48 +01: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
}));
row.upcast()
}));
2020-12-02 15:41:19 +01:00
2021-02-03 15:46:48 +01: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> {
let selector = Selector::<Recording>::new();
selector.set_title(&gettext("Select recording"));
selector.set_subtitle(&work.get_title());
let this = Rc::new(Self {
handle,
work,
selector,
});
this.selector.set_back_cb(clone!(@weak this => move || {
this.handle.pop(None);
}));
this.selector.set_add_cb(clone!(@weak this => move || {
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));
}
});
}));
this.selector.set_load_online(clone!(@weak this => move || {
async move { this.handle.backend.get_recordings_for_work(&this.work.id).await }
}));
this.selector.set_load_local(clone!(@weak this => move || {
async move { this.handle.backend.db().get_recordings_for_work(&this.work.id).await.unwrap() }
}));
this.selector.set_make_widget(clone!(@weak this => move |recording| {
let row = libadwaita::ActionRow::new();
row.set_activatable(true);
row.set_title(Some(&recording.get_performers()));
let recording = recording.to_owned();
row.connect_activated(clone!(@weak this => move |_| {
this.handle.pop(Some(recording.clone()));
}));
row.upcast()
}));
this.selector
.set_filter(|search, recording| recording.get_performers().to_lowercase().contains(search));
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()
}
}