use fuzzy_matcher::skim::SkimMatcherV2; use fuzzy_matcher::FuzzyMatcher; use std::path::PathBuf; use zellij_tile::prelude::*; #[derive(Default)] pub struct NewSessionInfo { name: String, layout_list: LayoutList, entering_new_session_info: EnteringState, pub new_session_folder: Option, } #[derive(Eq, PartialEq)] enum EnteringState { EnteringName, EnteringLayoutSearch, } impl Default for EnteringState { fn default() -> Self { EnteringState::EnteringName } } impl NewSessionInfo { pub fn name(&self) -> &str { &self.name } pub fn layout_search_term(&self) -> &str { &self.layout_list.layout_search_term } pub fn entering_new_session_name(&self) -> bool { self.entering_new_session_info == EnteringState::EnteringName } pub fn entering_layout_search_term(&self) -> bool { self.entering_new_session_info == EnteringState::EnteringLayoutSearch } pub fn add_char(&mut self, character: char) { match self.entering_new_session_info { EnteringState::EnteringName => { self.name.push(character); }, EnteringState::EnteringLayoutSearch => { self.layout_list.layout_search_term.push(character); self.update_layout_search_term(); }, } } pub fn handle_backspace(&mut self) { match self.entering_new_session_info { EnteringState::EnteringName => { self.name.pop(); }, EnteringState::EnteringLayoutSearch => { self.layout_list.layout_search_term.pop(); self.update_layout_search_term(); }, } } pub fn handle_break(&mut self) { match self.entering_new_session_info { EnteringState::EnteringName => { self.name.clear(); }, EnteringState::EnteringLayoutSearch => { self.layout_list.layout_search_term.clear(); self.entering_new_session_info = EnteringState::EnteringName; self.update_layout_search_term(); }, } } pub fn handle_key(&mut self, key: KeyWithModifier) { match key.bare_key { BareKey::Backspace if key.has_no_modifiers() => { self.handle_backspace(); }, BareKey::Char('c') if key.has_modifiers(&[KeyModifier::Ctrl]) => { self.handle_break(); }, BareKey::Esc if key.has_no_modifiers() => { self.handle_break(); }, BareKey::Char(character) if key.has_no_modifiers() => { self.add_char(character); }, BareKey::Up if key.has_no_modifiers() => { self.move_selection_up(); }, BareKey::Down if key.has_no_modifiers() => { self.move_selection_down(); }, _ => {}, } } pub fn handle_selection(&mut self, current_session_name: &Option) { match self.entering_new_session_info { EnteringState::EnteringLayoutSearch => { let new_session_layout: Option = self.selected_layout_info(); let new_session_name = if self.name.is_empty() { None } else { Some(self.name.as_str()) }; if new_session_name != current_session_name.as_ref().map(|s| s.as_str()) { match new_session_layout { Some(new_session_layout) => { let cwd = self.new_session_folder.as_ref().map(|c| PathBuf::from(c)); switch_session_with_layout(new_session_name, new_session_layout, cwd) }, None => { switch_session(new_session_name); }, } } self.name.clear(); self.layout_list.clear_selection(); hide_self(); }, EnteringState::EnteringName => { self.entering_new_session_info = EnteringState::EnteringLayoutSearch; }, } } pub fn update_layout_list(&mut self, layout_info: Vec) { self.layout_list.update_layout_list(layout_info); } pub fn layout_list(&self, max_rows: usize) -> Vec<(LayoutInfo, bool)> { // bool - is_selected let range_to_render = self.range_to_render( max_rows, self.layout_count(), Some(self.layout_list.selected_layout_index), ); self.layout_list .layout_list .iter() .enumerate() .map(|(i, l)| (l.clone(), i == self.layout_list.selected_layout_index)) .take(range_to_render.1) .skip(range_to_render.0) .collect() } pub fn layouts_to_render(&self, max_rows: usize) -> Vec<(LayoutInfo, Vec, bool)> { // (layout_info, // search_indices, // is_selected) if self.is_searching() { self.layout_search_results(max_rows) .into_iter() .map(|(layout_search_result, is_selected)| { ( layout_search_result.layout_info, layout_search_result.indices, is_selected, ) }) .collect() } else { self.layout_list(max_rows) .into_iter() .map(|(layout_info, is_selected)| (layout_info, vec![], is_selected)) .collect() } } pub fn layout_search_results(&self, max_rows: usize) -> Vec<(LayoutSearchResult, bool)> { // bool - is_selected let range_to_render = self.range_to_render( max_rows, self.layout_list.layout_search_results.len(), Some(self.layout_list.selected_layout_index), ); self.layout_list .layout_search_results .iter() .enumerate() .map(|(i, l)| (l.clone(), i == self.layout_list.selected_layout_index)) .take(range_to_render.1) .skip(range_to_render.0) .collect() } // TODO: merge with similar function in resurrectable_sessions fn range_to_render( &self, table_rows: usize, results_len: usize, selected_index: Option, ) -> (usize, usize) { if table_rows <= results_len { let row_count_to_render = table_rows.saturating_sub(1); // 1 for the title let first_row_index_to_render = selected_index .unwrap_or(0) .saturating_sub(row_count_to_render / 2); let last_row_index_to_render = first_row_index_to_render + row_count_to_render; (first_row_index_to_render, last_row_index_to_render) } else { let first_row_index_to_render = 0; let last_row_index_to_render = results_len; (first_row_index_to_render, last_row_index_to_render) } } pub fn is_searching(&self) -> bool { !self.layout_list.layout_search_term.is_empty() } pub fn layout_count(&self) -> usize { self.layout_list.layout_list.len() } pub fn selected_layout_info(&self) -> Option { self.layout_list.selected_layout_info() } fn update_layout_search_term(&mut self) { if self.layout_list.layout_search_term.is_empty() { self.layout_list.clear_selection(); self.layout_list.layout_search_results = vec![]; } else { let mut matches = vec![]; let matcher = SkimMatcherV2::default().use_cache(true); for layout_info in &self.layout_list.layout_list { if let Some((score, indices)) = matcher.fuzzy_indices(&layout_info.name(), &self.layout_list.layout_search_term) { matches.push(LayoutSearchResult { layout_info: layout_info.clone(), score, indices, }); } } matches.sort_by(|a, b| b.score.cmp(&a.score)); self.layout_list.layout_search_results = matches; self.layout_list.clear_selection(); } } fn move_selection_up(&mut self) { self.layout_list.move_selection_up(); } fn move_selection_down(&mut self) { self.layout_list.move_selection_down(); } } #[derive(Default)] pub struct LayoutList { layout_list: Vec, layout_search_results: Vec, selected_layout_index: usize, layout_search_term: String, } impl LayoutList { pub fn update_layout_list(&mut self, layout_list: Vec) { let old_layout_length = self.layout_list.len(); self.layout_list = layout_list; if old_layout_length != self.layout_list.len() { // honestly, this is just the UX choice that sucks the least... self.clear_selection(); } } pub fn selected_layout_info(&self) -> Option { if !self.layout_search_term.is_empty() { self.layout_search_results .get(self.selected_layout_index) .map(|l| l.layout_info.clone()) } else { self.layout_list.get(self.selected_layout_index).cloned() } } pub fn clear_selection(&mut self) { self.selected_layout_index = 0; } fn max_index(&self) -> usize { if self.layout_search_term.is_empty() { self.layout_list.len().saturating_sub(1) } else { self.layout_search_results.len().saturating_sub(1) } } fn move_selection_up(&mut self) { let max_index = self.max_index(); if self.selected_layout_index > 0 { self.selected_layout_index -= 1; } else { self.selected_layout_index = max_index; } } fn move_selection_down(&mut self) { let max_index = self.max_index(); if self.selected_layout_index < max_index { self.selected_layout_index += 1; } else { self.selected_layout_index = 0; } } } #[derive(Clone)] pub struct LayoutSearchResult { pub layout_info: LayoutInfo, pub score: i64, pub indices: Vec, }