Detailed changes
@@ -3548,7 +3548,7 @@ mod tests {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Ok(vec![]))
}
@@ -2,7 +2,7 @@ use super::{SlashCommand, SlashCommandOutput};
use crate::prompt_library::PromptStore;
use anyhow::{anyhow, Result};
use assistant_slash_command::{ArgumentCompletion, SlashCommandOutputSection};
-use gpui::{AppContext, Task, WeakView};
+use gpui::{Task, WeakView};
use language::LspAdapterDelegate;
use std::{
fmt::Write,
@@ -35,7 +35,7 @@ impl SlashCommand for DefaultSlashCommand {
_query: String,
_cancellation_flag: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Err(anyhow!("this command does not require argument")))
}
@@ -107,7 +107,7 @@ impl SlashCommand for DiagnosticsSlashCommand {
query: String,
cancellation_flag: Arc<AtomicBool>,
workspace: Option<WeakView<Workspace>>,
- cx: &mut AppContext,
+ cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
let Some(workspace) = workspace.and_then(|workspace| workspace.upgrade()) else {
return Task::ready(Err(anyhow!("workspace was dropped")));
@@ -171,7 +171,7 @@ impl SlashCommand for DocsSlashCommand {
query: String,
_cancel: Arc<AtomicBool>,
workspace: Option<WeakView<Workspace>>,
- cx: &mut AppContext,
+ cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
self.ensure_rust_doc_providers_are_registered(workspace, cx);
@@ -8,7 +8,7 @@ use assistant_slash_command::{
ArgumentCompletion, SlashCommand, SlashCommandOutput, SlashCommandOutputSection,
};
use futures::AsyncReadExt;
-use gpui::{AppContext, Task, WeakView};
+use gpui::{Task, WeakView};
use html_to_markdown::{convert_html_to_markdown, markdown, TagHandler};
use http_client::{AsyncBody, HttpClient, HttpClientWithUrl};
use language::LspAdapterDelegate;
@@ -120,7 +120,7 @@ impl SlashCommand for FetchSlashCommand {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Ok(Vec::new()))
}
@@ -104,7 +104,7 @@ impl SlashCommand for FileSlashCommand {
query: String,
cancellation_flag: Arc<AtomicBool>,
workspace: Option<WeakView<Workspace>>,
- cx: &mut AppContext,
+ cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
let Some(workspace) = workspace.and_then(|workspace| workspace.upgrade()) else {
return Task::ready(Err(anyhow!("workspace was dropped")));
@@ -6,7 +6,7 @@ use assistant_slash_command::{
ArgumentCompletion, SlashCommand, SlashCommandOutput, SlashCommandOutputSection,
};
use chrono::Local;
-use gpui::{AppContext, Task, WeakView};
+use gpui::{Task, WeakView};
use language::LspAdapterDelegate;
use ui::prelude::*;
use workspace::Workspace;
@@ -35,7 +35,7 @@ impl SlashCommand for NowSlashCommand {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Ok(Vec::new()))
}
@@ -106,7 +106,7 @@ impl SlashCommand for ProjectSlashCommand {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Err(anyhow!("this command does not require argument")))
}
@@ -2,7 +2,7 @@ use super::{SlashCommand, SlashCommandOutput};
use crate::prompt_library::PromptStore;
use anyhow::{anyhow, Context, Result};
use assistant_slash_command::{ArgumentCompletion, SlashCommandOutputSection};
-use gpui::{AppContext, Task, WeakView};
+use gpui::{Task, WeakView};
use language::LspAdapterDelegate;
use std::sync::{atomic::AtomicBool, Arc};
use ui::prelude::*;
@@ -32,7 +32,7 @@ impl SlashCommand for PromptSlashCommand {
query: String,
_cancellation_flag: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- cx: &mut AppContext,
+ cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
let store = PromptStore::global(cx);
cx.background_executor().spawn(async move {
@@ -52,7 +52,7 @@ impl SlashCommand for SearchSlashCommand {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Ok(Vec::new()))
}
@@ -2,7 +2,7 @@ use super::{SlashCommand, SlashCommandOutput};
use anyhow::{anyhow, Context as _, Result};
use assistant_slash_command::{ArgumentCompletion, SlashCommandOutputSection};
use editor::Editor;
-use gpui::{AppContext, Task, WeakView};
+use gpui::{Task, WeakView};
use language::LspAdapterDelegate;
use std::sync::Arc;
use std::{path::Path, sync::atomic::AtomicBool};
@@ -29,7 +29,7 @@ impl SlashCommand for OutlineSlashCommand {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Err(anyhow!("this command does not require argument")))
}
@@ -3,55 +3,23 @@ use super::{
file_command::{build_entry_output_section, codeblock_fence_for_path},
SlashCommand, SlashCommandOutput,
};
-use anyhow::Result;
+use anyhow::{Context, Result};
use assistant_slash_command::ArgumentCompletion;
use collections::HashMap;
use editor::Editor;
-use gpui::{AppContext, Entity, Task, WeakView};
-use language::LspAdapterDelegate;
-use std::{fmt::Write, sync::Arc};
+use gpui::{Entity, Task, WeakView};
+use language::{BufferSnapshot, LspAdapterDelegate};
+use std::{
+ fmt::Write,
+ path::PathBuf,
+ sync::{atomic::AtomicBool, Arc},
+};
use ui::WindowContext;
use workspace::Workspace;
pub(crate) struct TabsSlashCommand;
-#[derive(Clone, Copy, Debug, Default, PartialEq, Eq)]
-enum TabsArgument {
- #[default]
- Active,
- All,
-}
-
-impl TabsArgument {
- fn for_query(mut query: String) -> Vec<Self> {
- query.make_ascii_lowercase();
- let query = query.trim();
-
- let mut matches = Vec::new();
- if Self::Active.name().contains(&query) {
- matches.push(Self::Active);
- }
- if Self::All.name().contains(&query) {
- matches.push(Self::All);
- }
- matches
- }
-
- fn name(&self) -> &'static str {
- match self {
- Self::Active => "active",
- Self::All => "all",
- }
- }
-
- fn from_name(name: &str) -> Option<Self> {
- match name {
- "active" => Some(Self::Active),
- "all" => Some(Self::All),
- _ => None,
- }
- }
-}
+const ALL_TABS_COMPLETION_ITEM: &str = "all";
impl SlashCommand for TabsSlashCommand {
fn name(&self) -> String {
@@ -59,33 +27,52 @@ impl SlashCommand for TabsSlashCommand {
}
fn description(&self) -> String {
- "insert open tabs".into()
+ "insert open tabs (active tab by default)".to_owned()
}
fn menu_text(&self) -> String {
- "Insert Open Tabs".into()
+ "Insert Open Tabs".to_owned()
}
fn requires_argument(&self) -> bool {
- true
+ false
}
fn complete_argument(
self: Arc<Self>,
query: String,
- _cancel: Arc<std::sync::atomic::AtomicBool>,
- _workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ cancel: Arc<AtomicBool>,
+ workspace: Option<WeakView<Workspace>>,
+ cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
- let arguments = TabsArgument::for_query(query);
- Task::ready(Ok(arguments
- .into_iter()
- .map(|arg| ArgumentCompletion {
- label: arg.name().to_owned(),
- new_text: arg.name().to_owned(),
+ let all_tabs_completion_item = if ALL_TABS_COMPLETION_ITEM.contains(&query) {
+ Some(ArgumentCompletion {
+ label: ALL_TABS_COMPLETION_ITEM.to_owned(),
+ new_text: ALL_TABS_COMPLETION_ITEM.to_owned(),
run_command: true,
})
- .collect()))
+ } else {
+ None
+ };
+ let tab_items_search = tab_items_for_query(workspace, query, cancel, false, cx);
+ cx.spawn(|_| async move {
+ let tab_completion_items =
+ tab_items_search
+ .await?
+ .into_iter()
+ .filter_map(|(path, ..)| {
+ let path_string = path.as_deref()?.to_string_lossy().to_string();
+ Some(ArgumentCompletion {
+ label: path_string.clone(),
+ new_text: path_string,
+ run_command: true,
+ })
+ });
+ Ok(all_tabs_completion_item
+ .into_iter()
+ .chain(tab_completion_items)
+ .collect::<Vec<_>>())
+ })
}
fn run(
@@ -95,89 +82,146 @@ impl SlashCommand for TabsSlashCommand {
_delegate: Option<Arc<dyn LspAdapterDelegate>>,
cx: &mut WindowContext,
) -> Task<Result<SlashCommandOutput>> {
- let argument = argument
- .and_then(TabsArgument::from_name)
- .unwrap_or_default();
- let open_buffers = workspace.update(cx, |workspace, cx| match argument {
- TabsArgument::Active => {
- let Some(active_item) = workspace.active_item(cx) else {
- anyhow::bail!("no active item")
- };
- let Some(buffer) = active_item
- .downcast::<Editor>()
- .and_then(|editor| editor.read(cx).buffer().read(cx).as_singleton())
- else {
- anyhow::bail!("active item is not an editor")
- };
- let snapshot = buffer.read(cx).snapshot();
- let full_path = snapshot.resolve_file_path(cx, true);
- anyhow::Ok(vec![(full_path, snapshot, 0)])
- }
- TabsArgument::All => {
- let mut timestamps_by_entity_id = HashMap::default();
- let mut open_buffers = Vec::new();
-
- for pane in workspace.panes() {
- let pane = pane.read(cx);
- for entry in pane.activation_history() {
- timestamps_by_entity_id.insert(entry.entity_id, entry.timestamp);
- }
+ let tab_items_search = tab_items_for_query(
+ Some(workspace),
+ argument.map(ToOwned::to_owned).unwrap_or_default(),
+ Arc::new(AtomicBool::new(false)),
+ true,
+ cx,
+ );
+
+ cx.background_executor().spawn(async move {
+ let mut sections = Vec::new();
+ let mut text = String::new();
+ let mut has_diagnostics = false;
+ for (full_path, buffer, _) in tab_items_search.await? {
+ let section_start_ix = text.len();
+ text.push_str(&codeblock_fence_for_path(full_path.as_deref(), None));
+ for chunk in buffer.as_rope().chunks() {
+ text.push_str(chunk);
}
-
- for editor in workspace.items_of_type::<Editor>(cx) {
- if let Some(buffer) = editor.read(cx).buffer().read(cx).as_singleton() {
- if let Some(timestamp) = timestamps_by_entity_id.get(&editor.entity_id()) {
- let snapshot = buffer.read(cx).snapshot();
- let full_path = snapshot.resolve_file_path(cx, true);
- open_buffers.push((full_path, snapshot, *timestamp));
- }
- }
+ if !text.ends_with('\n') {
+ text.push('\n');
+ }
+ writeln!(text, "```").unwrap();
+ if write_single_file_diagnostics(&mut text, full_path.as_deref(), &buffer) {
+ has_diagnostics = true;
+ }
+ if !text.ends_with('\n') {
+ text.push('\n');
}
- Ok(open_buffers)
+ let section_end_ix = text.len() - 1;
+ sections.push(build_entry_output_section(
+ section_start_ix..section_end_ix,
+ full_path.as_deref(),
+ false,
+ None,
+ ));
}
- });
- match open_buffers {
- Ok(Ok(mut open_buffers)) => cx.background_executor().spawn(async move {
- open_buffers.sort_by_key(|(_, _, timestamp)| *timestamp);
+ Ok(SlashCommandOutput {
+ text,
+ sections,
+ run_commands_in_text: has_diagnostics,
+ })
+ })
+ }
+}
- let mut sections = Vec::new();
- let mut text = String::new();
- let mut has_diagnostics = false;
- for (full_path, buffer, _) in open_buffers {
- let section_start_ix = text.len();
- text.push_str(&codeblock_fence_for_path(full_path.as_deref(), None));
- for chunk in buffer.as_rope().chunks() {
- text.push_str(chunk);
- }
- if !text.ends_with('\n') {
- text.push('\n');
+fn tab_items_for_query(
+ workspace: Option<WeakView<Workspace>>,
+ mut query: String,
+ cancel: Arc<AtomicBool>,
+ use_active_tab_for_empty_query: bool,
+ cx: &mut WindowContext,
+) -> Task<anyhow::Result<Vec<(Option<PathBuf>, BufferSnapshot, usize)>>> {
+ cx.spawn(|mut cx| async move {
+ query.make_ascii_lowercase();
+ let mut open_buffers =
+ workspace
+ .context("no workspace")?
+ .update(&mut cx, |workspace, cx| {
+ if use_active_tab_for_empty_query && query.trim().is_empty() {
+ let active_editor = workspace
+ .active_item(cx)
+ .context("no active item")?
+ .downcast::<Editor>()
+ .context("active item is not an editor")?;
+ let snapshot = active_editor
+ .read(cx)
+ .buffer()
+ .read(cx)
+ .as_singleton()
+ .context("active editor is not a singleton buffer")?
+ .read(cx)
+ .snapshot();
+ let full_path = snapshot.resolve_file_path(cx, true);
+ return anyhow::Ok(vec![(full_path, snapshot, 0)]);
}
- writeln!(text, "```").unwrap();
- if write_single_file_diagnostics(&mut text, full_path.as_deref(), &buffer) {
- has_diagnostics = true;
+
+ let mut timestamps_by_entity_id = HashMap::default();
+ let mut open_buffers = Vec::new();
+
+ for pane in workspace.panes() {
+ let pane = pane.read(cx);
+ for entry in pane.activation_history() {
+ timestamps_by_entity_id.insert(entry.entity_id, entry.timestamp);
+ }
}
- if !text.ends_with('\n') {
- text.push('\n');
+
+ for editor in workspace.items_of_type::<Editor>(cx) {
+ if let Some(buffer) = editor.read(cx).buffer().read(cx).as_singleton() {
+ if let Some(timestamp) =
+ timestamps_by_entity_id.get(&editor.entity_id())
+ {
+ let snapshot = buffer.read(cx).snapshot();
+ let full_path = snapshot.resolve_file_path(cx, true);
+ open_buffers.push((full_path, snapshot, *timestamp));
+ }
+ }
}
- let section_end_ix = text.len() - 1;
- sections.push(build_entry_output_section(
- section_start_ix..section_end_ix,
- full_path.as_deref(),
- false,
- None,
- ));
+ Ok(open_buffers)
+ })??;
+
+ let background_executor = cx.background_executor().clone();
+ cx.background_executor()
+ .spawn(async move {
+ open_buffers.sort_by_key(|(_, _, timestamp)| *timestamp);
+ let query = query.trim();
+ if query.is_empty() || query == ALL_TABS_COMPLETION_ITEM {
+ return Ok(open_buffers);
}
- Ok(SlashCommandOutput {
- text,
- sections,
- run_commands_in_text: has_diagnostics,
- })
- }),
- Ok(Err(error)) | Err(error) => Task::ready(Err(error)),
- }
- }
+ let match_candidates = open_buffers
+ .iter()
+ .enumerate()
+ .filter_map(|(id, (full_path, ..))| {
+ let path_string = full_path.as_deref()?.to_string_lossy().to_string();
+ Some(fuzzy::StringMatchCandidate {
+ id,
+ char_bag: path_string.as_str().into(),
+ string: path_string,
+ })
+ })
+ .collect::<Vec<_>>();
+ let string_matches = fuzzy::match_strings(
+ &match_candidates,
+ &query,
+ true,
+ usize::MAX,
+ &cancel,
+ background_executor,
+ )
+ .await;
+
+ Ok(string_matches
+ .into_iter()
+ .filter_map(|string_match| open_buffers.get(string_match.candidate_id))
+ .cloned()
+ .collect())
+ })
+ .await
+ })
}
@@ -45,7 +45,7 @@ impl SlashCommand for TerminalSlashCommand {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Ok(vec![ArgumentCompletion {
label: LINE_COUNT_ARG.to_string(),
@@ -7,7 +7,7 @@ use anyhow::Result;
use assistant_slash_command::{
ArgumentCompletion, SlashCommand, SlashCommandOutput, SlashCommandOutputSection,
};
-use gpui::{AppContext, Task, WeakView};
+use gpui::{Task, WeakView};
use language::LspAdapterDelegate;
use ui::prelude::*;
@@ -45,7 +45,7 @@ impl SlashCommand for WorkflowSlashCommand {
_query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- _cx: &mut AppContext,
+ _cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
Task::ready(Ok(Vec::new()))
}
@@ -37,7 +37,7 @@ pub trait SlashCommand: 'static + Send + Sync {
query: String,
cancel: Arc<AtomicBool>,
workspace: Option<WeakView<Workspace>>,
- cx: &mut AppContext,
+ cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>>;
fn requires_argument(&self) -> bool;
fn run(
@@ -5,7 +5,7 @@ use assistant_slash_command::{
ArgumentCompletion, SlashCommand, SlashCommandOutput, SlashCommandOutputSection,
};
use futures::FutureExt;
-use gpui::{AppContext, Task, WeakView, WindowContext};
+use gpui::{Task, WeakView, WindowContext};
use language::LspAdapterDelegate;
use ui::prelude::*;
use wasmtime_wasi::WasiView;
@@ -42,7 +42,7 @@ impl SlashCommand for ExtensionSlashCommand {
query: String,
_cancel: Arc<AtomicBool>,
_workspace: Option<WeakView<Workspace>>,
- cx: &mut AppContext,
+ cx: &mut WindowContext,
) -> Task<Result<Vec<ArgumentCompletion>>> {
cx.background_executor().spawn(async move {
self.extension