feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
use strum::IntoEnumIterator;
|
2025-06-26 13:03:31 -07:00
|
|
|
use strum_macros::AsRefStr;
|
feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
use strum_macros::EnumIter;
|
|
|
|
|
use strum_macros::EnumString;
|
|
|
|
|
use strum_macros::IntoStaticStr;
|
|
|
|
|
|
|
|
|
|
/// Commands that can be invoked by starting a message with a leading slash.
|
2025-05-14 13:15:41 -07:00
|
|
|
#[derive(
|
|
|
|
|
Debug, Clone, Copy, PartialEq, Eq, Hash, EnumString, EnumIter, AsRefStr, IntoStaticStr,
|
|
|
|
|
)]
|
feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
#[strum(serialize_all = "kebab-case")]
|
|
|
|
|
pub enum SlashCommand {
|
2025-06-26 13:03:31 -07:00
|
|
|
// DO NOT ALPHA-SORT! Enum order is presentation order in the popup, so
|
|
|
|
|
// more frequently used commands should be listed first.
|
2025-08-19 10:55:07 -07:00
|
|
|
Model,
|
2025-08-19 22:34:37 -07:00
|
|
|
Approvals,
|
2025-09-18 14:14:16 -07:00
|
|
|
Review,
|
2025-06-06 16:29:37 -07:00
|
|
|
New,
|
2025-08-06 09:10:23 -07:00
|
|
|
Init,
|
2025-07-31 21:34:32 -07:00
|
|
|
Compact,
|
2025-09-23 16:59:52 +01:00
|
|
|
Undo,
|
2025-06-26 13:03:31 -07:00
|
|
|
Diff,
|
2025-08-11 14:15:41 -07:00
|
|
|
Mention,
|
2025-08-05 23:57:52 -07:00
|
|
|
Status,
|
2025-08-19 09:00:31 -07:00
|
|
|
Mcp,
|
2025-08-07 01:17:33 -07:00
|
|
|
Logout,
|
feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
Quit,
|
2025-07-31 00:43:21 -07:00
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
|
TestApproval,
|
feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl SlashCommand {
|
|
|
|
|
/// User-visible description shown in the popup.
|
|
|
|
|
pub fn description(self) -> &'static str {
|
|
|
|
|
match self {
|
2025-08-07 03:29:33 -07:00
|
|
|
SlashCommand::New => "start a new chat during a conversation",
|
|
|
|
|
SlashCommand::Init => "create an AGENTS.md file with instructions for Codex",
|
|
|
|
|
SlashCommand::Compact => "summarize conversation to prevent hitting the context limit",
|
2025-09-23 16:59:52 +01:00
|
|
|
SlashCommand::Review => "review my current changes and find issues",
|
|
|
|
|
SlashCommand::Undo => "restore the workspace to the last Codex snapshot",
|
2025-08-07 03:29:33 -07:00
|
|
|
SlashCommand::Quit => "exit Codex",
|
|
|
|
|
SlashCommand::Diff => "show git diff (including untracked files)",
|
2025-08-11 14:15:41 -07:00
|
|
|
SlashCommand::Mention => "mention a file",
|
2025-08-07 03:29:33 -07:00
|
|
|
SlashCommand::Status => "show current session configuration and token usage",
|
2025-08-20 00:26:14 -07:00
|
|
|
SlashCommand::Model => "choose what model and reasoning effort to use",
|
2025-08-19 22:34:37 -07:00
|
|
|
SlashCommand::Approvals => "choose what Codex can do without approval",
|
2025-08-19 09:00:31 -07:00
|
|
|
SlashCommand::Mcp => "list configured MCP tools",
|
2025-08-07 03:29:33 -07:00
|
|
|
SlashCommand::Logout => "log out of Codex",
|
2025-07-31 00:43:21 -07:00
|
|
|
#[cfg(debug_assertions)]
|
2025-08-07 03:29:33 -07:00
|
|
|
SlashCommand::TestApproval => "test approval request",
|
feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Command string without the leading '/'. Provided for compatibility with
|
|
|
|
|
/// existing code that expects a method named `command()`.
|
|
|
|
|
pub fn command(self) -> &'static str {
|
|
|
|
|
self.into()
|
|
|
|
|
}
|
2025-08-28 10:15:59 -07:00
|
|
|
|
|
|
|
|
/// Whether this command can be run while a task is in progress.
|
|
|
|
|
pub fn available_during_task(self) -> bool {
|
|
|
|
|
match self {
|
|
|
|
|
SlashCommand::New
|
|
|
|
|
| SlashCommand::Init
|
|
|
|
|
| SlashCommand::Compact
|
2025-09-23 16:59:52 +01:00
|
|
|
| SlashCommand::Undo
|
2025-08-28 10:15:59 -07:00
|
|
|
| SlashCommand::Model
|
|
|
|
|
| SlashCommand::Approvals
|
2025-09-18 14:14:16 -07:00
|
|
|
| SlashCommand::Review
|
2025-08-28 10:15:59 -07:00
|
|
|
| SlashCommand::Logout => false,
|
|
|
|
|
SlashCommand::Diff
|
|
|
|
|
| SlashCommand::Mention
|
|
|
|
|
| SlashCommand::Status
|
|
|
|
|
| SlashCommand::Mcp
|
2025-08-28 20:06:10 -07:00
|
|
|
| SlashCommand::Quit => true,
|
|
|
|
|
|
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
|
SlashCommand::TestApproval => true,
|
2025-08-28 10:15:59 -07:00
|
|
|
}
|
|
|
|
|
}
|
feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
}
|
|
|
|
|
|
2025-06-26 13:03:31 -07:00
|
|
|
/// Return all built-in commands in a Vec paired with their command string.
|
|
|
|
|
pub fn built_in_slash_commands() -> Vec<(&'static str, SlashCommand)> {
|
2025-09-23 16:59:52 +01:00
|
|
|
let show_beta_features = beta_features_enabled();
|
|
|
|
|
|
|
|
|
|
SlashCommand::iter()
|
|
|
|
|
.filter(|cmd| {
|
|
|
|
|
if *cmd == SlashCommand::Undo {
|
|
|
|
|
show_beta_features
|
|
|
|
|
} else {
|
|
|
|
|
true
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
.map(|c| (c.command(), c))
|
|
|
|
|
.collect()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn beta_features_enabled() -> bool {
|
|
|
|
|
std::env::var_os("BETA_FEATURE").is_some()
|
feat: add support for commands in the Rust TUI (#935)
Introduces support for slash commands like in the TypeScript CLI. We do
not support the full set of commands yet, but the core abstraction is
there now.
In particular, we have a `SlashCommand` enum and due to thoughtful use
of the [strum](https://crates.io/crates/strum) crate, it requires
minimal boilerplate to add a new command to the list.
The key new piece of UI is `CommandPopup`, though the keyboard events
are still handled by `ChatComposer`. The behavior is roughly as follows:
* if the first character in the composer is `/`, the command popup is
displayed (if you really want to send a message to Codex that starts
with a `/`, simply put a space before the `/`)
* while the popup is displayed, up/down can be used to change the
selection of the popup
* if there is a selection, hitting tab completes the command, but does
not send it
* if there is a selection, hitting enter sends the command
* if the prefix of the composer matches a command, the command will be
visible in the popup so the user can see the description (commands could
take arguments, so additional text may appear after the command name
itself)
https://github.com/user-attachments/assets/39c3e6ee-eeb7-4ef7-a911-466d8184975f
Incidentally, Codex wrote almost all the code for this PR!
2025-05-14 12:55:49 -07:00
|
|
|
}
|