use serde_json::Number; use crate::app::App; use crate::event::Key; use crate::handlers::table_handler::{TableHandlingConfig, handle_table}; use crate::handlers::{KeyEventHandler, handle_prompt_toggle}; use crate::matches_key; use crate::models::radarr_models::{RadarrRelease, RadarrReleaseDownloadBody}; use crate::models::servarr_data::radarr::radarr_data::{ ActiveRadarrBlock, EDIT_MOVIE_SELECTION_BLOCKS, MOVIE_DETAILS_BLOCKS, }; use crate::models::servarr_models::Language; use crate::models::stateful_table::SortOption; use crate::models::{BlockSelectionState, Scrollable}; use crate::network::radarr_network::RadarrEvent; #[cfg(test)] #[path = "movie_details_handler_tests.rs"] mod movie_details_handler_tests; pub(super) struct MovieDetailsHandler<'a, 'b> { key: Key, app: &'a mut App<'b>, active_radarr_block: ActiveRadarrBlock, _context: Option, } impl MovieDetailsHandler<'_, '_> { fn build_radarr_release_download_body(&self) -> RadarrReleaseDownloadBody { let movie_id = self.app.data.radarr_data.movies.current_selection().id; let (guid, indexer_id) = { let RadarrRelease { guid, indexer_id, .. } = self .app .data .radarr_data .movie_details_modal .as_ref() .unwrap() .movie_releases .current_selection(); (guid.clone(), *indexer_id) }; RadarrReleaseDownloadBody { guid, indexer_id, movie_id, } } fn extract_movie_id(&self) -> i64 { self.app.data.radarr_data.movies.current_selection().id } } impl<'a, 'b> KeyEventHandler<'a, 'b, ActiveRadarrBlock> for MovieDetailsHandler<'a, 'b> { fn handle(&mut self) { let movie_history_table_handling_config = TableHandlingConfig::new(ActiveRadarrBlock::MovieHistory.into()); let movie_releases_table_handling_config = TableHandlingConfig::new(ActiveRadarrBlock::ManualSearch.into()) .sorting_block(ActiveRadarrBlock::ManualSearchSortPrompt.into()) .sort_options(releases_sorting_options()); let movie_cast_table_handling_config = TableHandlingConfig::new(ActiveRadarrBlock::Cast.into()); let movie_crew_table_handling_config = TableHandlingConfig::new(ActiveRadarrBlock::Crew.into()); if !handle_table( self, |app| { &mut app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_history }, movie_history_table_handling_config, ) && !handle_table( self, |app| { &mut app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_releases }, movie_releases_table_handling_config, ) && !handle_table( self, |app| { &mut app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_cast }, movie_cast_table_handling_config, ) && !handle_table( self, |app| { &mut app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_crew }, movie_crew_table_handling_config, ) { self.handle_key_event(); } } fn accepts(active_block: ActiveRadarrBlock) -> bool { MOVIE_DETAILS_BLOCKS.contains(&active_block) } fn ignore_special_keys(&self) -> bool { self.app.ignore_special_keys_for_textbox_input } fn new( key: Key, app: &'a mut App<'b>, active_block: ActiveRadarrBlock, _context: Option, ) -> MovieDetailsHandler<'a, 'b> { MovieDetailsHandler { key, app, active_radarr_block: active_block, _context, } } fn get_key(&self) -> Key { self.key } fn is_ready(&self) -> bool { let Some(movie_details_modal) = &self.app.data.radarr_data.movie_details_modal else { return false; }; match self.active_radarr_block { ActiveRadarrBlock::MovieDetails => { !self.app.is_loading && !movie_details_modal.movie_details.is_empty() } ActiveRadarrBlock::MovieHistory => { !self.app.is_loading && !movie_details_modal.movie_history.is_empty() } ActiveRadarrBlock::Cast => !self.app.is_loading && !movie_details_modal.movie_cast.is_empty(), ActiveRadarrBlock::Crew => !self.app.is_loading && !movie_details_modal.movie_crew.is_empty(), ActiveRadarrBlock::ManualSearch => { !self.app.is_loading && !movie_details_modal.movie_releases.is_empty() } _ => !self.app.is_loading, } } fn handle_scroll_up(&mut self) { if self.active_radarr_block == ActiveRadarrBlock::MovieDetails { self .app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_details .scroll_up() } } fn handle_scroll_down(&mut self) { if self.active_radarr_block == ActiveRadarrBlock::MovieDetails { self .app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_details .scroll_down() } } fn handle_home(&mut self) { if self.active_radarr_block == ActiveRadarrBlock::MovieDetails { self .app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_details .scroll_to_top() } } fn handle_end(&mut self) { if let ActiveRadarrBlock::MovieDetails = self.active_radarr_block { self .app .data .radarr_data .movie_details_modal .as_mut() .unwrap() .movie_details .scroll_to_bottom() } } fn handle_delete(&mut self) {} fn handle_left_right_action(&mut self) { match self.active_radarr_block { ActiveRadarrBlock::MovieDetails | ActiveRadarrBlock::MovieHistory | ActiveRadarrBlock::FileInfo | ActiveRadarrBlock::Cast | ActiveRadarrBlock::Crew | ActiveRadarrBlock::ManualSearch => match self.key { _ if matches_key!(left, self.key) => { self.app.data.radarr_data.movie_info_tabs.previous(); self.app.pop_and_push_navigation_stack( self.app.data.radarr_data.movie_info_tabs.get_active_route(), ); } _ if matches_key!(right, self.key) => { self.app.data.radarr_data.movie_info_tabs.next(); self.app.pop_and_push_navigation_stack( self.app.data.radarr_data.movie_info_tabs.get_active_route(), ); } _ => (), }, ActiveRadarrBlock::AutomaticallySearchMoviePrompt | ActiveRadarrBlock::UpdateAndScanPrompt | ActiveRadarrBlock::ManualSearchConfirmPrompt => handle_prompt_toggle(self.app, self.key), _ => (), } } fn handle_submit(&mut self) { match self.active_radarr_block { ActiveRadarrBlock::AutomaticallySearchMoviePrompt => { if self.app.data.radarr_data.prompt_confirm { self.app.data.radarr_data.prompt_confirm_action = Some(RadarrEvent::TriggerAutomaticSearch(self.extract_movie_id())); } self.app.pop_navigation_stack(); } ActiveRadarrBlock::UpdateAndScanPrompt => { if self.app.data.radarr_data.prompt_confirm { self.app.data.radarr_data.prompt_confirm_action = Some(RadarrEvent::UpdateAndScan(self.extract_movie_id())); } self.app.pop_navigation_stack(); } ActiveRadarrBlock::ManualSearch => { self .app .push_navigation_stack(ActiveRadarrBlock::ManualSearchConfirmPrompt.into()); } ActiveRadarrBlock::ManualSearchConfirmPrompt => { if self.app.data.radarr_data.prompt_confirm { self.app.data.radarr_data.prompt_confirm_action = Some(RadarrEvent::DownloadRelease( self.build_radarr_release_download_body(), )); } self.app.pop_navigation_stack(); } _ => (), } } fn handle_esc(&mut self) { match self.active_radarr_block { ActiveRadarrBlock::MovieDetails | ActiveRadarrBlock::MovieHistory | ActiveRadarrBlock::FileInfo | ActiveRadarrBlock::Cast | ActiveRadarrBlock::Crew | ActiveRadarrBlock::ManualSearch => { self.app.pop_navigation_stack(); self.app.data.radarr_data.reset_movie_info_tabs(); } ActiveRadarrBlock::AutomaticallySearchMoviePrompt | ActiveRadarrBlock::UpdateAndScanPrompt | ActiveRadarrBlock::ManualSearchConfirmPrompt => { self.app.pop_navigation_stack(); self.app.data.radarr_data.prompt_confirm = false; } _ => (), } } fn handle_char_key_event(&mut self) { let key = self.key; match self.active_radarr_block { ActiveRadarrBlock::MovieDetails | ActiveRadarrBlock::MovieHistory | ActiveRadarrBlock::FileInfo | ActiveRadarrBlock::Cast | ActiveRadarrBlock::Crew | ActiveRadarrBlock::ManualSearch => match self.key { _ if matches_key!(auto_search, key) => { self .app .push_navigation_stack(ActiveRadarrBlock::AutomaticallySearchMoviePrompt.into()); } _ if matches_key!(edit, key) => { self.app.push_navigation_stack( ( ActiveRadarrBlock::EditMoviePrompt, Some(self.active_radarr_block), ) .into(), ); self.app.data.radarr_data.edit_movie_modal = Some((&self.app.data.radarr_data).into()); self.app.data.radarr_data.selected_block = BlockSelectionState::new(EDIT_MOVIE_SELECTION_BLOCKS); } _ if matches_key!(update, key) => { self .app .push_navigation_stack(ActiveRadarrBlock::UpdateAndScanPrompt.into()); } _ if matches_key!(refresh, key) => { self .app .pop_and_push_navigation_stack(self.active_radarr_block.into()); } _ => (), }, ActiveRadarrBlock::AutomaticallySearchMoviePrompt if matches_key!(confirm, key) => { self.app.data.radarr_data.prompt_confirm = true; self.app.data.radarr_data.prompt_confirm_action = Some(RadarrEvent::TriggerAutomaticSearch(self.extract_movie_id())); self.app.pop_navigation_stack(); } ActiveRadarrBlock::UpdateAndScanPrompt if matches_key!(confirm, key) => { self.app.data.radarr_data.prompt_confirm = true; self.app.data.radarr_data.prompt_confirm_action = Some(RadarrEvent::UpdateAndScan(self.extract_movie_id())); self.app.pop_navigation_stack(); } ActiveRadarrBlock::ManualSearchConfirmPrompt if matches_key!(confirm, key) => { self.app.data.radarr_data.prompt_confirm = true; self.app.data.radarr_data.prompt_confirm_action = Some(RadarrEvent::DownloadRelease( self.build_radarr_release_download_body(), )); self.app.pop_navigation_stack(); } _ => (), } } fn app_mut(&mut self) -> &mut App<'b> { self.app } fn current_route(&self) -> crate::models::Route { self.app.get_current_route() } } fn releases_sorting_options() -> Vec> { vec![ SortOption { name: "Source", cmp_fn: Some(|a, b| a.protocol.cmp(&b.protocol)), }, SortOption { name: "Age", cmp_fn: Some(|a, b| a.age.cmp(&b.age)), }, SortOption { name: "Rejected", cmp_fn: Some(|a, b| a.rejected.cmp(&b.rejected)), }, SortOption { name: "Title", cmp_fn: Some(|a, b| { a.title .text .to_lowercase() .cmp(&b.title.text.to_lowercase()) }), }, SortOption { name: "Indexer", cmp_fn: Some(|a, b| a.indexer.to_lowercase().cmp(&b.indexer.to_lowercase())), }, SortOption { name: "Size", cmp_fn: Some(|a, b| a.size.cmp(&b.size)), }, SortOption { name: "Peers", cmp_fn: Some(|a, b| { let default_number = Number::from(i64::MAX); let seeder_a = a .seeders .as_ref() .unwrap_or(&default_number) .as_u64() .unwrap(); let seeder_b = b .seeders .as_ref() .unwrap_or(&default_number) .as_u64() .unwrap(); seeder_a.cmp(&seeder_b) }), }, SortOption { name: "Language", cmp_fn: Some(|a, b| { let default_language_vec = vec![Language { id: 1, name: "_".to_owned(), }]; let language_a = &a.languages.as_ref().unwrap_or(&default_language_vec)[0]; let language_b = &b.languages.as_ref().unwrap_or(&default_language_vec)[0]; language_a.cmp(language_b) }), }, SortOption { name: "Quality", cmp_fn: Some(|a, b| a.quality.cmp(&b.quality)), }, ] }