use anyhow::Result; use client::UserStore; use copilot::{Copilot, Status}; use editor::{actions::ShowInlineCompletion, scroll::Autoscroll, Editor}; use feature_flags::{ FeatureFlagAppExt, PredictEditsFeatureFlag, PredictEditsRateCompletionsFeatureFlag, }; use fs::Fs; use gpui::{ actions, div, pulsating_between, Action, Animation, AnimationExt, App, AsyncWindowContext, Corner, Entity, FocusHandle, Focusable, IntoElement, ParentElement, Render, Subscription, WeakEntity, }; use language::{ language_settings::{ self, all_language_settings, AllLanguageSettings, InlineCompletionProvider, }, File, Language, }; use settings::{update_settings_file, Settings, SettingsStore}; use std::{path::Path, sync::Arc, time::Duration}; use supermaven::{AccountStatus, Supermaven}; use ui::{ prelude::*, Clickable, ContextMenu, ContextMenuEntry, IconButton, IconButtonShape, PopoverMenu, PopoverMenuHandle, Tooltip, }; use workspace::{ create_and_open_local_file, item::ItemHandle, notifications::NotificationId, StatusItemView, Toast, Workspace, }; use zed_actions::OpenBrowser; use zeta::RateCompletionModal; actions!(zeta, [RateCompletions]); actions!(inline_completion, [ToggleMenu]); const COPILOT_SETTINGS_URL: &str = "https://github.com/settings/copilot"; struct CopilotErrorToast; pub struct InlineCompletionButton { editor_subscription: Option<(Subscription, usize)>, editor_enabled: Option, editor_focus_handle: Option, language: Option>, file: Option>, inline_completion_provider: Option>, fs: Arc, workspace: WeakEntity, user_store: Entity, popover_menu_handle: PopoverMenuHandle, } enum SupermavenButtonStatus { Ready, Errored(String), NeedsActivation(String), Initializing, } impl Render for InlineCompletionButton { fn render(&mut self, _: &mut Window, cx: &mut Context) -> impl IntoElement { let all_language_settings = all_language_settings(None, cx); match all_language_settings.inline_completions.provider { InlineCompletionProvider::None => div(), InlineCompletionProvider::Copilot => { let Some(copilot) = Copilot::global(cx) else { return div(); }; let status = copilot.read(cx).status(); let enabled = self.editor_enabled.unwrap_or_else(|| { all_language_settings.inline_completions_enabled(None, None, cx) }); let icon = match status { Status::Error(_) => IconName::CopilotError, Status::Authorized => { if enabled { IconName::Copilot } else { IconName::CopilotDisabled } } _ => IconName::CopilotInit, }; if let Status::Error(e) = status { return div().child( IconButton::new("copilot-error", icon) .icon_size(IconSize::Small) .on_click(cx.listener(move |_, _, window, cx| { if let Some(workspace) = window.root::().flatten() { workspace.update(cx, |workspace, cx| { workspace.show_toast( Toast::new( NotificationId::unique::(), format!("Copilot can't be started: {}", e), ) .on_click( "Reinstall Copilot", |_, cx| { if let Some(copilot) = Copilot::global(cx) { copilot .update(cx, |copilot, cx| { copilot.reinstall(cx) }) .detach(); } }, ), cx, ); }); } })) .tooltip(|window, cx| { Tooltip::for_action("GitHub Copilot", &ToggleMenu, window, cx) }), ); } let this = cx.entity().clone(); div().child( PopoverMenu::new("copilot") .menu(move |window, cx| { Some(match status { Status::Authorized => this.update(cx, |this, cx| { this.build_copilot_context_menu(window, cx) }), _ => this.update(cx, |this, cx| { this.build_copilot_start_menu(window, cx) }), }) }) .anchor(Corner::BottomRight) .trigger(IconButton::new("copilot-icon", icon).tooltip(|window, cx| { Tooltip::for_action("GitHub Copilot", &ToggleMenu, window, cx) })) .with_handle(self.popover_menu_handle.clone()), ) } InlineCompletionProvider::Supermaven => { let Some(supermaven) = Supermaven::global(cx) else { return div(); }; let supermaven = supermaven.read(cx); let status = match supermaven { Supermaven::Starting => SupermavenButtonStatus::Initializing, Supermaven::FailedDownload { error } => { SupermavenButtonStatus::Errored(error.to_string()) } Supermaven::Spawned(agent) => { let account_status = agent.account_status.clone(); match account_status { AccountStatus::NeedsActivation { activate_url } => { SupermavenButtonStatus::NeedsActivation(activate_url.clone()) } AccountStatus::Unknown => SupermavenButtonStatus::Initializing, AccountStatus::Ready => SupermavenButtonStatus::Ready, } } Supermaven::Error { error } => { SupermavenButtonStatus::Errored(error.to_string()) } }; let icon = status.to_icon(); let tooltip_text = status.to_tooltip(); let has_menu = status.has_menu(); let this = cx.entity().clone(); let fs = self.fs.clone(); return div().child( PopoverMenu::new("supermaven") .menu(move |window, cx| match &status { SupermavenButtonStatus::NeedsActivation(activate_url) => { Some(ContextMenu::build(window, cx, |menu, _, _| { let fs = fs.clone(); let activate_url = activate_url.clone(); menu.entry("Sign In", None, move |_, cx| { cx.open_url(activate_url.as_str()) }) .entry( "Use Copilot", None, move |_, cx| { set_completion_provider( fs.clone(), cx, InlineCompletionProvider::Copilot, ) }, ) })) } SupermavenButtonStatus::Ready => Some(this.update(cx, |this, cx| { this.build_supermaven_context_menu(window, cx) })), _ => None, }) .anchor(Corner::BottomRight) .trigger(IconButton::new("supermaven-icon", icon).tooltip( move |window, cx| { if has_menu { Tooltip::for_action( tooltip_text.clone(), &ToggleMenu, window, cx, ) } else { Tooltip::text(tooltip_text.clone())(window, cx) } }, )) .with_handle(self.popover_menu_handle.clone()), ); } InlineCompletionProvider::Zed => { if !cx.has_flag::() { return div(); } fn icon_button() -> IconButton { IconButton::new("zed-predict-pending-button", IconName::ZedPredict) .shape(IconButtonShape::Square) } let current_user_terms_accepted = self.user_store.read(cx).current_user_has_accepted_terms(); if !current_user_terms_accepted.unwrap_or(false) { let signed_in = current_user_terms_accepted.is_some(); let tooltip_meta = if signed_in { "Read Terms of Service" } else { "Sign in to use" }; return div().child( icon_button() .tooltip(move |window, cx| { Tooltip::with_meta( "Edit Predictions", None, tooltip_meta, window, cx, ) }) .on_click(cx.listener(move |_, _, window, cx| { window.dispatch_action( zed_actions::OpenZedPredictOnboarding.boxed_clone(), cx, ); })), ); } let this = cx.entity().clone(); if !self.popover_menu_handle.is_deployed() { icon_button().tooltip(|window, cx| { Tooltip::for_action("Edit Prediction", &ToggleMenu, window, cx) }); } let mut popover_menu = PopoverMenu::new("zeta") .menu(move |window, cx| { Some(this.update(cx, |this, cx| this.build_zeta_context_menu(window, cx))) }) .anchor(Corner::BottomRight) .with_handle(self.popover_menu_handle.clone()); let is_refreshing = self .inline_completion_provider .as_ref() .map_or(false, |provider| provider.is_refreshing(cx)); if is_refreshing { popover_menu = popover_menu.trigger( icon_button().with_animation( "pulsating-label", Animation::new(Duration::from_secs(2)) .repeat() .with_easing(pulsating_between(0.2, 1.0)), |icon_button, delta| icon_button.alpha(delta), ), ); } else { popover_menu = popover_menu.trigger(icon_button()); } div().child(popover_menu.into_any_element()) } } } } impl InlineCompletionButton { pub fn new( workspace: WeakEntity, fs: Arc, user_store: Entity, popover_menu_handle: PopoverMenuHandle, cx: &mut Context, ) -> Self { if let Some(copilot) = Copilot::global(cx) { cx.observe(&copilot, |_, _, cx| cx.notify()).detach() } cx.observe_global::(move |_, cx| cx.notify()) .detach(); Self { editor_subscription: None, editor_enabled: None, editor_focus_handle: None, language: None, file: None, inline_completion_provider: None, popover_menu_handle, workspace, fs, user_store, } } pub fn build_copilot_start_menu( &mut self, window: &mut Window, cx: &mut Context, ) -> Entity { let fs = self.fs.clone(); ContextMenu::build(window, cx, |menu, _, _| { menu.entry("Sign In", None, copilot::initiate_sign_in) .entry("Disable Copilot", None, { let fs = fs.clone(); move |_window, cx| hide_copilot(fs.clone(), cx) }) .entry("Use Supermaven", None, { let fs = fs.clone(); move |_window, cx| { set_completion_provider( fs.clone(), cx, InlineCompletionProvider::Supermaven, ) } }) }) } // Predict Edits at Cursor – alt-tab // Automatically Predict: // ✓ PATH // ✓ Rust // ✓ All Files pub fn build_language_settings_menu(&self, mut menu: ContextMenu, cx: &mut App) -> ContextMenu { let fs = self.fs.clone(); menu = menu.header("Predict Edits For:"); if let Some(language) = self.language.clone() { let fs = fs.clone(); let language_enabled = language_settings::language_settings(Some(language.name()), None, cx) .show_inline_completions; menu = menu.toggleable_entry( language.name(), language_enabled, IconPosition::Start, None, move |_, cx| { toggle_inline_completions_for_language(language.clone(), fs.clone(), cx) }, ); } let settings = AllLanguageSettings::get_global(cx); if let Some(file) = &self.file { let path = file.path().clone(); let path_enabled = settings.inline_completions_enabled_for_path(&path); menu = menu.toggleable_entry( "This File", path_enabled, IconPosition::Start, None, move |window, cx| { if let Some(workspace) = window.root().flatten() { let workspace = workspace.downgrade(); window .spawn(cx, |cx| { configure_disabled_globs( workspace, path_enabled.then_some(path.clone()), cx, ) }) .detach_and_log_err(cx); } }, ); } let globally_enabled = settings.inline_completions_enabled(None, None, cx); menu = menu.toggleable_entry( "All Files", globally_enabled, IconPosition::Start, None, move |_, cx| toggle_inline_completions_globally(fs.clone(), cx), ); if let Some(provider) = &self.inline_completion_provider { let data_collection = provider.data_collection_state(cx); if data_collection.is_supported() { let provider = provider.clone(); menu = menu .separator() .header("Help Improve The Model") .header("For OSS Projects Only"); menu = menu.item( // TODO: We want to add something later that communicates whether // the current project is open-source. ContextMenuEntry::new("Share Training Data") .toggleable(IconPosition::Start, data_collection.is_enabled()) .handler(move |_, cx| { provider.toggle_data_collection(cx); }), ); } } if let Some(editor_focus_handle) = self.editor_focus_handle.clone() { menu = menu .separator() .entry( "Predict Edit at Cursor", Some(Box::new(ShowInlineCompletion)), { let editor_focus_handle = editor_focus_handle.clone(); move |window, cx| { editor_focus_handle.dispatch_action(&ShowInlineCompletion, window, cx); } }, ) .context(editor_focus_handle); } menu } fn build_copilot_context_menu( &self, window: &mut Window, cx: &mut Context, ) -> Entity { ContextMenu::build(window, cx, |menu, _, cx| { self.build_language_settings_menu(menu, cx) .separator() .link( "Go to Copilot Settings", OpenBrowser { url: COPILOT_SETTINGS_URL.to_string(), } .boxed_clone(), ) .action("Sign Out", copilot::SignOut.boxed_clone()) }) } fn build_supermaven_context_menu( &self, window: &mut Window, cx: &mut Context, ) -> Entity { ContextMenu::build(window, cx, |menu, _, cx| { self.build_language_settings_menu(menu, cx) .separator() .action("Sign Out", supermaven::SignOut.boxed_clone()) }) } fn build_zeta_context_menu( &self, window: &mut Window, cx: &mut Context, ) -> Entity { let workspace = self.workspace.clone(); ContextMenu::build(window, cx, |menu, _window, cx| { self.build_language_settings_menu(menu, cx).when( cx.has_flag::(), |this| { this.entry( "Rate Completions", Some(RateCompletions.boxed_clone()), move |window, cx| { workspace .update(cx, |workspace, cx| { RateCompletionModal::toggle(workspace, window, cx) }) .ok(); }, ) }, ) }) } pub fn update_enabled(&mut self, editor: Entity, cx: &mut Context) { let editor = editor.read(cx); let snapshot = editor.buffer().read(cx).snapshot(cx); let suggestion_anchor = editor.selections.newest_anchor().start; let language = snapshot.language_at(suggestion_anchor); let file = snapshot.file_at(suggestion_anchor).cloned(); self.editor_enabled = { let file = file.as_ref(); Some( file.map(|file| !file.is_private()).unwrap_or(true) && all_language_settings(file, cx).inline_completions_enabled( language, file.map(|file| file.path().as_ref()), cx, ), ) }; self.inline_completion_provider = editor.inline_completion_provider(); self.language = language.cloned(); self.file = file; self.editor_focus_handle = Some(editor.focus_handle(cx)); cx.notify(); } pub fn toggle_menu(&mut self, window: &mut Window, cx: &mut Context) { self.popover_menu_handle.toggle(window, cx); } } impl StatusItemView for InlineCompletionButton { fn set_active_pane_item( &mut self, item: Option<&dyn ItemHandle>, _: &mut Window, cx: &mut Context, ) { if let Some(editor) = item.and_then(|item| item.act_as::(cx)) { self.editor_subscription = Some(( cx.observe(&editor, Self::update_enabled), editor.entity_id().as_u64() as usize, )); self.update_enabled(editor, cx); } else { self.language = None; self.editor_subscription = None; self.editor_enabled = None; } cx.notify(); } } impl SupermavenButtonStatus { fn to_icon(&self) -> IconName { match self { SupermavenButtonStatus::Ready => IconName::Supermaven, SupermavenButtonStatus::Errored(_) => IconName::SupermavenError, SupermavenButtonStatus::NeedsActivation(_) => IconName::SupermavenInit, SupermavenButtonStatus::Initializing => IconName::SupermavenInit, } } fn to_tooltip(&self) -> String { match self { SupermavenButtonStatus::Ready => "Supermaven is ready".to_string(), SupermavenButtonStatus::Errored(error) => format!("Supermaven error: {}", error), SupermavenButtonStatus::NeedsActivation(_) => "Supermaven needs activation".to_string(), SupermavenButtonStatus::Initializing => "Supermaven initializing".to_string(), } } fn has_menu(&self) -> bool { match self { SupermavenButtonStatus::Ready | SupermavenButtonStatus::NeedsActivation(_) => true, SupermavenButtonStatus::Errored(_) | SupermavenButtonStatus::Initializing => false, } } } async fn configure_disabled_globs( workspace: WeakEntity, path_to_disable: Option>, mut cx: AsyncWindowContext, ) -> Result<()> { let settings_editor = workspace .update_in(&mut cx, |_, window, cx| { create_and_open_local_file(paths::settings_file(), window, cx, || { settings::initial_user_settings_content().as_ref().into() }) })? .await? .downcast::() .unwrap(); settings_editor .downgrade() .update_in(&mut cx, |item, window, cx| { let text = item.buffer().read(cx).snapshot(cx).text(); let settings = cx.global::(); let edits = settings.edits_for_update::(&text, |file| { let copilot = file.inline_completions.get_or_insert_with(Default::default); let globs = copilot.disabled_globs.get_or_insert_with(|| { settings .get::(None) .inline_completions .disabled_globs .iter() .map(|glob| glob.glob().to_string()) .collect() }); if let Some(path_to_disable) = &path_to_disable { globs.push(path_to_disable.to_string_lossy().into_owned()); } else { globs.clear(); } }); if !edits.is_empty() { item.change_selections(Some(Autoscroll::newest()), window, cx, |selections| { selections.select_ranges(edits.iter().map(|e| e.0.clone())); }); // When *enabling* a path, don't actually perform an edit, just select the range. if path_to_disable.is_some() { item.edit(edits.iter().cloned(), cx); } } })?; anyhow::Ok(()) } fn toggle_inline_completions_globally(fs: Arc, cx: &mut App) { let show_inline_completions = all_language_settings(None, cx).inline_completions_enabled(None, None, cx); update_settings_file::(fs, cx, move |file, _| { file.defaults.show_inline_completions = Some(!show_inline_completions) }); } fn set_completion_provider(fs: Arc, cx: &mut App, provider: InlineCompletionProvider) { update_settings_file::(fs, cx, move |file, _| { file.features .get_or_insert(Default::default()) .inline_completion_provider = Some(provider); }); } fn toggle_inline_completions_for_language(language: Arc, fs: Arc, cx: &mut App) { let show_inline_completions = all_language_settings(None, cx).inline_completions_enabled(Some(&language), None, cx); update_settings_file::(fs, cx, move |file, _| { file.languages .entry(language.name()) .or_default() .show_inline_completions = Some(!show_inline_completions); }); } fn hide_copilot(fs: Arc, cx: &mut App) { update_settings_file::(fs, cx, move |file, _| { file.features .get_or_insert(Default::default()) .inline_completion_provider = Some(InlineCompletionProvider::None); }); }