mirror of
				https://github.com/johrpan/musicus.git
				synced 2025-10-26 19:57:25 +01:00 
			
		
		
		
	Use new navigator for all screens
This commit is contained in:
		
							parent
							
								
									c72bc71432
								
							
						
					
					
						commit
						ee71a905e1
					
				
					 6 changed files with 156 additions and 226 deletions
				
			
		|  | @ -1,11 +1,10 @@ | |||
| use super::{WorkScreen, RecordingScreen}; | ||||
| 
 | ||||
| use crate::backend::Backend; | ||||
| use crate::database::{Person, Recording, Work}; | ||||
| use crate::editors::PersonEditor; | ||||
| use crate::navigator::NavigatorWindow; | ||||
| use crate::widgets::{List, Navigator, NavigatorScreen, Screen, Section}; | ||||
| 
 | ||||
| use crate::navigator::{NavigatorWindow, NavigationHandle, Screen}; | ||||
| use crate::widgets; | ||||
| use crate::widgets::{List, Section, Widget}; | ||||
| use gettextrs::gettext; | ||||
| use glib::clone; | ||||
| use gtk::prelude::*; | ||||
|  | @ -15,67 +14,60 @@ use std::rc::Rc; | |||
| 
 | ||||
| /// A screen for showing works by and recordings with a person.
 | ||||
| pub struct PersonScreen { | ||||
|     backend: Rc<Backend>, | ||||
|     handle: NavigationHandle<()>, | ||||
|     person: Person, | ||||
|     widget: Screen, | ||||
|     widget: widgets::Screen, | ||||
|     work_list: Rc<List>, | ||||
|     recording_list: Rc<List>, | ||||
|     works: RefCell<Vec<Work>>, | ||||
|     recordings: RefCell<Vec<Recording>>, | ||||
|     navigator: RefCell<Option<Rc<Navigator>>>, | ||||
| } | ||||
| 
 | ||||
