use std::str::FromStr; use std::sync::Arc; use client::ExtensionMetadata; use extension_host::{ExtensionSettings, ExtensionStore}; use fs::Fs; use fuzzy::{StringMatch, StringMatchCandidate, match_strings}; use gpui::{App, DismissEvent, Entity, EventEmitter, Focusable, Task, WeakEntity, prelude::*}; use picker::{Picker, PickerDelegate}; use release_channel::ReleaseChannel; use semantic_version::SemanticVersion; use settings::update_settings_file; use ui::{HighlightedLabel, ListItem, ListItemSpacing, prelude::*}; use util::ResultExt; use workspace::ModalView; pub struct ExtensionVersionSelector { picker: Entity>, } impl ModalView for ExtensionVersionSelector {} impl EventEmitter for ExtensionVersionSelector {} impl Focusable for ExtensionVersionSelector { fn focus_handle(&self, cx: &App) -> gpui::FocusHandle { self.picker.focus_handle(cx) } } impl Render for ExtensionVersionSelector { fn render(&mut self, _window: &mut Window, _cx: &mut Context) -> impl IntoElement { v_flex().w(rems(34.)).child(self.picker.clone()) } } impl ExtensionVersionSelector { pub fn new( delegate: ExtensionVersionSelectorDelegate, window: &mut Window, cx: &mut Context, ) -> Self { let picker = cx.new(|cx| Picker::uniform_list(delegate, window, cx)); Self { picker } } } pub struct ExtensionVersionSelectorDelegate { fs: Arc, selector: WeakEntity, extension_versions: Vec, selected_index: usize, matches: Vec, } impl ExtensionVersionSelectorDelegate { pub fn new( fs: Arc, selector: WeakEntity, mut extension_versions: Vec, ) -> Self { extension_versions.sort_unstable_by(|a, b| { let a_version = SemanticVersion::from_str(&a.manifest.version); let b_version = SemanticVersion::from_str(&b.manifest.version); match (a_version, b_version) { (Ok(a_version), Ok(b_version)) => b_version.cmp(&a_version), _ => b.published_at.cmp(&a.published_at), } }); let matches = extension_versions .iter() .map(|extension| StringMatch { candidate_id: 0, score: 0.0, positions: Default::default(), string: format!("v{}", extension.manifest.version), }) .collect(); Self { fs, selector, extension_versions, selected_index: 0, matches, } } } impl PickerDelegate for ExtensionVersionSelectorDelegate { type ListItem = ui::ListItem; fn placeholder_text(&self, _window: &mut Window, _cx: &mut App) -> Arc { "Select extension version...".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, _window: &mut Window, _cx: &mut Context>, ) { self.selected_index = ix; } fn update_matches( &mut self, query: String, window: &mut Window, cx: &mut Context>, ) -> Task<()> { let background_executor = cx.background_executor().clone(); let candidates = self .extension_versions .iter() .enumerate() .map(|(id, extension)| { StringMatchCandidate::new(id, &format!("v{}", extension.manifest.version)) }) .collect::>(); cx.spawn_in(window, async move |this, cx| { 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, true, 100, &Default::default(), background_executor, ) .await }; this.update(cx, |this, _cx| { 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, _secondary: bool, window: &mut Window, cx: &mut Context>) { if self.matches.is_empty() { self.dismissed(window, cx); return; } let candidate_id = self.matches[self.selected_index].candidate_id; let extension_version = &self.extension_versions[candidate_id]; if !extension_host::is_version_compatible(ReleaseChannel::global(cx), extension_version) { return; } let extension_store = ExtensionStore::global(cx); extension_store.update(cx, |store, cx| { let extension_id = extension_version.id.clone(); let version = extension_version.manifest.version.clone(); update_settings_file::(self.fs.clone(), cx, { let extension_id = extension_id.clone(); move |settings, _| { settings.auto_update_extensions.insert(extension_id, false); } }); store.install_extension(extension_id, version, cx); }); } fn dismissed(&mut self, _: &mut Window, cx: &mut Context>) { self.selector .update(cx, |_, cx| cx.emit(DismissEvent)) .log_err(); } fn render_match( &self, ix: usize, selected: bool, _: &mut Window, cx: &mut Context>, ) -> Option { let version_match = &self.matches[ix]; let extension_version = &self.extension_versions[version_match.candidate_id]; let is_version_compatible = extension_host::is_version_compatible(ReleaseChannel::global(cx), extension_version); let disabled = !is_version_compatible; Some( ListItem::new(ix) .inset(true) .spacing(ListItemSpacing::Sparse) .toggle_state(selected) .disabled(disabled) .child( HighlightedLabel::new( version_match.string.clone(), version_match.positions.clone(), ) .when(disabled, |label| label.color(Color::Muted)), ) .end_slot( h_flex() .gap_2() .when(!is_version_compatible, |this| { this.child(Label::new("Incompatible").color(Color::Muted)) }) .child( Label::new( extension_version .published_at .format("%Y-%m-%d") .to_string(), ) .when(disabled, |label| label.color(Color::Muted)), ), ), ) } }