This PR improves the completions for arguments in the `/docs` slash command. We achieved this by extending the `complete_argument` method on the `SlashCommand` trait to return a `Vec<ArgumentCompletion>` instead of a `Vec<String>`. In addition to the completion `label`, `ArgumentCompletion` has two new fields that are can be used to customize the completion behavior: - `new_text`: The actual text that will be inserted when the completion is accepted, which may be different from what is shown by the completion label. - `run_command`: Whether the command is run when the completion is accepted. This can be set to `false` to allow accepting a completion without running the command. Release Notes: - N/A --------- Co-authored-by: Antonio <antonio@zed.dev>
103 lines
3.4 KiB
Rust
103 lines
3.4 KiB
Rust
use super::{
|
|
diagnostics_command::write_single_file_diagnostics,
|
|
file_command::{build_entry_output_section, codeblock_fence_for_path},
|
|
SlashCommand, SlashCommandOutput,
|
|
};
|
|
use anyhow::{anyhow, Result};
|
|
use assistant_slash_command::ArgumentCompletion;
|
|
use editor::Editor;
|
|
use gpui::{AppContext, Task, WeakView};
|
|
use language::LspAdapterDelegate;
|
|
use std::sync::atomic::AtomicBool;
|
|
use std::sync::Arc;
|
|
use ui::WindowContext;
|
|
use workspace::Workspace;
|
|
|
|
pub(crate) struct ActiveSlashCommand;
|
|
|
|
impl SlashCommand for ActiveSlashCommand {
|
|
fn name(&self) -> String {
|
|
"active".into()
|
|
}
|
|
|
|
fn description(&self) -> String {
|
|
"insert active tab".into()
|
|
}
|
|
|
|
fn menu_text(&self) -> String {
|
|
"Insert Active Tab".into()
|
|
}
|
|
|
|
fn complete_argument(
|
|
self: Arc<Self>,
|
|
_query: String,
|
|
_cancel: Arc<AtomicBool>,
|
|
_workspace: Option<WeakView<Workspace>>,
|
|
_cx: &mut AppContext,
|
|
) -> Task<Result<Vec<ArgumentCompletion>>> {
|
|
Task::ready(Err(anyhow!("this command does not require argument")))
|
|
}
|
|
|
|
fn requires_argument(&self) -> bool {
|
|
false
|
|
}
|
|
|
|
fn run(
|
|
self: Arc<Self>,
|
|
_argument: Option<&str>,
|
|
workspace: WeakView<Workspace>,
|
|
_delegate: Arc<dyn LspAdapterDelegate>,
|
|
cx: &mut WindowContext,
|
|
) -> Task<Result<SlashCommandOutput>> {
|
|
let output = workspace.update(cx, |workspace, cx| {
|
|
let Some(active_item) = workspace.active_item(cx) else {
|
|
return Task::ready(Err(anyhow!("no active tab")));
|
|
};
|
|
let Some(buffer) = active_item
|
|
.downcast::<Editor>()
|
|
.and_then(|editor| editor.read(cx).buffer().read(cx).as_singleton())
|
|
else {
|
|
return Task::ready(Err(anyhow!("active tab is not an editor")));
|
|
};
|
|
|
|
let snapshot = buffer.read(cx).snapshot();
|
|
let path = snapshot.resolve_file_path(cx, true);
|
|
let task = cx.background_executor().spawn({
|
|
let path = path.clone();
|
|
async move {
|
|
let mut output = String::new();
|
|
output.push_str(&codeblock_fence_for_path(path.as_deref(), None));
|
|
for chunk in snapshot.as_rope().chunks() {
|
|
output.push_str(chunk);
|
|
}
|
|
if !output.ends_with('\n') {
|
|
output.push('\n');
|
|
}
|
|
output.push_str("```\n");
|
|
let has_diagnostics =
|
|
write_single_file_diagnostics(&mut output, path.as_deref(), &snapshot);
|
|
if output.ends_with('\n') {
|
|
output.pop();
|
|
}
|
|
(output, has_diagnostics)
|
|
}
|
|
});
|
|
cx.foreground_executor().spawn(async move {
|
|
let (text, has_diagnostics) = task.await;
|
|
let range = 0..text.len();
|
|
Ok(SlashCommandOutput {
|
|
text,
|
|
sections: vec![build_entry_output_section(
|
|
range,
|
|
path.as_deref(),
|
|
false,
|
|
None,
|
|
)],
|
|
run_commands_in_text: has_diagnostics,
|
|
})
|
|
})
|
|
});
|
|
output.unwrap_or_else(|error| Task::ready(Err(error)))
|
|
}
|
|
}
|