| impl PersonScreen { | ||||
| impl Screen<Person, ()> for PersonScreen { | ||||
|     /// Create a new person screen for the specified person and load the
 | ||||
|     /// contents asynchronously.
 | ||||
|     pub fn new(backend: Rc<Backend>, person: Person) -> Rc<Self> { | ||||
|         let widget = Screen::new(); | ||||
|     fn new(person: Person, handle: NavigationHandle<()>) -> Rc<Self> { | ||||
|         let widget = widgets::Screen::new(); | ||||
|         widget.set_title(&person.name_fl()); | ||||
| 
 | ||||
|         let work_list = List::new(); | ||||
|         let recording_list = List::new(); | ||||
| 
 | ||||
|         let this = Rc::new(Self { | ||||
|             backend, | ||||
|             handle, | ||||
|             person, | ||||
|             widget, | ||||
|             work_list, | ||||
|             recording_list, | ||||
|             works: RefCell::new(Vec::new()), | ||||
|             recordings: RefCell::new(Vec::new()), | ||||
|             navigator: RefCell::new(None), | ||||
|         }); | ||||
| 
 | ||||
|         this.widget.set_back_cb(clone!(@strong this => move || { | ||||
|             let navigator = this.navigator.borrow().clone(); | ||||
|             if let Some(navigator) = navigator { | ||||
|                 navigator.pop(); | ||||
|             } | ||||
|         this.widget.set_back_cb(clone!(@weak this => move || { | ||||
|             this.handle.pop(None); | ||||
|         })); | ||||
| 
 | ||||
| 
 | ||||
|         this.widget.add_action(&gettext("Edit person"), clone!(@strong this => move || { | ||||
|         this.widget.add_action(&gettext("Edit person"), clone!(@weak this => move || { | ||||
|             spawn!(@clone this, async move { | ||||
|                 let window = NavigatorWindow::new(this.backend.clone()); | ||||
|                 replace!(window.navigator, PersonEditor, None).await; | ||||
|                 let window = NavigatorWindow::new(this.handle.backend.clone()); | ||||
|                 replace!(window.navigator, PersonEditor, Some(this.person.clone())).await; | ||||
|             }); | ||||
|         })); | ||||
| 
 | ||||
|         this.widget.add_action(&gettext("Delete person"), clone!(@strong this => move || { | ||||
|             let context = glib::MainContext::default(); | ||||
|             let clone = this.clone(); | ||||
|             context.spawn_local(async move { | ||||
|                 clone.backend.db().delete_person(&clone.person.id).await.unwrap(); | ||||
|                 clone.backend.library_changed(); | ||||
|         this.widget.add_action(&gettext("Delete person"), clone!(@weak this => move || { | ||||
|             spawn!(@clone this, async move { | ||||
|                 this.handle.backend.db().delete_person(&this.person.id).await.unwrap(); | ||||
|                 this.handle.backend.library_changed(); | ||||
|             }); | ||||
|         })); | ||||
| 
 | ||||
|         this.widget.set_search_cb(clone!(@strong this => move || { | ||||
|         this.widget.set_search_cb(clone!(@weak this => move || { | ||||
|             this.work_list.invalidate_filter(); | ||||
|             this.recording_list.invalidate_filter(); | ||||
|         })); | ||||
| 
 | ||||
|         this.work_list.set_make_widget_cb(clone!(@strong this => move |index| { | ||||
|         this.work_list.set_make_widget_cb(clone!(@weak this => move |index| { | ||||
|             let work = &this.works.borrow()[index]; | ||||
| 
 | ||||
|             let row = libadwaita::ActionRow::new(); | ||||
|  | @ -83,24 +75,24 @@ impl PersonScreen { | |||
|             row.set_title(Some(&work.title)); | ||||
| 
 | ||||
|             let work = work.to_owned(); | ||||
|             row.connect_activated(clone!(@strong this => move |_| { | ||||
|                 let navigator = this.navigator.borrow().clone(); | ||||
|                 if let Some(navigator) = navigator { | ||||
|                     navigator.push(WorkScreen::new(this.backend.clone(), work.clone())); | ||||
|                 } | ||||
|             row.connect_activated(clone!(@weak this => move |_| { | ||||
|                 let work = work.clone(); | ||||
|                 spawn!(@clone this, async move { | ||||
|                     push!(this.handle, WorkScreen, work.clone()).await; | ||||
|                 }); | ||||
|             })); | ||||
| 
 | ||||
|             row.upcast() | ||||
|         })); | ||||
| 
 | ||||
|         this.work_list.set_filter_cb(clone!(@strong this => move |index| { | ||||
|         this.work_list.set_filter_cb(clone!(@weak this => move |index| { | ||||
|             let work = &this.works.borrow()[index]; | ||||
|             let search = this.widget.get_search(); | ||||
|             let title = work.title.to_lowercase(); | ||||
|             search.is_empty() || title.contains(&search) | ||||
|         })); | ||||
| 
 | ||||
|         this.recording_list.set_make_widget_cb(clone!(@strong this => move |index| { | ||||
|         this.recording_list.set_make_widget_cb(clone!(@weak this => move |index| { | ||||
|             let recording = &this.recordings.borrow()[index]; | ||||
| 
 | ||||
|             let row = libadwaita::ActionRow::new(); | ||||
|  | @ -109,17 +101,17 @@ impl PersonScreen { | |||
|             row.set_subtitle(Some(&recording.get_performers())); | ||||
| 
 | ||||
|             let recording = recording.to_owned(); | ||||
|             row.connect_activated(clone!(@strong this => move |_| { | ||||
|                 let navigator = this.navigator.borrow().clone(); | ||||
|                 if let Some(navigator) = navigator { | ||||
|                     navigator.push(RecordingScreen::new(this.backend.clone(), recording.clone())); | ||||
|                 } | ||||
|             row.connect_activated(clone!(@weak this => move |_| { | ||||
|                 let recording = recording.clone(); | ||||
|                 spawn!(@clone this, async move { | ||||
|                     push!(this.handle, RecordingScreen, recording.clone()).await; | ||||
|                 }); | ||||
|             })); | ||||
| 
 | ||||
|             row.upcast() | ||||
|         })); | ||||
| 
 | ||||
|         this.recording_list.set_filter_cb(clone!(@strong this => move |index| { | ||||
|         this.recording_list.set_filter_cb(clone!(@weak this => move |index| { | ||||
|             let recording = &this.recordings.borrow()[index]; | ||||
|             let search = this.widget.get_search(); | ||||
|             let text = recording.work.get_title() + &recording.get_performers(); | ||||
|  | @ -128,59 +120,48 @@ impl PersonScreen { | |||
| 
 | ||||
|         // Load the content asynchronously.
 | ||||
| 
 | ||||
|         let context = glib::MainContext::default(); | ||||
|         let clone = Rc::clone(&this); | ||||
| 
 | ||||
|         context.spawn_local(async move { | ||||
|             let works = clone | ||||
|         spawn!(@clone this, async move { | ||||
|             let works = this.handle | ||||
|                 .backend | ||||
|                 .db() | ||||
|                 .get_works(&clone.person.id) | ||||
|                 .get_works(&this.person.id) | ||||
|                 .await | ||||
|                 .unwrap(); | ||||
| 
 | ||||
|             let recordings = clone | ||||
|             let recordings = this.handle | ||||
|                 .backend | ||||
|                 .db() | ||||
|                 .get_recordings_for_person(&clone.person.id) | ||||
|                 .get_recordings_for_person(&this.person.id) | ||||
|                 .await | ||||
|                 .unwrap(); | ||||
| 
 | ||||
|             if !works.is_empty() { | ||||
|                 let length = works.len(); | ||||
|                 clone.works.replace(works); | ||||
|                 clone.work_list.update(length); | ||||
|                 this.works.replace(works); | ||||
|                 this.work_list.update(length); | ||||
| 
 | ||||
|                 let section = Section::new("Works", &clone.work_list.widget); | ||||
|                 clone.widget.add_content(§ion.widget); | ||||
|                 let section = Section::new("Works", &this.work_list.widget); | ||||
|                 this.widget.add_content(§ion.widget); | ||||
|             } | ||||
| 
 | ||||
|             if !recordings.is_empty() { | ||||
|                 let length = recordings.len(); | ||||
|                 clone.recordings.replace(recordings); | ||||
|                 clone.recording_list.update(length); | ||||
|                 this.recordings.replace(recordings); | ||||
|                 this.recording_list.update(length); | ||||
| 
 | ||||
|                 let section = Section::new("Recordings", &clone.recording_list.widget); | ||||
|                 clone.widget.add_content(§ion.widget); | ||||
|                 let section = Section::new("Recordings", &this.recording_list.widget); | ||||
|                 this.widget.add_content(§ion.widget); | ||||
|             } | ||||
| 
 | ||||
|             clone.widget.ready(); | ||||
|             this.widget.ready(); | ||||
|         }); | ||||
| 
 | ||||
|         this | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl NavigatorScreen for PersonScreen { | ||||
|     fn attach_navigator(&self, navigator: Rc<Navigator>) { | ||||
|         self.navigator.replace(Some(navigator)); | ||||
|     } | ||||
| 
 | ||||
| impl Widget for PersonScreen { | ||||
|     fn get_widget(&self) -> gtk::Widget { | ||||
|         self.widget.widget.clone().upcast() | ||||
|     } | ||||
| 
 | ||||
|     fn detach_navigator(&self) { | ||||
|         self.navigator.replace(None); | ||||
|     } | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Elias Projahn
						Elias Projahn