mod active_toolchain; pub use active_toolchain::ActiveToolchain; use editor::Editor; use fuzzy::{StringMatch, StringMatchCandidate, match_strings}; use gpui::{ App, Context, DismissEvent, Entity, EventEmitter, FocusHandle, Focusable, ParentElement, Render, Styled, Task, WeakEntity, Window, actions, }; use language::{LanguageName, Toolchain, ToolchainList}; use picker::{Picker, PickerDelegate}; use project::{Project, ProjectPath, WorktreeId}; use std::{borrow::Cow, path::Path, sync::Arc}; use ui::{HighlightedLabel, ListItem, ListItemSpacing, prelude::*}; use util::ResultExt; use workspace::{ModalView, Workspace}; actions!(toolchain, [Select]); pub fn init(cx: &mut App) { cx.observe_new(ToolchainSelector::register).detach(); } pub struct ToolchainSelector { picker: Entity>, } impl ToolchainSelector { fn register( workspace: &mut Workspace, _window: Option<&mut Window>, _: &mut Context, ) { workspace.register_action(move |workspace, _: &Select, window, cx| { Self::toggle(workspace, window, cx); }); } fn toggle( workspace: &mut Workspace, window: &mut Window, cx: &mut Context, ) -> Option<()> { let (_, buffer, _) = workspace .active_item(cx)? .act_as::(cx)? .read(cx) .active_excerpt(cx)?; let project = workspace.project().clone(); let language_name = buffer.read(cx).language()?.name(); let worktree_id = buffer.read(cx).file()?.worktree_id(cx); let relative_path: Arc = Arc::from(buffer.read(cx).file()?.path().parent()?); let worktree_root_path = project .read(cx) .worktree_for_id(worktree_id, cx)? .read(cx) .abs_path(); let workspace_id = workspace.database_id()?; let weak = workspace.weak_handle(); cx.spawn_in(window, async move |workspace, cx| { let as_str = relative_path.to_string_lossy().into_owned(); let active_toolchain = workspace::WORKSPACE_DB .toolchain(workspace_id, worktree_id, as_str, language_name.clone()) .await .ok() .flatten(); workspace .update_in(cx, |this, window, cx| { this.toggle_modal(window, cx, move |window, cx| { ToolchainSelector::new( weak, project, active_toolchain, worktree_id, worktree_root_path, relative_path, language_name, window, cx, ) }); }) .ok(); }) .detach(); Some(()) } fn new( workspace: WeakEntity, project: Entity, active_toolchain: Option, worktree_id: WorktreeId, worktree_root: Arc, relative_path: Arc, language_name: LanguageName, window: &mut Window, cx: &mut Context, ) -> Self { let toolchain_selector = cx.entity().downgrade(); let picker = cx.new(|cx| { let delegate = ToolchainSelectorDelegate::new( active_toolchain, toolchain_selector, workspace, worktree_id, worktree_root, project, relative_path, language_name, window, cx, ); Picker::uniform_list(delegate, window, cx) }); Self { picker } } } impl Render for ToolchainSelector { fn render(&mut self, _window: &mut Window, _cx: &mut Context) -> impl IntoElement { v_flex().w(rems(34.)).child(self.picker.clone()) } } impl Focusable for ToolchainSelector { fn focus_handle(&self, cx: &App) -> FocusHandle { self.picker.focus_handle(cx) } } impl EventEmitter for ToolchainSelector {} impl ModalView for ToolchainSelector {} pub struct ToolchainSelectorDelegate { toolchain_selector: WeakEntity, candidates: ToolchainList, matches: Vec, selected_index: usize, workspace: WeakEntity, worktree_id: WorktreeId, worktree_abs_path_root: Arc, relative_path: Arc, placeholder_text: Arc, _fetch_candidates_task: Task>, } impl ToolchainSelectorDelegate { fn new( active_toolchain: Option, toolchain_selector: WeakEntity, workspace: WeakEntity, worktree_id: WorktreeId, worktree_abs_path_root: Arc, project: Entity, relative_path: Arc, language_name: LanguageName, window: &mut Window, cx: &mut Context>, ) -> Self { let _fetch_candidates_task = cx.spawn_in(window, { let project = project.clone(); async move |this, cx| { let term = project .read_with(cx, |this, _| { Project::toolchain_term(this.languages().clone(), language_name.clone()) }) .ok()? .await?; let relative_path = this .read_with(cx, |this, _| this.delegate.relative_path.clone()) .ok()?; let (available_toolchains, relative_path) = project .update(cx, |this, cx| { this.available_toolchains( ProjectPath { worktree_id, path: relative_path.clone(), }, language_name, cx, ) }) .ok()? .await?; let pretty_path = { let path = relative_path.to_string_lossy(); if path.is_empty() { Cow::Borrowed("worktree root") } else { Cow::Owned(format!("`{}`", path)) } }; let placeholder_text = format!("Select a {} for {pretty_path}…", term.to_lowercase(),).into(); let _ = this.update_in(cx, move |this, window, cx| { this.delegate.relative_path = relative_path; this.delegate.placeholder_text = placeholder_text; this.refresh_placeholder(window, cx); }); let _ = this.update_in(cx, move |this, window, cx| { this.delegate.candidates = available_toolchains; if let Some(active_toolchain) = active_toolchain { if let Some(position) = this .delegate .candidates .toolchains .iter() .position(|toolchain| *toolchain == active_toolchain) { this.delegate.set_selected_index(position, window, cx); } } this.update_matches(this.query(cx), window, cx); }); Some(()) } }); let placeholder_text = "Select a toolchain…".to_string().into(); Self { toolchain_selector, candidates: Default::default(), matches: vec![], selected_index: 0, workspace, worktree_id, worktree_abs_path_root, placeholder_text, relative_path, _fetch_candidates_task, } } fn relativize_path(path: SharedString, worktree_root: &Path) -> SharedString { Path::new(&path.as_ref()) .strip_prefix(&worktree_root) .ok() .map(|suffix| Path::new(".").join(suffix)) .and_then(|path| path.to_str().map(String::from).map(SharedString::from)) .unwrap_or(path) } } impl PickerDelegate for ToolchainSelectorDelegate { type ListItem = ListItem; fn placeholder_text(&self, _window: &mut Window, _cx: &mut App) -> Arc { self.placeholder_text.clone() } fn match_count(&self) -> usize { self.matches.len() } fn confirm(&mut self, _: bool, window: &mut Window, cx: &mut Context>) { if let Some(string_match) = self.matches.get(self.selected_index) { let toolchain = self.candidates.toolchains[string_match.candidate_id].clone(); if let Some(workspace_id) = self .workspace .read_with(cx, |this, _| this.database_id()) .ok() .flatten() { let workspace = self.workspace.clone(); let worktree_id = self.worktree_id; let path = self.relative_path.clone(); let relative_path = self.relative_path.to_string_lossy().into_owned(); cx.spawn_in(window, async move |_, cx| { workspace::WORKSPACE_DB .set_toolchain(workspace_id, worktree_id, relative_path, toolchain.clone()) .await .log_err(); workspace .update(cx, |this, cx| { this.project().update(cx, |this, cx| { this.activate_toolchain( ProjectPath { worktree_id, path }, toolchain, cx, ) }) }) .ok()? .await; Some(()) }) .detach(); } } self.dismissed(window, cx); } fn dismissed(&mut self, _: &mut Window, cx: &mut Context>) { self.toolchain_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, _window: &mut Window, _: &mut Context>, ) { self.selected_index = ix; } fn update_matches( &mut self, query: String, window: &mut Window, cx: &mut Context>, ) -> gpui::Task<()> { let background = cx.background_executor().clone(); let candidates = self.candidates.clone(); let worktree_root_path = self.worktree_abs_path_root.clone(); cx.spawn_in(window, async move |this, cx| { let matches = if query.is_empty() { candidates .toolchains .into_iter() .enumerate() .map(|(index, candidate)| { let path = Self::relativize_path(candidate.path, &worktree_root_path); let string = format!("{}{}", candidate.name, path); StringMatch { candidate_id: index, string, positions: Vec::new(), score: 0.0, } }) .collect() } else { let candidates = candidates .toolchains .into_iter() .enumerate() .map(|(candidate_id, toolchain)| { let path = Self::relativize_path(toolchain.path, &worktree_root_path); let string = format!("{}{}", toolchain.name, path); StringMatchCandidate::new(candidate_id, &string) }) .collect::>(); match_strings( &candidates, &query, false, 100, &Default::default(), background, ) .await }; this.update(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, _window: &mut Window, _: &mut Context>, ) -> Option { let mat = &self.matches[ix]; let toolchain = &self.candidates.toolchains[mat.candidate_id]; let label = toolchain.name.clone(); let path = Self::relativize_path(toolchain.path.clone(), &self.worktree_abs_path_root); let (name_highlights, mut path_highlights) = mat .positions .iter() .cloned() .partition::, _>(|index| *index < label.len()); path_highlights.iter_mut().for_each(|index| { *index -= label.len(); }); Some( ListItem::new(ix) .inset(true) .spacing(ListItemSpacing::Sparse) .toggle_state(selected) .child(HighlightedLabel::new(label, name_highlights)) .child( HighlightedLabel::new(path, path_highlights) .size(LabelSize::Small) .color(Color::Muted), ), ) } }