musicus/src/screens/person_screen.rs

209 lines
7.1 KiB
Rust
Raw Normal View History

use super::*;
2020-10-16 21:24:55 +02:00
use crate::backend::*;
use crate::database::*;
2020-12-02 15:41:19 +01:00
use crate::editors::PersonEditor;
use crate::widgets::{List, Navigator, NavigatorScreen, NavigatorWindow};
use gettextrs::gettext;
2020-11-17 16:48:21 +01:00
use gio::prelude::*;
2020-10-16 21:24:55 +02:00
use glib::clone;
use gtk::prelude::*;
use gtk_macros::get_widget;
use libhandy::HeaderBarExt;
use std::cell::RefCell;
use std::rc::Rc;
pub struct PersonScreen {
backend: Rc<Backend>,
2020-11-17 16:48:21 +01:00
person: Person,
widget: gtk::Box,
2020-10-16 21:24:55 +02:00
stack: gtk::Stack,
2020-11-17 15:52:47 +01:00
work_list: Rc<List<Work>>,
recording_list: Rc<List<Recording>>,
navigator: RefCell<Option<Rc<Navigator>>>,
2020-10-16 21:24:55 +02:00
}
impl PersonScreen {
2020-12-02 15:54:07 +01:00
pub fn new(backend: Rc<Backend>, person: Person) -> Rc<Self> {
let builder = gtk::Builder::from_resource("/de/johrpan/musicus/ui/person_screen.ui");
2020-10-16 21:24:55 +02:00
get_widget!(builder, gtk::Box, widget);
get_widget!(builder, libhandy::HeaderBar, header);
get_widget!(builder, gtk::Button, back_button);
get_widget!(builder, gtk::SearchEntry, search_entry);
get_widget!(builder, gtk::Stack, stack);
get_widget!(builder, gtk::Box, work_box);
get_widget!(builder, gtk::Frame, work_frame);
get_widget!(builder, gtk::Box, recording_box);
get_widget!(builder, gtk::Frame, recording_frame);
header.set_title(Some(&person.name_fl()));
2020-11-17 16:48:21 +01:00
let edit_action = gio::SimpleAction::new("edit", None);
let delete_action = gio::SimpleAction::new("delete", None);
2020-11-17 16:48:21 +01:00
let actions = gio::SimpleActionGroup::new();
actions.add_action(&edit_action);
actions.add_action(&delete_action);
2020-11-17 16:48:21 +01:00
widget.insert_action_group("widget", Some(&actions));
let work_list = List::new(&gettext("No works found."));
2020-11-17 15:52:47 +01:00
work_list.set_make_widget(|work: &Work| {
let label = gtk::Label::new(Some(&work.title));
label.set_halign(gtk::Align::Start);
label.set_margin_start(6);
label.set_margin_end(6);
label.set_margin_top(6);
label.set_margin_bottom(6);
label.upcast()
});
2020-11-17 15:52:47 +01:00
work_list.set_filter(clone!(@strong search_entry => move |work: &Work| {
let search = search_entry.get_text().to_string().to_lowercase();
let title = work.title.to_lowercase();
search.is_empty() || title.contains(&search)
}));
2020-10-16 21:24:55 +02:00
let recording_list = List::new(&gettext("No recordings found."));
2020-11-17 15:52:47 +01:00
recording_list.set_make_widget(|recording: &Recording| {
let work_label = gtk::Label::new(Some(&recording.work.get_title()));
2020-10-16 21:24:55 +02:00
work_label.set_ellipsize(pango::EllipsizeMode::End);
work_label.set_halign(gtk::Align::Start);
2020-10-16 21:24:55 +02:00
let performers_label = gtk::Label::new(Some(&recording.get_performers()));
performers_label.set_ellipsize(pango::EllipsizeMode::End);
performers_label.set_opacity(0.5);
performers_label.set_halign(gtk::Align::Start);
2020-10-16 21:24:55 +02:00
let vbox = gtk::Box::new(gtk::Orientation::Vertical, 0);
vbox.set_border_width(6);
vbox.add(&work_label);
vbox.add(&performers_label);
vbox.upcast()
});
2020-10-16 21:24:55 +02:00
recording_list.set_filter(
2020-11-17 15:52:47 +01:00
clone!(@strong search_entry => move |recording: &Recording| {
2020-10-16 21:24:55 +02:00
let search = search_entry.get_text().to_string().to_lowercase();
let text = recording.work.get_title() + &recording.get_performers();
search.is_empty() || text.contains(&search)
}),
);
work_frame.add(&work_list.widget);
recording_frame.add(&recording_list.widget);
let result = Rc::new(Self {
backend,
2020-11-17 16:48:21 +01:00
person,
2020-10-16 21:24:55 +02:00
widget,
stack,
work_list,
recording_list,
navigator: RefCell::new(None),
2020-10-16 21:24:55 +02:00
});
search_entry.connect_search_changed(clone!(@strong result => move |_| {
result.work_list.invalidate_filter();
result.recording_list.invalidate_filter();
}));
back_button.connect_clicked(clone!(@strong result => move |_| {
let navigator = result.navigator.borrow().clone();
if let Some(navigator) = navigator {
navigator.clone().pop();
2020-10-16 21:24:55 +02:00
}
}));
result
.work_list
.set_selected(clone!(@strong result => move |work| {
result.recording_list.clear_selection();
let navigator = result.navigator.borrow().clone();
if let Some(navigator) = navigator {
2020-12-02 15:54:07 +01:00
navigator.push(WorkScreen::new(result.backend.clone(), work.clone()));
}
}));
result
.recording_list
.set_selected(clone!(@strong result => move |recording| {
result.work_list.clear_selection();
let navigator = result.navigator.borrow().clone();
if let Some(navigator) = navigator {
2020-12-02 15:54:07 +01:00
navigator.push(RecordingScreen::new(result.backend.clone(), recording.clone()));
}
}));
2020-11-17 16:48:21 +01:00
edit_action.connect_activate(clone!(@strong result => move |_, _| {
2020-12-02 15:41:19 +01:00
let editor = PersonEditor::new(result.backend.clone(), Some(result.person.clone()));
let window = NavigatorWindow::new(editor);
window.show();
2020-11-17 16:48:21 +01:00
}));
delete_action.connect_activate(clone!(@strong result => move |_, _| {
let context = glib::MainContext::default();
let clone = result.clone();
context.spawn_local(async move {
2020-11-28 21:45:22 +01:00
clone.backend.db().delete_person(&clone.person.id).await.unwrap();
2020-11-28 22:23:35 +01:00
clone.backend.library_changed();
2020-11-17 16:48:21 +01:00
});
}));
2020-10-16 21:24:55 +02:00
let context = glib::MainContext::default();
let clone = result.clone();
context.spawn_local(async move {
let works = clone
.backend
2020-11-17 15:52:47 +01:00
.db()
2020-11-28 21:45:22 +01:00
.get_works(&clone.person.id)
.await
.unwrap();
let recordings = clone
.backend
2020-11-17 15:52:47 +01:00
.db()
2020-11-28 21:45:22 +01:00
.get_recordings_for_person(&clone.person.id)
.await
.unwrap();
2020-10-16 21:24:55 +02:00
if works.is_empty() && recordings.is_empty() {
clone.stack.set_visible_child_name("nothing");
} else {
if works.is_empty() {
work_box.hide();
} else {
clone.work_list.show_items(works);
}
if recordings.is_empty() {
recording_box.hide();
} else {
clone.recording_list.show_items(recordings);
}
clone.stack.set_visible_child_name("content");
}
});
result
}
}
2020-10-16 21:24:55 +02:00
impl NavigatorScreen for PersonScreen {
fn attach_navigator(&self, navigator: Rc<Navigator>) {
self.navigator.replace(Some(navigator));
2020-10-16 21:24:55 +02:00
}
fn get_widget(&self) -> gtk::Widget {
self.widget.clone().upcast()
2020-10-16 21:24:55 +02:00
}
fn detach_navigator(&self) {
self.navigator.replace(None);
2020-10-16 21:24:55 +02:00
}
}