mirror of
https://github.com/johrpan/musicus.git
synced 2025-10-26 19:57:25 +01:00
Show persons and ensembles in main window
This commit is contained in:
parent
9101fb053d
commit
cca722dcba
4 changed files with 157 additions and 27 deletions
|
|
@ -121,7 +121,7 @@
|
||||||
<property name="can-focus">False</property>
|
<property name="can-focus">False</property>
|
||||||
<property name="shadow-type">none</property>
|
<property name="shadow-type">none</property>
|
||||||
<child>
|
<child>
|
||||||
<object class="GtkListBox" id="person_list">
|
<object class="GtkListBox" id="sidebar_list">
|
||||||
<property name="visible">True</property>
|
<property name="visible">True</property>
|
||||||
<property name="can-focus">False</property>
|
<property name="can-focus">False</property>
|
||||||
<child type="placeholder">
|
<child type="placeholder">
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,7 @@ enum BackendAction {
|
||||||
GetEnsembles(Sender<Vec<Ensemble>>),
|
GetEnsembles(Sender<Vec<Ensemble>>),
|
||||||
UpdateRecording(RecordingInsertion, Sender<Result<(), String>>),
|
UpdateRecording(RecordingInsertion, Sender<Result<(), String>>),
|
||||||
GetRecordingsForPerson(i64, Sender<Vec<RecordingDescription>>),
|
GetRecordingsForPerson(i64, Sender<Vec<RecordingDescription>>),
|
||||||
|
GetRecordingsForEnsemble(i64, Sender<Vec<RecordingDescription>>),
|
||||||
}
|
}
|
||||||
|
|
||||||
use BackendAction::*;
|
use BackendAction::*;
|
||||||
|
|
@ -133,6 +134,12 @@ impl Backend {
|
||||||
.send(recordings)
|
.send(recordings)
|
||||||
.expect("Failed to send result from database thread!");
|
.expect("Failed to send result from database thread!");
|
||||||
}
|
}
|
||||||
|
GetRecordingsForEnsemble(id, sender) => {
|
||||||
|
let recordings = db.get_recordings_for_ensemble(id);
|
||||||
|
sender
|
||||||
|
.send(recordings)
|
||||||
|
.expect("Failed to send result from database thread!");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
@ -396,4 +403,22 @@ impl Backend {
|
||||||
.send(GetRecordingsForPerson(id, sender))
|
.send(GetRecordingsForPerson(id, sender))
|
||||||
.expect("Failed to send action to database thread!");
|
.expect("Failed to send action to database thread!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn get_recordings_for_ensemble<F: Fn(Vec<RecordingDescription>) -> () + 'static>(
|
||||||
|
&self,
|
||||||
|
id: i64,
|
||||||
|
callback: F,
|
||||||
|
) {
|
||||||
|
let (sender, receiver) =
|
||||||
|
glib::MainContext::channel::<Vec<RecordingDescription>>(glib::PRIORITY_DEFAULT);
|
||||||
|
|
||||||
|
receiver.attach(None, move |result| {
|
||||||
|
callback(result);
|
||||||
|
glib::Continue(true)
|
||||||
|
});
|
||||||
|
|
||||||
|
self.action_sender
|
||||||
|
.send(GetRecordingsForEnsemble(id, sender))
|
||||||
|
.expect("Failed to send action to database thread!");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -349,6 +349,21 @@ impl Database {
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn get_recordings_for_ensemble(&self, id: i64) -> Vec<RecordingDescription> {
|
||||||
|
let recordings = recordings::table
|
||||||
|
.inner_join(performances::table.on(performances::recording.eq(recordings::id)))
|
||||||
|
.inner_join(ensembles::table.on(ensembles::id.nullable().eq(performances::ensemble)))
|
||||||
|
.filter(ensembles::id.eq(id))
|
||||||
|
.select(recordings::table::all_columns())
|
||||||
|
.load::<Recording>(&self.c)
|
||||||
|
.expect("Error loading recordings for ensemble!");
|
||||||
|
|
||||||
|
recordings
|
||||||
|
.iter()
|
||||||
|
.map(|recording| self.get_recording_description_for_recording(recording.clone()))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn delete_recording(&self, id: i64) {
|
pub fn delete_recording(&self, id: i64) {
|
||||||
diesel::delete(recordings::table.filter(recordings::id.eq(id)))
|
diesel::delete(recordings::table.filter(recordings::id.eq(id)))
|
||||||
.execute(&self.c)
|
.execute(&self.c)
|
||||||
|
|
|
||||||
142
src/window.rs
142
src/window.rs
|
|
@ -11,12 +11,28 @@ use std::cell::{Cell, RefCell};
|
||||||
use std::convert::TryInto;
|
use std::convert::TryInto;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
enum PersonOrEnsemble {
|
||||||
|
Person(Person),
|
||||||
|
Ensemble(Ensemble),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PersonOrEnsemble {
|
||||||
|
pub fn get_title(&self) -> String {
|
||||||
|
match self {
|
||||||
|
PersonOrEnsemble::Person(person) => person.name_lf(),
|
||||||
|
PersonOrEnsemble::Ensemble(ensemble) => ensemble.name.clone(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
enum WindowState {
|
enum WindowState {
|
||||||
Loading,
|
Loading,
|
||||||
Persons(Vec<Person>),
|
Selection(Vec<PersonOrEnsemble>),
|
||||||
PersonLoading(Person),
|
PersonLoading(Person),
|
||||||
Person(Vec<WorkDescription>, Vec<RecordingDescription>, String),
|
EnsembleLoading(Ensemble),
|
||||||
|
Selected(Vec<WorkDescription>, Vec<RecordingDescription>, String),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Window {
|
pub struct Window {
|
||||||
|
|
@ -26,7 +42,7 @@ pub struct Window {
|
||||||
leaflet: libhandy::Leaflet,
|
leaflet: libhandy::Leaflet,
|
||||||
sidebar_stack: gtk::Stack,
|
sidebar_stack: gtk::Stack,
|
||||||
person_search_entry: gtk::SearchEntry,
|
person_search_entry: gtk::SearchEntry,
|
||||||
person_list: gtk::ListBox,
|
sidebar_list: gtk::ListBox,
|
||||||
stack: gtk::Stack,
|
stack: gtk::Stack,
|
||||||
header: libhandy::HeaderBar,
|
header: libhandy::HeaderBar,
|
||||||
header_menu_button: gtk::MenuButton,
|
header_menu_button: gtk::MenuButton,
|
||||||
|
|
@ -56,7 +72,7 @@ impl Window {
|
||||||
get_widget!(builder, libhandy::Leaflet, leaflet);
|
get_widget!(builder, libhandy::Leaflet, leaflet);
|
||||||
get_widget!(builder, gtk::SearchEntry, person_search_entry);
|
get_widget!(builder, gtk::SearchEntry, person_search_entry);
|
||||||
get_widget!(builder, gtk::Stack, sidebar_stack);
|
get_widget!(builder, gtk::Stack, sidebar_stack);
|
||||||
get_widget!(builder, gtk::ListBox, person_list);
|
get_widget!(builder, gtk::ListBox, sidebar_list);
|
||||||
get_widget!(builder, gtk::Stack, stack);
|
get_widget!(builder, gtk::Stack, stack);
|
||||||
get_widget!(builder, libhandy::HeaderBar, header);
|
get_widget!(builder, libhandy::HeaderBar, header);
|
||||||
get_widget!(builder, gtk::MenuButton, header_menu_button);
|
get_widget!(builder, gtk::MenuButton, header_menu_button);
|
||||||
|
|
@ -78,7 +94,7 @@ impl Window {
|
||||||
backend: Rc::new(backend),
|
backend: Rc::new(backend),
|
||||||
leaflet: leaflet,
|
leaflet: leaflet,
|
||||||
sidebar_stack: sidebar_stack,
|
sidebar_stack: sidebar_stack,
|
||||||
person_list: person_list,
|
sidebar_list: sidebar_list,
|
||||||
person_search_entry: person_search_entry,
|
person_search_entry: person_search_entry,
|
||||||
stack: stack,
|
stack: stack,
|
||||||
header: header,
|
header: header,
|
||||||
|
|
@ -182,16 +198,41 @@ impl Window {
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
|
action!(
|
||||||
|
result.window,
|
||||||
|
"edit-ensemble",
|
||||||
|
Some(glib::VariantTy::new("x").unwrap()),
|
||||||
|
clone!(@strong result => move |_, id| {
|
||||||
|
result.backend.get_ensemble(id.unwrap().get().unwrap(), clone!(@strong result => move |ensemble| {
|
||||||
|
let ensemble = ensemble.unwrap();
|
||||||
|
EnsembleEditor::new(result.backend.clone(), &result.window, Some(ensemble), clone!(@strong result => move |_| {
|
||||||
|
result.clone().set_state(Loading);
|
||||||
|
})).show();
|
||||||
|
}));
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
action!(
|
||||||
|
result.window,
|
||||||
|
"delete-ensemble",
|
||||||
|
Some(glib::VariantTy::new("x").unwrap()),
|
||||||
|
clone!(@strong result => move |_, id| {
|
||||||
|
result.backend.delete_ensemble(id.unwrap().get().unwrap(), clone!(@strong result => move |_| {
|
||||||
|
result.clone().set_state(Loading);
|
||||||
|
}));
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
result
|
result
|
||||||
.person_search_entry
|
.person_search_entry
|
||||||
.connect_search_changed(clone!(@strong result => move |_| {
|
.connect_search_changed(clone!(@strong result => move |_| {
|
||||||
result.person_list.invalidate_filter();
|
result.sidebar_list.invalidate_filter();
|
||||||
}));
|
}));
|
||||||
|
|
||||||
result.search_entry.connect_search_changed(clone!(@strong result => move |_| {
|
result.search_entry.connect_search_changed(clone!(@strong result => move |_| {
|
||||||
match result.get_state() {
|
match result.get_state() {
|
||||||
Person(works, recordings, _) => {
|
Selected(works, recordings, _) => {
|
||||||
result.clone().set_state(Person(works.clone(), recordings.clone(), result.search_entry.get_text().to_string()));
|
result.clone().set_state(Selected(works.clone(), recordings.clone(), result.search_entry.get_text().to_string()));
|
||||||
}
|
}
|
||||||
_ => (),
|
_ => (),
|
||||||
}
|
}
|
||||||
|
|
@ -220,7 +261,19 @@ impl Window {
|
||||||
Loading => {
|
Loading => {
|
||||||
self.backend
|
self.backend
|
||||||
.get_persons(clone!(@strong self as self_ => move |persons| {
|
.get_persons(clone!(@strong self as self_ => move |persons| {
|
||||||
self_.clone().set_state(Persons(persons));
|
self_.backend.get_ensembles(clone!(@strong self_ => move |ensembles| {
|
||||||
|
let mut poes: Vec<PersonOrEnsemble> = Vec::new();
|
||||||
|
|
||||||
|
for person in &persons {
|
||||||
|
poes.push(PersonOrEnsemble::Person(person.clone()));
|
||||||
|
}
|
||||||
|
|
||||||
|
for ensemble in &ensembles {
|
||||||
|
poes.push(PersonOrEnsemble::Ensemble(ensemble.clone()));
|
||||||
|
}
|
||||||
|
|
||||||
|
self_.clone().set_state(Selection(poes));
|
||||||
|
}));
|
||||||
}));
|
}));
|
||||||
|
|
||||||
self.actions_revealer.set_reveal_child(false);
|
self.actions_revealer.set_reveal_child(false);
|
||||||
|
|
@ -228,45 +281,48 @@ impl Window {
|
||||||
self.stack.set_visible_child_name("empty_screen");
|
self.stack.set_visible_child_name("empty_screen");
|
||||||
self.leaflet.set_visible_child_name("sidebar");
|
self.leaflet.set_visible_child_name("sidebar");
|
||||||
}
|
}
|
||||||
Persons(persons) => {
|
Selection(poes) => {
|
||||||
for child in self.person_list.get_children() {
|
for child in self.sidebar_list.get_children() {
|
||||||
self.person_list.remove(&child);
|
self.sidebar_list.remove(&child);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (index, person) in persons.iter().enumerate() {
|
for (index, poe) in poes.iter().enumerate() {
|
||||||
let label = gtk::Label::new(Some(&person.name_lf()));
|
let label = gtk::Label::new(Some(&poe.get_title()));
|
||||||
label.set_ellipsize(pango::EllipsizeMode::End);
|
label.set_ellipsize(pango::EllipsizeMode::End);
|
||||||
label.set_halign(gtk::Align::Start);
|
label.set_halign(gtk::Align::Start);
|
||||||
let row = SelectorRow::new(index.try_into().unwrap(), &label);
|
let row = SelectorRow::new(index.try_into().unwrap(), &label);
|
||||||
row.show_all();
|
row.show_all();
|
||||||
self.person_list.insert(&row, -1);
|
self.sidebar_list.insert(&row, -1);
|
||||||
}
|
}
|
||||||
|
|
||||||
match self.person_list_row_activated_handler_id.take() {
|
match self.person_list_row_activated_handler_id.take() {
|
||||||
Some(id) => self.person_list.disconnect(id),
|
Some(id) => self.sidebar_list.disconnect(id),
|
||||||
None => (),
|
None => (),
|
||||||
}
|
}
|
||||||
|
|
||||||
let handler_id = self.person_list.connect_row_activated(
|
let handler_id = self.sidebar_list.connect_row_activated(
|
||||||
clone!(@strong self as self_, @strong persons => move |_, row| {
|
clone!(@strong self as self_, @strong poes => move |_, row| {
|
||||||
let row = row.get_child().unwrap().downcast::<SelectorRow>().unwrap();
|
let row = row.get_child().unwrap().downcast::<SelectorRow>().unwrap();
|
||||||
let index: usize = row.get_index().try_into().unwrap();
|
let index: usize = row.get_index().try_into().unwrap();
|
||||||
let person = persons[index].clone();
|
let poe = poes[index].clone();
|
||||||
self_.clone().set_state(PersonLoading(person));
|
self_.clone().set_state(match poe {
|
||||||
|
PersonOrEnsemble::Person(person) => PersonLoading(person),
|
||||||
|
PersonOrEnsemble::Ensemble(ensemble) => EnsembleLoading(ensemble),
|
||||||
|
});
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
self.person_list_row_activated_handler_id
|
self.person_list_row_activated_handler_id
|
||||||
.set(Some(handler_id));
|
.set(Some(handler_id));
|
||||||
|
|
||||||
self.person_list.set_filter_func(Some(Box::new(
|
self.sidebar_list.set_filter_func(Some(Box::new(
|
||||||
clone!(@strong self as self_, @strong persons => move |row| {
|
clone!(@strong self as self_, @strong poes => move |row| {
|
||||||
let row = row.get_child().unwrap().downcast::<SelectorRow>().unwrap();
|
let row = row.get_child().unwrap().downcast::<SelectorRow>().unwrap();
|
||||||
let index: usize = row.get_index().try_into().unwrap();
|
let index: usize = row.get_index().try_into().unwrap();
|
||||||
let search = self_.person_search_entry.get_text().to_string().to_lowercase();
|
let search = self_.person_search_entry.get_text().to_string().to_lowercase();
|
||||||
|
|
||||||
search.is_empty() || persons[index]
|
search.is_empty() || poes[index]
|
||||||
.name_lf()
|
.get_title()
|
||||||
.to_lowercase()
|
.to_lowercase()
|
||||||
.contains(&search)
|
.contains(&search)
|
||||||
}),
|
}),
|
||||||
|
|
@ -306,7 +362,7 @@ impl Window {
|
||||||
self_.backend.get_recordings_for_person(
|
self_.backend.get_recordings_for_person(
|
||||||
person.id,
|
person.id,
|
||||||
clone!(@strong self_ => move |recordings| {
|
clone!(@strong self_ => move |recordings| {
|
||||||
self_.clone().set_state(Person(works.clone(), recordings, String::from("")));
|
self_.clone().set_state(Selected(works.clone(), recordings, String::from("")));
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
}),
|
}),
|
||||||
|
|
@ -317,7 +373,41 @@ impl Window {
|
||||||
self.stack.set_visible_child_name("person_screen");
|
self.stack.set_visible_child_name("person_screen");
|
||||||
self.leaflet.set_visible_child_name("content");
|
self.leaflet.set_visible_child_name("content");
|
||||||
}
|
}
|
||||||
Person(works, recordings, search) => {
|
EnsembleLoading(ensemble) => {
|
||||||
|
self.header.set_title(Some(&ensemble.name));
|
||||||
|
self.search_entry.set_text("");
|
||||||
|
|
||||||
|
let edit_menu_item = gio::MenuItem::new(Some("Edit ensemble"), None);
|
||||||
|
edit_menu_item.set_action_and_target_value(
|
||||||
|
Some("win.edit-ensemble"),
|
||||||
|
Some(&glib::Variant::from(ensemble.id)),
|
||||||
|
);
|
||||||
|
|
||||||
|
let delete_menu_item = gio::MenuItem::new(Some("Delete ensemble"), None);
|
||||||
|
delete_menu_item.set_action_and_target_value(
|
||||||
|
Some("win.delete-ensemble"),
|
||||||
|
Some(&glib::Variant::from(ensemble.id)),
|
||||||
|
);
|
||||||
|
|
||||||
|
let menu = gio::Menu::new();
|
||||||
|
menu.append_item(&edit_menu_item);
|
||||||
|
menu.append_item(&delete_menu_item);
|
||||||
|
|
||||||
|
self.header_menu_button.set_menu_model(Some(&menu));
|
||||||
|
|
||||||
|
self.backend.get_recordings_for_ensemble(
|
||||||
|
ensemble.id,
|
||||||
|
clone!(@strong self as self_ => move |recordings| {
|
||||||
|
self_.clone().set_state(Selected(Vec::new(), recordings, String::from("")));
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
self.actions_revealer.set_reveal_child(false);
|
||||||
|
self.content_stack.set_visible_child_name("loading");
|
||||||
|
self.stack.set_visible_child_name("person_screen");
|
||||||
|
self.leaflet.set_visible_child_name("content");
|
||||||
|
}
|
||||||
|
Selected(works, recordings, search) => {
|
||||||
for child in self.work_list.get_children() {
|
for child in self.work_list.get_children() {
|
||||||
self.work_list.remove(&child);
|
self.work_list.remove(&child);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue