mirror of
				https://github.com/johrpan/musicus.git
				synced 2025-10-26 19:57:25 +01:00 
			
		
		
		
	
		
			
	
	
		
			479 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
		
		
			
		
	
	
			479 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
|  | use std::cell::{OnceCell, RefCell};
 | ||
|  | 
 | ||
|  | use adw::subclass::{navigation_page::NavigationPageImpl, prelude::*};
 | ||
|  | use formatx::formatx;
 | ||
|  | use gettextrs::gettext;
 | ||
|  | use gtk::{
 | ||
|  |     gio,
 | ||
|  |     glib::{self, Properties},
 | ||
|  |     prelude::*,
 | ||
|  | };
 | ||
|  | 
 | ||
|  | use crate::{
 | ||
|  |     album_page::AlbumPage,
 | ||
|  |     album_tile::AlbumTile,
 | ||
|  |     config,
 | ||
|  |     db::models::*,
 | ||
|  |     editor::{
 | ||
|  |         ensemble::EnsembleEditor, instrument::InstrumentEditor, person::PersonEditor,
 | ||
|  |         work::WorkEditor,
 | ||
|  |     },
 | ||
|  |     library::{Library, LibraryQuery},
 | ||
|  |     player::Player,
 | ||
|  |     program::Program,
 | ||
|  |     program_tile::ProgramTile,
 | ||
|  |     recording_tile::RecordingTile,
 | ||
|  |     search_tag::Tag,
 | ||
|  |     tag_tile::TagTile,
 | ||
|  | };
 | ||
|  | 
 | ||
|  | mod imp {
 | ||
|  |     use super::*;
 | ||
|  | 
 | ||
|  |     #[derive(Properties, Debug, Default, gtk::CompositeTemplate)]
 | ||
|  |     #[properties(wrapper_type = super::SearchPage)]
 | ||
|  |     #[template(file = "data/ui/search_page.blp")]
 | ||
|  |     pub struct SearchPage {
 | ||
|  |         #[property(get, construct_only)]
 | ||
|  |         pub navigation: OnceCell<adw::NavigationView>,
 | ||
|  | 
 | ||
|  |         #[property(get, construct_only)]
 | ||
|  |         pub library: OnceCell<Library>,
 | ||
|  | 
 | ||
|  |         #[property(get, construct_only)]
 | ||
|  |         pub player: OnceCell<Player>,
 | ||
|  | 
 | ||
|  |         pub query: OnceCell<LibraryQuery>,
 | ||
|  |         pub highlight: RefCell<Option<Tag>>,
 | ||
|  | 
 | ||
|  |         pub programs: RefCell<Vec<Program>>,
 | ||
|  |         pub composers: RefCell<Vec<Person>>,
 | ||
|  |         pub performers: RefCell<Vec<Person>>,
 | ||
|  |         pub ensembles: RefCell<Vec<Ensemble>>,
 | ||
|  |         pub instruments: RefCell<Vec<Instrument>>,
 | ||
|  |         pub works: RefCell<Vec<Work>>,
 | ||
|  |         pub recordings: RefCell<Vec<Recording>>,
 | ||
|  |         pub albums: RefCell<Vec<Album>>,
 | ||
|  | 
 | ||
|  |         #[template_child]
 | ||
|  |         pub scrolled_window: TemplateChild<gtk::ScrolledWindow>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub header_bar: TemplateChild<adw::HeaderBar>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub search_entry: TemplateChild<gtk::SearchEntry>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub stack: TemplateChild<gtk::Stack>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub header_box: TemplateChild<gtk::Box>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub title_label: TemplateChild<gtk::Label>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub subtitle_label: TemplateChild<gtk::Label>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub programs_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub composers_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub performers_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub ensembles_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub instruments_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub works_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub recordings_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |         #[template_child]
 | ||
|  |         pub albums_flow_box: TemplateChild<gtk::FlowBox>,
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[glib::object_subclass]
 | ||
|  |     impl ObjectSubclass for SearchPage {
 | ||
|  |         const NAME: &'static str = "MusicusSearchPage";
 | ||
|  |         type Type = super::SearchPage;
 | ||
|  |         type ParentType = adw::NavigationPage;
 | ||
|  | 
 | ||
|  |         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]
 | ||
|  |     impl ObjectImpl for SearchPage {
 | ||
|  |         fn constructed(&self) {
 | ||
|  |             self.parent_constructed();
 | ||
|  | 
 | ||
|  |             self.search_entry.set_key_capture_widget(Some(&*self.obj()));
 | ||
|  | 
 | ||
|  |             let obj = self.obj().to_owned();
 | ||
|  |             self.search_entry.connect_search_changed(move |entry| {
 | ||
|  |                 obj.imp().scrolled_window.vadjustment().set_value(0.0);
 | ||
|  |                 obj.search(&entry.text());
 | ||
|  |             });
 | ||
|  |         }
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     impl WidgetImpl for SearchPage {
 | ||
|  |         fn map(&self) {
 | ||
|  |             self.parent_map();
 | ||
|  |             self.search_entry.grab_focus();
 | ||
|  |         }
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     impl NavigationPageImpl for SearchPage {}
 | ||
|  | }
 | ||
|  | 
 | ||
|  | glib::wrapper! {
 | ||
|  |     pub struct SearchPage(ObjectSubclass<imp::SearchPage>)
 | ||
|  |         @extends gtk::Widget, adw::NavigationPage;
 | ||
|  | }
 | ||
|  | 
 | ||
|  | #[gtk::template_callbacks]
 | ||
|  | impl SearchPage {
 | ||
|  |     pub fn new(
 | ||
|  |         navigation: &adw::NavigationView,
 | ||
|  |         library: &Library,
 | ||
|  |         player: &Player,
 | ||
|  |         query: LibraryQuery,
 | ||
|  |     ) -> Self {
 | ||
|  |         let obj: Self = glib::Object::builder()
 | ||
|  |             .property("navigation", navigation)
 | ||
|  |             .property("library", library)
 | ||
|  |             .property("player", player)
 | ||
|  |             .build();
 | ||
|  | 
 | ||
|  |         if query.is_empty() {
 | ||
|  |             let settings = gio::Settings::new(&config::APP_ID);
 | ||
|  | 
 | ||
|  |             let programs = vec![
 | ||
|  |                 Program::deserialize(&settings.string("program1")).unwrap(),
 | ||
|  |                 Program::deserialize(&settings.string("program2")).unwrap(),
 | ||
|  |                 Program::deserialize(&settings.string("program3")).unwrap(),
 | ||
|  |             ];
 | ||
|  | 
 | ||
|  |             for program in &programs {
 | ||
|  |                 obj.imp()
 | ||
|  |                     .programs_flow_box
 | ||
|  |                     .append(&ProgramTile::new(program.to_owned()));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             obj.imp().programs.replace(programs);
 | ||
|  |         }
 | ||
|  | 
 | ||
|  |         obj.imp().query.set(query).unwrap();
 | ||
|  |         obj.search("");
 | ||
|  | 
 | ||
|  |         obj
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[template_callback]
 | ||
|  |     fn edit_button_clicked(&self) {
 | ||
|  |         if let Some(highlight) = &*self.imp().highlight.borrow() {
 | ||
|  |             match highlight {
 | ||
|  |                 Tag::Composer(person) | Tag::Performer(person) => {
 | ||
|  |                     self.navigation().push(&PersonEditor::new(
 | ||
|  |                         &self.navigation(),
 | ||
|  |                         &self.library(),
 | ||
|  |                         Some(person),
 | ||
|  |                     ));
 | ||
|  |                 }
 | ||
|  |                 Tag::Ensemble(ensemble) => {
 | ||
|  |                     self.navigation().push(&EnsembleEditor::new(
 | ||
|  |                         &self.navigation(),
 | ||
|  |                         &self.library(),
 | ||
|  |                         Some(ensemble),
 | ||
|  |                     ));
 | ||
|  |                 }
 | ||
|  |                 Tag::Instrument(instrument) => self.navigation().push(&InstrumentEditor::new(
 | ||
|  |                     &self.navigation(),
 | ||
|  |                     &self.library(),
 | ||
|  |                     Some(instrument),
 | ||
|  |                 )),
 | ||
|  |                 Tag::Work(work) => self.navigation().push(&WorkEditor::new(
 | ||
|  |                     &self.navigation(),
 | ||
|  |                     &self.library(),
 | ||
|  |                     Some(work),
 | ||
|  |                     false,
 | ||
|  |                 )),
 | ||
|  |             }
 | ||
|  |         }
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[template_callback]
 | ||
|  |     fn play_button_clicked(&self) {
 | ||
|  |         let program = Program::from_query(self.imp().query.get().unwrap().clone());
 | ||
|  |         self.player().set_program(program);
 | ||
|  |         self.player().play();
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[template_callback]
 | ||
|  |     fn select(&self) {
 | ||
|  |         let imp = self.imp();
 | ||
|  | 
 | ||
|  |         if imp.programs_flow_box.is_visible() {
 | ||
|  |             if let Some(program) = imp.programs.borrow().first().cloned() {
 | ||
|  |                 self.player().set_program(program);
 | ||
|  |             }
 | ||
|  |         } else {
 | ||
|  |             let mut new_query = self.imp().query.get().unwrap().clone();
 | ||
|  | 
 | ||
|  |             let query_changed = if let Some(person) = imp.composers.borrow().first().cloned() {
 | ||
|  |                 new_query.composer = Some(person);
 | ||
|  |                 true
 | ||
|  |             } else if let Some(person) = imp.performers.borrow().first().cloned() {
 | ||
|  |                 new_query.performer = Some(person);
 | ||
|  |                 true
 | ||
|  |             } else if let Some(ensemble) = imp.ensembles.borrow().first().cloned() {
 | ||
|  |                 new_query.ensemble = Some(ensemble);
 | ||
|  |                 true
 | ||
|  |             } else if let Some(instrument) = imp.instruments.borrow().first().cloned() {
 | ||
|  |                 new_query.instrument = Some(instrument);
 | ||
|  |                 true
 | ||
|  |             } else if let Some(work) = imp.works.borrow().first().cloned() {
 | ||
|  |                 new_query.work = Some(work);
 | ||
|  |                 true
 | ||
|  |             } else if let Some(recording) = imp.recordings.borrow().first().cloned() {
 | ||
|  |                 let playlist = self.player().recording_to_playlist(&recording);
 | ||
|  |                 self.player().append_and_play(playlist);
 | ||
|  |                 false
 | ||
|  |             } else if let Some(album) = imp.albums.borrow().first().cloned() {
 | ||
|  |                 self.show_album(&album);
 | ||
|  |                 false
 | ||
|  |             } else {
 | ||
|  |                 false
 | ||
|  |             };
 | ||
|  | 
 | ||
|  |             if query_changed {
 | ||
|  |                 self.navigation().push(&SearchPage::new(
 | ||
|  |                     &self.navigation(),
 | ||
|  |                     &self.library(),
 | ||
|  |                     &self.player(),
 | ||
|  |                     new_query,
 | ||
|  |                 ));
 | ||
|  |             }
 | ||
|  |         }
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[template_callback]
 | ||
|  |     fn program_selected(&self, tile: >k::FlowBoxChild) {
 | ||
|  |         self.player()
 | ||
|  |             .set_program(tile.downcast_ref::<ProgramTile>().unwrap().program());
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[template_callback]
 | ||
|  |     fn tile_selected(&self, tile: >k::FlowBoxChild) {
 | ||
|  |         let mut new_query = self.imp().query.get().unwrap().clone();
 | ||
|  |         match tile.downcast_ref::<TagTile>().unwrap().tag().clone() {
 | ||
|  |             Tag::Composer(person) => new_query.composer = Some(person),
 | ||
|  |             Tag::Performer(person) => new_query.performer = Some(person),
 | ||
|  |             Tag::Ensemble(ensemble) => new_query.ensemble = Some(ensemble),
 | ||
|  |             Tag::Instrument(instrument) => new_query.instrument = Some(instrument),
 | ||
|  |             Tag::Work(work) => new_query.work = Some(work),
 | ||
|  |         }
 | ||
|  | 
 | ||
|  |         self.navigation().push(&SearchPage::new(
 | ||
|  |             &self.navigation(),
 | ||
|  |             &self.library(),
 | ||
|  |             &self.player(),
 | ||
|  |             new_query,
 | ||
|  |         ));
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[template_callback]
 | ||
|  |     fn recording_selected(&self, tile: >k::FlowBoxChild) {
 | ||
|  |         let playlist = self
 | ||
|  |             .player()
 | ||
|  |             .recording_to_playlist(tile.downcast_ref::<RecordingTile>().unwrap().recording());
 | ||
|  |         self.player().append_and_play(playlist);
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     #[template_callback]
 | ||
|  |     fn album_selected(&self, tile: >k::FlowBoxChild) {
 | ||
|  |         self.show_album(tile.downcast_ref::<AlbumTile>().unwrap().album());
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     fn show_album(&self, album: &Album) {
 | ||
|  |         self.navigation().push(&AlbumPage::new(
 | ||
|  |             &self.navigation(),
 | ||
|  |             &self.library(),
 | ||
|  |             &self.player(),
 | ||
|  |             album.to_owned(),
 | ||
|  |         ));
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     fn search(&self, search: &str) {
 | ||
|  |         let query = self.imp().query.get().unwrap();
 | ||
|  | 
 | ||
|  |         let imp = self.imp();
 | ||
|  |         let results = self.library().search(query, search).unwrap();
 | ||
|  | 
 | ||
|  |         for flowbox in [
 | ||
|  |             &imp.composers_flow_box,
 | ||
|  |             &imp.performers_flow_box,
 | ||
|  |             &imp.ensembles_flow_box,
 | ||
|  |             &imp.instruments_flow_box,
 | ||
|  |             &imp.works_flow_box,
 | ||
|  |             &imp.recordings_flow_box,
 | ||
|  |             &imp.albums_flow_box,
 | ||
|  |         ] {
 | ||
|  |             while let Some(widget) = flowbox.first_child() {
 | ||
|  |                 flowbox.remove(&widget);
 | ||
|  |             }
 | ||
|  |         }
 | ||
|  | 
 | ||
|  |         // Only show programs initially.
 | ||
|  |         imp.programs_flow_box
 | ||
|  |             .set_visible(query.is_empty() && search.is_empty());
 | ||
|  | 
 | ||
|  |         imp.header_bar.set_show_title(query.is_empty());
 | ||
|  |         imp.header_box.set_visible(!query.is_empty());
 | ||
|  | 
 | ||
|  |         let highlight = if let Some(work) = &query.work {
 | ||
|  |             imp.title_label.set_text(&work.name.get());
 | ||
|  |             if let Some(composers) = work.composers_string() {
 | ||
|  |                 imp.subtitle_label.set_text(&composers);
 | ||
|  |                 imp.subtitle_label.set_visible(true);
 | ||
|  |             } else {
 | ||
|  |                 imp.subtitle_label.set_visible(false);
 | ||
|  |             }
 | ||
|  |             Some(Tag::Work(work.to_owned()))
 | ||
|  |         } else if let Some(person) = &query.composer {
 | ||
|  |             imp.title_label.set_text(&person.name.get());
 | ||
|  |             imp.subtitle_label.set_visible(false);
 | ||
|  |             Some(Tag::Composer(person.to_owned()))
 | ||
|  |         } else if let Some(person) = &query.performer {
 | ||
|  |             imp.title_label.set_text(&person.name.get());
 | ||
|  |             imp.subtitle_label.set_visible(false);
 | ||
|  |             Some(Tag::Performer(person.to_owned()))
 | ||
|  |         } else if let Some(ensemble) = &query.ensemble {
 | ||
|  |             imp.title_label.set_text(&ensemble.name.get());
 | ||
|  |             imp.subtitle_label.set_visible(false);
 | ||
|  |             Some(Tag::Ensemble(ensemble.to_owned()))
 | ||
|  |         } else if let Some(instrument) = &query.instrument {
 | ||
|  |             imp.title_label
 | ||
|  |                 .set_text(&formatx!(gettext("Music for {}"), &instrument.name.get()).unwrap());
 | ||
|  |             imp.subtitle_label.set_visible(false);
 | ||
|  |             Some(Tag::Instrument(instrument.to_owned()))
 | ||
|  |         } else {
 | ||
|  |             None
 | ||
|  |         };
 | ||
|  | 
 | ||
|  |         if let Some(highlight) = &highlight {
 | ||
|  |             if !matches!(highlight, Tag::Work(_)) {
 | ||
|  |                 let mut details = Vec::new();
 | ||
|  | 
 | ||
|  |                 match highlight {
 | ||
|  |                     Tag::Composer(_) => {
 | ||
|  |                         if let Some(instrument) = &query.instrument {
 | ||
|  |                             details.push(formatx!(gettext("Works with {}"), instrument).unwrap());
 | ||
|  |                         }
 | ||
|  | 
 | ||
|  |                         if let (Some(person), Some(ensemble)) = (&query.performer, &query.ensemble)
 | ||
|  |                         {
 | ||
|  |                             details.push(
 | ||
|  |                                 formatx!(gettext("Performed by {} and {}"), person, ensemble)
 | ||
|  |                                     .unwrap(),
 | ||
|  |                             );
 | ||
|  |                         } else if let Some(person) = &query.performer {
 | ||
|  |                             details.push(formatx!(gettext("Performed by {}"), person).unwrap());
 | ||
|  |                         } else if let Some(ensemble) = &query.ensemble {
 | ||
|  |                             details.push(formatx!(gettext("Performed by {}"), ensemble).unwrap());
 | ||
|  |                         }
 | ||
|  |                     }
 | ||
|  |                     Tag::Performer(_) => {
 | ||
|  |                         if let Some(instrument) = &query.instrument {
 | ||
|  |                             details.push(formatx!(gettext("Works with {}"), instrument).unwrap());
 | ||
|  |                         }
 | ||
|  | 
 | ||
|  |                         if let Some(ensemble) = &query.ensemble {
 | ||
|  |                             details.push(formatx!(gettext("Performed with {}"), ensemble).unwrap());
 | ||
|  |                         }
 | ||
|  |                     }
 | ||
|  |                     Tag::Ensemble(_) => {
 | ||
|  |                         if let Some(instrument) = &query.instrument {
 | ||
|  |                             details.push(formatx!(gettext("Works with {}"), instrument).unwrap());
 | ||
|  |                         }
 | ||
|  |                     }
 | ||
|  |                     Tag::Instrument(_) => (),
 | ||
|  |                     // Already covered.
 | ||
|  |                     Tag::Work(_) => unreachable!(),
 | ||
|  |                 }
 | ||
|  | 
 | ||
|  |                 imp.subtitle_label.set_visible(!details.is_empty());
 | ||
|  |                 imp.subtitle_label.set_text(&details.join(", "));
 | ||
|  |             }
 | ||
|  |         }
 | ||
|  | 
 | ||
|  |         imp.highlight.replace(highlight);
 | ||
|  | 
 | ||
|  |         if results.is_empty() {
 | ||
|  |             imp.stack.set_visible_child_name("empty");
 | ||
|  |         } else {
 | ||
|  |             imp.stack.set_visible_child_name("results");
 | ||
|  | 
 | ||
|  |             imp.composers_flow_box
 | ||
|  |                 .set_visible(!results.composers.is_empty());
 | ||
|  |             imp.performers_flow_box
 | ||
|  |                 .set_visible(!results.performers.is_empty());
 | ||
|  |             imp.ensembles_flow_box
 | ||
|  |                 .set_visible(!results.ensembles.is_empty());
 | ||
|  |             imp.instruments_flow_box
 | ||
|  |                 .set_visible(!results.instruments.is_empty());
 | ||
|  |             imp.works_flow_box.set_visible(!results.works.is_empty());
 | ||
|  |             imp.recordings_flow_box
 | ||
|  |                 .set_visible(!results.recordings.is_empty());
 | ||
|  |             imp.albums_flow_box.set_visible(!results.albums.is_empty());
 | ||
|  | 
 | ||
|  |             for composer in &results.composers {
 | ||
|  |                 imp.composers_flow_box
 | ||
|  |                     .append(&TagTile::new(Tag::Composer(composer.clone())));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             for performer in &results.performers {
 | ||
|  |                 imp.performers_flow_box
 | ||
|  |                     .append(&TagTile::new(Tag::Performer(performer.clone())));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             for ensemble in &results.ensembles {
 | ||
|  |                 imp.ensembles_flow_box
 | ||
|  |                     .append(&TagTile::new(Tag::Ensemble(ensemble.clone())));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             for instrument in &results.instruments {
 | ||
|  |                 imp.instruments_flow_box
 | ||
|  |                     .append(&TagTile::new(Tag::Instrument(instrument.clone())));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             for work in &results.works {
 | ||
|  |                 imp.works_flow_box
 | ||
|  |                     .append(&TagTile::new(Tag::Work(work.clone())));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             for recording in &results.recordings {
 | ||
|  |                 imp.recordings_flow_box.append(&RecordingTile::new(
 | ||
|  |                     &self.navigation(),
 | ||
|  |                     &self.library(),
 | ||
|  |                     recording,
 | ||
|  |                 ));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             for album in &results.albums {
 | ||
|  |                 imp.albums_flow_box.append(&AlbumTile::new(album));
 | ||
|  |             }
 | ||
|  | 
 | ||
|  |             imp.composers.replace(results.composers);
 | ||
|  |             imp.performers.replace(results.performers);
 | ||
|  |             imp.ensembles.replace(results.ensembles);
 | ||
|  |             imp.instruments.replace(results.instruments);
 | ||
|  |             imp.works.replace(results.works);
 | ||
|  |             imp.recordings.replace(results.recordings);
 | ||
|  |             imp.albums.replace(results.albums);
 | ||
|  |         }
 | ||
|  |     }
 | ||
|  | }
 |