| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  | use std::cell::{OnceCell, RefCell};
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | use gettextrs::gettext;
 | 
					
						
							|  |  |  | use gtk::{
 | 
					
						
							|  |  |  |     glib::{self, subclass::Signal, Properties},
 | 
					
						
							|  |  |  |     prelude::*,
 | 
					
						
							|  |  |  |     subclass::prelude::*,
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | use once_cell::sync::Lazy;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-01 15:52:59 +01:00
										 |  |  | use crate::{db::models::Ensemble, library::Library, util::activatable_row::ActivatableRow};
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | mod imp {
 | 
					
						
							|  |  |  |     use super::*;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[derive(Debug, Default, gtk::CompositeTemplate, Properties)]
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |     #[properties(wrapper_type = super::EnsembleSelectorPopover)]
 | 
					
						
							|  |  |  |     #[template(file = "data/ui/selector/ensemble.blp")]
 | 
					
						
							|  |  |  |     pub struct EnsembleSelectorPopover {
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         #[property(get, construct_only)]
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |         pub library: OnceCell<Library>,
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         pub ensembles: RefCell<Vec<Ensemble>>,
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         #[template_child]
 | 
					
						
							|  |  |  |         pub search_entry: TemplateChild<gtk::SearchEntry>,
 | 
					
						
							|  |  |  |         #[template_child]
 | 
					
						
							|  |  |  |         pub scrolled_window: TemplateChild<gtk::ScrolledWindow>,
 | 
					
						
							|  |  |  |         #[template_child]
 | 
					
						
							|  |  |  |         pub list_box: TemplateChild<gtk::ListBox>,
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[glib::object_subclass]
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |     impl ObjectSubclass for EnsembleSelectorPopover {
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         const NAME: &'static str = "MusicusEnsembleSelectorPopover";
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |         type Type = super::EnsembleSelectorPopover;
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         type ParentType = gtk::Popover;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fn class_init(klass: &mut Self::Class) {
 | 
					
						
							|  |  |  |             klass.bind_template();
 | 
					
						
							|  |  |  |             klass.bind_template_instance_callbacks();
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fn instance_init(obj: &glib::subclass::InitializingObject<Self>) {
 | 
					
						
							|  |  |  |             obj.init_template();
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[glib::derived_properties]
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |     impl ObjectImpl for EnsembleSelectorPopover {
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         fn constructed(&self) {
 | 
					
						
							|  |  |  |             self.parent_constructed();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |             self.obj().connect_visible_notify(|obj| {
 | 
					
						
							|  |  |  |                 if obj.is_visible() {
 | 
					
						
							|  |  |  |                     obj.imp().search_entry.set_text("");
 | 
					
						
							|  |  |  |                     obj.imp().search_entry.grab_focus();
 | 
					
						
							|  |  |  |                     obj.imp().scrolled_window.vadjustment().set_value(0.0);
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             });
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             self.obj().search("");
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fn signals() -> &'static [Signal] {
 | 
					
						
							|  |  |  |             static SIGNALS: Lazy<Vec<Signal>> = Lazy::new(|| {
 | 
					
						
							|  |  |  |                 vec![
 | 
					
						
							|  |  |  |                     Signal::builder("ensemble-selected")
 | 
					
						
							|  |  |  |                         .param_types([Ensemble::static_type()])
 | 
					
						
							|  |  |  |                         .build(),
 | 
					
						
							|  |  |  |                     Signal::builder("create").build(),
 | 
					
						
							|  |  |  |                 ]
 | 
					
						
							|  |  |  |             });
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             SIGNALS.as_ref()
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |     impl WidgetImpl for EnsembleSelectorPopover {
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         // TODO: Fix focus.
 | 
					
						
							|  |  |  |         fn focus(&self, direction_type: gtk::DirectionType) -> bool {
 | 
					
						
							|  |  |  |             if direction_type == gtk::DirectionType::Down {
 | 
					
						
							|  |  |  |                 self.list_box.child_focus(direction_type)
 | 
					
						
							|  |  |  |             } else {
 | 
					
						
							|  |  |  |                 self.parent_focus(direction_type)
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |     impl PopoverImpl for EnsembleSelectorPopover {}
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | glib::wrapper! {
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |     pub struct EnsembleSelectorPopover(ObjectSubclass<imp::EnsembleSelectorPopover>)
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         @extends gtk::Widget, gtk::Popover;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #[gtk::template_callbacks]
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  | impl EnsembleSelectorPopover {
 | 
					
						
							|  |  |  |     pub fn new(library: &Library) -> Self {
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         glib::Object::builder().property("library", library).build()
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn connect_ensemble_selected<F: Fn(&Self, Ensemble) + 'static>(
 | 
					
						
							|  |  |  |         &self,
 | 
					
						
							|  |  |  |         f: F,
 | 
					
						
							|  |  |  |     ) -> glib::SignalHandlerId {
 | 
					
						
							|  |  |  |         self.connect_local("ensemble-selected", true, move |values| {
 | 
					
						
							|  |  |  |             let obj = values[0].get::<Self>().unwrap();
 | 
					
						
							|  |  |  |             let ensemble = values[1].get::<Ensemble>().unwrap();
 | 
					
						
							|  |  |  |             f(&obj, ensemble);
 | 
					
						
							|  |  |  |             None
 | 
					
						
							|  |  |  |         })
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pub fn connect_create<F: Fn(&Self) + 'static>(&self, f: F) -> glib::SignalHandlerId {
 | 
					
						
							|  |  |  |         self.connect_local("create", true, move |values| {
 | 
					
						
							|  |  |  |             let obj = values[0].get::<Self>().unwrap();
 | 
					
						
							|  |  |  |             f(&obj);
 | 
					
						
							|  |  |  |             None
 | 
					
						
							|  |  |  |         })
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[template_callback]
 | 
					
						
							|  |  |  |     fn search_changed(&self, entry: >k::SearchEntry) {
 | 
					
						
							|  |  |  |         self.search(&entry.text());
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[template_callback]
 | 
					
						
							|  |  |  |     fn activate(&self, _: >k::SearchEntry) {
 | 
					
						
							|  |  |  |         if let Some(ensemble) = self.imp().ensembles.borrow().first() {
 | 
					
						
							|  |  |  |             self.select(ensemble.clone());
 | 
					
						
							|  |  |  |         } else {
 | 
					
						
							|  |  |  |             self.create();
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[template_callback]
 | 
					
						
							|  |  |  |     fn stop_search(&self, _: >k::SearchEntry) {
 | 
					
						
							|  |  |  |         self.popdown();
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn search(&self, search: &str) {
 | 
					
						
							|  |  |  |         let imp = self.imp();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-09 08:14:46 +01:00
										 |  |  |         let ensembles = imp.library.get().unwrap().search_ensembles(search).unwrap();
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         imp.list_box.remove_all();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for ensemble in &ensembles {
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |             let row = ActivatableRow::new(
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |                 >k::Label::builder()
 | 
					
						
							|  |  |  |                     .label(ensemble.to_string())
 | 
					
						
							|  |  |  |                     .halign(gtk::Align::Start)
 | 
					
						
							|  |  |  |                     .build(),
 | 
					
						
							|  |  |  |             );
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-09 08:14:46 +01:00
										 |  |  |             row.set_tooltip_text(Some(&ensemble.to_string()));
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |             let ensemble = ensemble.clone();
 | 
					
						
							|  |  |  |             let obj = self.clone();
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |             row.connect_activated(move |_: &ActivatableRow| {
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |                 obj.select(ensemble.clone());
 | 
					
						
							|  |  |  |             });
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             imp.list_box.append(&row);
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let create_box = gtk::Box::builder().spacing(12).build();
 | 
					
						
							|  |  |  |         create_box.append(>k::Image::builder().icon_name("list-add-symbolic").build());
 | 
					
						
							|  |  |  |         create_box.append(
 | 
					
						
							|  |  |  |             >k::Label::builder()
 | 
					
						
							|  |  |  |                 .label(gettext("Create new ensemble"))
 | 
					
						
							|  |  |  |                 .halign(gtk::Align::Start)
 | 
					
						
							|  |  |  |                 .build(),
 | 
					
						
							|  |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |         let create_row = ActivatableRow::new(&create_box);
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |         let obj = self.clone();
 | 
					
						
							| 
									
										
										
										
											2025-03-01 09:57:01 +01:00
										 |  |  |         create_row.connect_activated(move |_: &ActivatableRow| {
 | 
					
						
							| 
									
										
										
										
											2025-01-15 11:23:04 +01:00
										 |  |  |             obj.create();
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         imp.list_box.append(&create_row);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         imp.ensembles.replace(ensembles);
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn select(&self, ensemble: Ensemble) {
 | 
					
						
							|  |  |  |         self.emit_by_name::<()>("ensemble-selected", &[&ensemble]);
 | 
					
						
							|  |  |  |         self.popdown();
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fn create(&self) {
 | 
					
						
							|  |  |  |         self.emit_by_name::<()>("create", &[]);
 | 
					
						
							|  |  |  |         self.popdown();
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 |