default_command.rs

 1use super::{SlashCommand, SlashCommandOutput};
 2use crate::prompt_library::PromptStore;
 3use anyhow::{anyhow, Result};
 4use assistant_slash_command::{ArgumentCompletion, SlashCommandOutputSection};
 5use gpui::{AppContext, Task, WeakView};
 6use language::LspAdapterDelegate;
 7use std::{
 8    fmt::Write,
 9    sync::{atomic::AtomicBool, Arc},
10};
11use ui::prelude::*;
12use workspace::Workspace;
13
14pub(crate) struct DefaultSlashCommand;
15
16impl SlashCommand for DefaultSlashCommand {
17    fn name(&self) -> String {
18        "default".into()
19    }
20
21    fn description(&self) -> String {
22        "insert default prompt".into()
23    }
24
25    fn menu_text(&self) -> String {
26        "Insert Default Prompt".into()
27    }
28
29    fn requires_argument(&self) -> bool {
30        false
31    }
32
33    fn complete_argument(
34        self: Arc<Self>,
35        _query: String,
36        _cancellation_flag: Arc<AtomicBool>,
37        _workspace: Option<WeakView<Workspace>>,
38        _cx: &mut AppContext,
39    ) -> Task<Result<Vec<ArgumentCompletion>>> {
40        Task::ready(Err(anyhow!("this command does not require argument")))
41    }
42
43    fn run(
44        self: Arc<Self>,
45        _argument: Option<&str>,
46        _workspace: WeakView<Workspace>,
47        _delegate: Arc<dyn LspAdapterDelegate>,
48        cx: &mut WindowContext,
49    ) -> Task<Result<SlashCommandOutput>> {
50        let store = PromptStore::global(cx);
51        cx.background_executor().spawn(async move {
52            let store = store.await?;
53            let prompts = store.default_prompt_metadata();
54
55            let mut text = String::new();
56            text.push('\n');
57            for prompt in prompts {
58                if let Some(title) = prompt.title {
59                    writeln!(text, "/prompt {}", title).unwrap();
60                }
61            }
62            text.pop();
63
64            if text.is_empty() {
65                text.push('\n');
66            }
67
68            Ok(SlashCommandOutput {
69                sections: vec![SlashCommandOutputSection {
70                    range: 0..text.len(),
71                    icon: IconName::Library,
72                    label: "Default".into(),
73                }],
74                text,
75                run_commands_in_text: true,
76            })
77        })
78    }
79}