mod active_buffer_language; pub use active_buffer_language::ActiveBufferLanguage; use anyhow::anyhow; use editor::Editor; use fuzzy::{match_strings, StringMatch, StringMatchCandidate}; use gpui::{ actions, AppContext, DismissEvent, Div, EventEmitter, FocusHandle, FocusableView, Model, ParentElement, Render, Styled, View, ViewContext, VisualContext, WeakView, }; use language::{Buffer, LanguageRegistry}; use picker::{Picker, PickerDelegate}; use project::Project; use std::sync::Arc; use ui::{v_stack, HighlightedLabel, ListItem, Selectable}; use util::ResultExt; use workspace::Workspace; actions!(Toggle); pub fn init(cx: &mut AppContext) { cx.observe_new_views(LanguageSelector::register).detach(); } pub struct LanguageSelector { picker: View>, } impl LanguageSelector { fn register(workspace: &mut Workspace, _: &mut ViewContext) { workspace.register_action(move |workspace, _: &Toggle, cx| { Self::toggle(workspace, cx); }); } fn toggle(workspace: &mut Workspace, cx: &mut ViewContext) -> Option<()> { let registry = workspace.app_state().languages.clone(); let (_, buffer, _) = workspace .active_item(cx)? .act_as::(cx)? .read(cx) .active_excerpt(cx)?; let project = workspace.project().clone(); workspace.toggle_modal(cx, move |cx| { LanguageSelector::new(buffer, project, registry, cx) }); Some(()) } fn new( buffer: Model, project: Model, language_registry: Arc, cx: &mut ViewContext, ) -> Self { let delegate = LanguageSelectorDelegate::new( cx.view().downgrade(), buffer, project, language_registry, ); let picker = cx.build_view(|cx| Picker::new(delegate, cx)); Self { picker } } } impl Render for LanguageSelector { type Element = Div; fn render(&mut self, _cx: &mut ViewContext) -> Self::Element { v_stack().min_w_96().child(self.picker.clone()) } } impl FocusableView for LanguageSelector { fn focus_handle(&self, cx: &AppContext) -> FocusHandle { self.picker.focus_handle(cx) } } impl EventEmitter for LanguageSelector {} pub struct LanguageSelectorDelegate { language_selector: WeakView, buffer: Model, project: Model, language_registry: Arc, candidates: Vec, matches: Vec, selected_index: usize, } impl LanguageSelectorDelegate { fn new( language_selector: WeakView, buffer: Model, project: Model, language_registry: Arc, ) -> Self { let candidates = language_registry .language_names() .into_iter() .enumerate() .map(|(candidate_id, name)| StringMatchCandidate::new(candidate_id, name)) .collect::>(); Self { language_selector, buffer, project, language_registry, candidates, matches: vec![], selected_index: 0, } } } impl PickerDelegate for LanguageSelectorDelegate { type ListItem = ListItem; fn placeholder_text(&self) -> Arc { "Select a language...".into() } fn match_count(&self) -> usize { self.matches.len() } fn confirm(&mut self, _: bool, cx: &mut ViewContext>) { if let Some(mat) = self.matches.get(self.selected_index) { let language_name = &self.candidates[mat.candidate_id].string; let language = self.language_registry.language_for_name(language_name); let project = self.project.downgrade(); let buffer = self.buffer.downgrade(); cx.spawn(|_, mut cx| async move { let language = language.await?; let project = project .upgrade() .ok_or_else(|| anyhow!("project was dropped"))?; let buffer = buffer .upgrade() .ok_or_else(|| anyhow!("buffer was dropped"))?; project.update(&mut cx, |project, cx| { project.set_language_for_buffer(&buffer, language, cx); }) }) .detach_and_log_err(cx); } self.dismissed(cx); } fn dismissed(&mut self, cx: &mut ViewContext>) { self.language_selector .update(cx, |_, cx| cx.emit(DismissEvent)) .log_err(); } 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>, ) -> gpui::Task<()> { let background = cx.background_executor().clone(); let candidates = self.candidates.clone(); 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, cx| { let delegate = &mut this.delegate; delegate.matches = matches; delegate.selected_index = delegate .selected_index .min(delegate.matches.len().saturating_sub(1)); cx.notify(); }) .log_err(); }) } fn render_match( &self, ix: usize, selected: bool, cx: &mut ViewContext>, ) -> Option { let mat = &self.matches[ix]; let buffer_language_name = self.buffer.read(cx).language().map(|l| l.name()); let mut label = mat.string.clone(); if buffer_language_name.as_deref() == Some(mat.string.as_str()) { label.push_str(" (current)"); } Some( ListItem::new(ix) .inset(true) .selected(selected) .child(HighlightedLabel::new(label, mat.positions.clone())), ) } }