Enable main search bar

This commit is contained in:
Elias Projahn 2020-10-10 11:32:30 +02:00
parent caf7681322
commit 9101fb053d
2 changed files with 74 additions and 27 deletions

View file

@ -432,6 +432,17 @@
<object class="GtkListBox" id="work_list"> <object class="GtkListBox" id="work_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">
<object class="GtkLabel">
<property name="visible">True</property>
<property name="can-focus">False</property>
<property name="margin_top">6</property>
<property name="margin_bottom">6</property>
<property name="margin_start">6</property>
<property name="margin_end">6</property>
<property name="label" translatable="yes">No works found.</property>
</object>
</child>
</object> </object>
</child> </child>
</object> </object>
@ -480,6 +491,17 @@
<object class="GtkListBox" id="recording_list"> <object class="GtkListBox" id="recording_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">
<object class="GtkLabel">
<property name="visible">True</property>
<property name="can-focus">False</property>
<property name="margin_top">6</property>
<property name="margin_bottom">6</property>
<property name="margin_start">6</property>
<property name="margin_end">6</property>
<property name="label" translatable="yes">No recordings found.</property>
</object>
</child>
</object> </object>
</child> </child>
</object> </object>

View file

@ -7,19 +7,21 @@ use gtk::prelude::*;
use gtk_macros::{action, get_widget}; use gtk_macros::{action, get_widget};
use libhandy::prelude::*; use libhandy::prelude::*;
use libhandy::HeaderBarExt; use libhandy::HeaderBarExt;
use std::cell::Cell; use std::cell::{Cell, RefCell};
use std::convert::TryInto; use std::convert::TryInto;
use std::rc::Rc; use std::rc::Rc;
#[derive(Clone)]
enum WindowState { enum WindowState {
Loading, Loading,
Persons(Vec<Person>), Persons(Vec<Person>),
PersonLoading(Person), PersonLoading(Person),
Person(Vec<WorkDescription>, Vec<RecordingDescription>), Person(Vec<WorkDescription>, Vec<RecordingDescription>, String),
} }
pub struct Window { pub struct Window {
window: libhandy::ApplicationWindow, window: libhandy::ApplicationWindow,
state: RefCell<WindowState>,
backend: Rc<Backend>, backend: Rc<Backend>,
leaflet: libhandy::Leaflet, leaflet: libhandy::Leaflet,
sidebar_stack: gtk::Stack, sidebar_stack: gtk::Stack,
@ -28,6 +30,7 @@ pub struct Window {
stack: gtk::Stack, stack: gtk::Stack,
header: libhandy::HeaderBar, header: libhandy::HeaderBar,
header_menu_button: gtk::MenuButton, header_menu_button: gtk::MenuButton,
search_entry: gtk::SearchEntry,
content_stack: gtk::Stack, content_stack: gtk::Stack,
work_box: gtk::Box, work_box: gtk::Box,
work_list: gtk::ListBox, work_list: gtk::ListBox,
@ -57,6 +60,7 @@ impl Window {
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);
get_widget!(builder, gtk::SearchEntry, search_entry);
get_widget!(builder, gtk::Stack, content_stack); get_widget!(builder, gtk::Stack, content_stack);
get_widget!(builder, gtk::Box, work_box); get_widget!(builder, gtk::Box, work_box);
get_widget!(builder, gtk::ListBox, work_list); get_widget!(builder, gtk::ListBox, work_list);
@ -70,6 +74,7 @@ impl Window {
let result = Rc::new(Window { let result = Rc::new(Window {
window: window, window: window,
state: RefCell::new(Loading),
backend: Rc::new(backend), backend: Rc::new(backend),
leaflet: leaflet, leaflet: leaflet,
sidebar_stack: sidebar_stack, sidebar_stack: sidebar_stack,
@ -78,6 +83,7 @@ impl Window {
stack: stack, stack: stack,
header: header, header: header,
header_menu_button: header_menu_button, header_menu_button: header_menu_button,
search_entry: search_entry,
content_stack: content_stack, content_stack: content_stack,
work_box: work_box, work_box: work_box,
work_list: work_list, work_list: work_list,
@ -182,6 +188,15 @@ impl Window {
result.person_list.invalidate_filter(); result.person_list.invalidate_filter();
})); }));
result.search_entry.connect_search_changed(clone!(@strong result => move |_| {
match result.get_state() {
Person(works, recordings, _) => {
result.clone().set_state(Person(works.clone(), recordings.clone(), result.search_entry.get_text().to_string()));
}
_ => (),
}
}));
result.window.set_application(Some(app)); result.window.set_application(Some(app));
result.clone().set_state(Loading); result.clone().set_state(Loading);
@ -192,9 +207,15 @@ impl Window {
self.window.present(); self.window.present();
} }
fn get_state(&self) -> WindowState {
self.state.borrow().clone()
}
fn set_state(self: Rc<Self>, state: WindowState) { fn set_state(self: Rc<Self>, state: WindowState) {
use WindowState::*; use WindowState::*;
self.state.replace(state.clone());
match state { match state {
Loading => { Loading => {
self.backend self.backend
@ -259,6 +280,7 @@ impl Window {
} }
PersonLoading(person) => { PersonLoading(person) => {
self.header.set_title(Some(&person.name_fl())); self.header.set_title(Some(&person.name_fl()));
self.search_entry.set_text("");
let edit_menu_item = gio::MenuItem::new(Some("Edit person"), None); let edit_menu_item = gio::MenuItem::new(Some("Edit person"), None);
edit_menu_item.set_action_and_target_value( edit_menu_item.set_action_and_target_value(
@ -284,7 +306,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)); self_.clone().set_state(Person(works.clone(), recordings, String::from("")));
}), }),
); );
}), }),
@ -295,7 +317,7 @@ 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) => { Person(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);
} }
@ -311,6 +333,7 @@ impl Window {
} }
for (index, work) in works.iter().enumerate() { for (index, work) in works.iter().enumerate() {
if search.is_empty() || work.title.to_lowercase().contains(&search) {
let label = gtk::Label::new(Some(&work.title)); let label = gtk::Label::new(Some(&work.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);
@ -318,6 +341,7 @@ impl Window {
row.show_all(); row.show_all();
self.work_list.insert(&row, -1); self.work_list.insert(&row, -1);
} }
}
match self.work_list_row_activated_handler_id.take() { match self.work_list_row_activated_handler_id.take() {
Some(id) => self.work_list.disconnect(id), Some(id) => self.work_list.disconnect(id),
@ -376,16 +400,16 @@ impl Window {
} }
for (index, recording) in recordings.iter().enumerate() { for (index, recording) in recordings.iter().enumerate() {
let work_label = gtk::Label::new(Some(&format!( let work_text = format!("{}: {}", recording.work.composer.name_fl(), recording.work.title);
"{}: {}", let performers_text = recording.get_performers();
recording.work.composer.name_fl(),
recording.work.title if search.is_empty() || (work_text.to_lowercase().contains(&search) || performers_text.to_lowercase().contains(&search)) {
))); let work_label = gtk::Label::new(Some(&work_text));
work_label.set_ellipsize(pango::EllipsizeMode::End); work_label.set_ellipsize(pango::EllipsizeMode::End);
work_label.set_halign(gtk::Align::Start); work_label.set_halign(gtk::Align::Start);
let performers_label = gtk::Label::new(Some(&recording.get_performers())); let performers_label = gtk::Label::new(Some(&performers_text));
performers_label.set_ellipsize(pango::EllipsizeMode::End); performers_label.set_ellipsize(pango::EllipsizeMode::End);
performers_label.set_opacity(0.5); performers_label.set_opacity(0.5);
performers_label.set_halign(gtk::Align::Start); performers_label.set_halign(gtk::Align::Start);
@ -398,6 +422,7 @@ impl Window {
row.show_all(); row.show_all();
self.recording_list.insert(&row, -1); self.recording_list.insert(&row, -1);
} }
}
match self.recording_list_row_activated_handler_id.take() { match self.recording_list_row_activated_handler_id.take() {
Some(id) => self.recording_list.disconnect(id), Some(id) => self.recording_list.disconnect(id),