use super::base_keymap_setting::BaseKeymap; use fuzzy::{match_strings, StringMatch, StringMatchCandidate}; use gpui::{ actions, AppContext, DismissEvent, EventEmitter, FocusableView, Render, Task, View, ViewContext, VisualContext, WeakView, }; use picker::{Picker, PickerDelegate}; use project::Fs; use settings::{update_settings_file, Settings}; use std::sync::Arc; use ui::{prelude::*, ListItem, ListItemSpacing}; use util::ResultExt; use workspace::{ui::HighlightedLabel, ModalView, Workspace}; actions!(welcome, [ToggleBaseKeymapSelector]); pub fn init(cx: &mut AppContext) { cx.observe_new_views(|workspace: &mut Workspace, _cx| { workspace.register_action(toggle); }) .detach(); } pub fn toggle( workspace: &mut Workspace, _: &ToggleBaseKeymapSelector, cx: &mut ViewContext, ) { let fs = workspace.app_state().fs.clone(); workspace.toggle_modal(cx, |cx| { BaseKeymapSelector::new( BaseKeymapSelectorDelegate::new(cx.view().downgrade(), fs, cx), cx, ) }); } pub struct BaseKeymapSelector { picker: View>, } impl FocusableView for BaseKeymapSelector { fn focus_handle(&self, cx: &AppContext) -> gpui::FocusHandle { self.picker.focus_handle(cx) } } impl EventEmitter for BaseKeymapSelector {} impl ModalView for BaseKeymapSelector {} impl BaseKeymapSelector { pub fn new( delegate: BaseKeymapSelectorDelegate, cx: &mut ViewContext, ) -> Self { let picker = cx.new_view(|cx| Picker::uniform_list(delegate, cx)); Self { picker } } } impl Render for BaseKeymapSelector { fn render(&mut self, _cx: &mut ViewContext) -> impl IntoElement { v_flex().w(rems(34.)).child(self.picker.clone()) } } pub struct BaseKeymapSelectorDelegate { view: WeakView, matches: Vec, selected_index: usize, fs: Arc, } impl BaseKeymapSelectorDelegate { fn new( weak_view: WeakView, fs: Arc, cx: &mut ViewContext, ) -> Self { let base = BaseKeymap::get(None, cx); let selected_index = BaseKeymap::OPTIONS .iter() .position(|(_, value)| value == base) .unwrap_or(0); Self { view: weak_view, matches: Vec::new(), selected_index, fs, } } } impl PickerDelegate for BaseKeymapSelectorDelegate { type ListItem = ui::ListItem; fn placeholder_text(&self, _cx: &mut WindowContext) -> Arc { "Select a base keymap...".into() } fn match_count(&self) -> usize { self.matches.len() } fn selected_index(&self) -> usize { self.selected_index } fn set_selected_index( &mut self, ix: usize, _: &mut ViewContext>, ) { self.selected_index = ix; } fn update_matches( &mut self, query: String, cx: &mut ViewContext>, ) -> Task<()> { let background = cx.background_executor().clone(); let candidates = BaseKeymap::names() .enumerate() .map(|(id, name)| StringMatchCandidate::new(id, name)) .collect::>(); cx.spawn(|this, mut cx| async move { let matches = if query.is_empty() { candidates .into_iter() .enumerate() .map(|(index, candidate)| StringMatch { candidate_id: index, string: candidate.string, positions: Vec::new(), score: 0.0, }) .collect() } else { match_strings( &candidates, &query, false, 100, &Default::default(), background, ) .await }; this.update(&mut cx, |this, _| { this.delegate.matches = matches; this.delegate.selected_index = this .delegate .selected_index .min(this.delegate.matches.len().saturating_sub(1)); }) .log_err(); }) } fn confirm(&mut self, _: bool, cx: &mut ViewContext>) { if let Some(selection) = self.matches.get(self.selected_index) { let base_keymap = BaseKeymap::from_names(&selection.string); telemetry::event!( "Settings Changed", setting = "keymap", value = base_keymap.to_string() ); update_settings_file::(self.fs.clone(), cx, move |setting, _| { *setting = Some(base_keymap) }); } self.view .update(cx, |_, cx| { cx.emit(DismissEvent); }) .ok(); } fn dismissed(&mut self, cx: &mut ViewContext>) { self.view .update(cx, |_, cx| { cx.emit(DismissEvent); }) .log_err(); } fn render_match( &self, ix: usize, selected: bool, _cx: &mut ViewContext>, ) -> Option { let keymap_match = &self.matches[ix]; Some( ListItem::new(ix) .inset(true) .spacing(ListItemSpacing::Sparse) .toggle_state(selected) .child(HighlightedLabel::new( keymap_match.string.clone(), keymap_match.positions.clone(), )), ) } }