mirror of
https://github.com/nushell/nushell
synced 2025-01-05 01:39:02 +00:00
0805f1fd90
# Description This PR overhauls the shell_integration system by allowing individual control over which ansi escape sequences are used. As we continue to broaden our support for more ansi escape sequences, we can't really have an all-or-nothing strategy. Some ansi escapes cause problems in certain operating systems or terminals. We should allow the user to choose which escapes they want. TODO: * Gather feedback * Should osc7, osc9_9 and osc633p be mutually exclusive? * Is the naming convention for these settings too nerdy osc2, osc7, etc? closes #11301 # User-Facing Changes shell_integration is no longer a boolean value. This is what is supported in the default_config.nu ```nushell shell_integration: { # osc2 abbreviates the path if in the home_dir, sets the tab/window title, shows the running command in the tab/window title osc2: true # osc7 is a way to communicate the path to the terminal, this is helpful for spawning new tabs in the same directory osc7: true # osc8 is also implemented as the deprecated setting ls.show_clickable_links, it shows clickable links in ls output if your terminal supports it osc8: true # osc9_9 is from ConEmu and is starting to get wider support. It's similar to osc7 in that it communicates the path to the terminal osc9_9: false # osc133 is several escapes invented by Final Term which include the supported ones below. # 133;A - Mark prompt start # 133;B - Mark prompt end # 133;C - Mark pre-execution # 133;D;exit - Mark execution finished with exit code # This is used to enable terminals to know where the prompt is, the command is, where the command finishes, and where the output of the command is osc133: true # osc633 is closely related to osc133 but only exists in visual studio code (vscode) and supports their shell integration features # 633;A - Mark prompt start # 633;B - Mark prompt end # 633;C - Mark pre-execution # 633;D;exit - Mark execution finished with exit code # 633;E - NOT IMPLEMENTED - Explicitly set the command line with an optional nonce # 633;P;Cwd=<path> - Mark the current working directory and communicate it to the terminal # and also helps with the run recent menu in vscode osc633: true # reset_application_mode is escape \x1b[?1l and was added to help ssh work better reset_application_mode: true } ``` # Tests + Formatting <!-- Don't forget to add tests that cover your changes. Make sure you've run and fixed any issues with these commands: - `cargo fmt --all -- --check` to check standard code formatting (`cargo fmt --all` applies these changes) - `cargo clippy --workspace -- -D warnings -D clippy::unwrap_used` to check that you're using the standard code style - `cargo test --workspace` to check that all tests pass (on Windows make sure to [enable developer mode](https://learn.microsoft.com/en-us/windows/apps/get-started/developer-mode-features-and-debugging)) - `cargo run -- -c "use std testing; testing run-tests --path crates/nu-std"` to run the tests for the standard library > **Note** > from `nushell` you can also use the `toolkit` as follows > ```bash > use toolkit.nu # or use an `env_change` hook to activate it automatically > toolkit check pr > ``` --> # After Submitting <!-- If your PR had any user-facing changes, update [the documentation](https://github.com/nushell/nushell.github.io) after the PR is merged, if necessary. This will help us keep the docs up to date. -->
210 lines
7.1 KiB
Rust
210 lines
7.1 KiB
Rust
use crate::prompt_update::{
|
|
POST_PROMPT_MARKER, PRE_PROMPT_MARKER, VSCODE_POST_PROMPT_MARKER, VSCODE_PRE_PROMPT_MARKER,
|
|
};
|
|
use nu_protocol::{
|
|
engine::{EngineState, Stack},
|
|
Value,
|
|
};
|
|
#[cfg(windows)]
|
|
use nu_utils::enable_vt_processing;
|
|
use reedline::{
|
|
DefaultPrompt, Prompt, PromptEditMode, PromptHistorySearch, PromptHistorySearchStatus,
|
|
PromptViMode,
|
|
};
|
|
use std::borrow::Cow;
|
|
|
|
/// Nushell prompt definition
|
|
#[derive(Clone)]
|
|
pub struct NushellPrompt {
|
|
shell_integration_osc133: bool,
|
|
shell_integration_osc633: bool,
|
|
left_prompt_string: Option<String>,
|
|
right_prompt_string: Option<String>,
|
|
default_prompt_indicator: Option<String>,
|
|
default_vi_insert_prompt_indicator: Option<String>,
|
|
default_vi_normal_prompt_indicator: Option<String>,
|
|
default_multiline_indicator: Option<String>,
|
|
render_right_prompt_on_last_line: bool,
|
|
engine_state: EngineState,
|
|
stack: Stack,
|
|
}
|
|
|
|
impl NushellPrompt {
|
|
pub fn new(
|
|
shell_integration_osc133: bool,
|
|
shell_integration_osc633: bool,
|
|
engine_state: EngineState,
|
|
stack: Stack,
|
|
) -> NushellPrompt {
|
|
NushellPrompt {
|
|
shell_integration_osc133,
|
|
shell_integration_osc633,
|
|
left_prompt_string: None,
|
|
right_prompt_string: None,
|
|
default_prompt_indicator: None,
|
|
default_vi_insert_prompt_indicator: None,
|
|
default_vi_normal_prompt_indicator: None,
|
|
default_multiline_indicator: None,
|
|
render_right_prompt_on_last_line: false,
|
|
engine_state,
|
|
stack,
|
|
}
|
|
}
|
|
|
|
pub fn update_prompt_left(&mut self, prompt_string: Option<String>) {
|
|
self.left_prompt_string = prompt_string;
|
|
}
|
|
|
|
pub fn update_prompt_right(
|
|
&mut self,
|
|
prompt_string: Option<String>,
|
|
render_right_prompt_on_last_line: bool,
|
|
) {
|
|
self.right_prompt_string = prompt_string;
|
|
self.render_right_prompt_on_last_line = render_right_prompt_on_last_line;
|
|
}
|
|
|
|
pub fn update_prompt_indicator(&mut self, prompt_indicator_string: Option<String>) {
|
|
self.default_prompt_indicator = prompt_indicator_string;
|
|
}
|
|
|
|
pub fn update_prompt_vi_insert(&mut self, prompt_vi_insert_string: Option<String>) {
|
|
self.default_vi_insert_prompt_indicator = prompt_vi_insert_string;
|
|
}
|
|
|
|
pub fn update_prompt_vi_normal(&mut self, prompt_vi_normal_string: Option<String>) {
|
|
self.default_vi_normal_prompt_indicator = prompt_vi_normal_string;
|
|
}
|
|
|
|
pub fn update_prompt_multiline(&mut self, prompt_multiline_indicator_string: Option<String>) {
|
|
self.default_multiline_indicator = prompt_multiline_indicator_string;
|
|
}
|
|
|
|
pub fn update_all_prompt_strings(
|
|
&mut self,
|
|
left_prompt_string: Option<String>,
|
|
right_prompt_string: Option<String>,
|
|
prompt_indicator_string: Option<String>,
|
|
prompt_multiline_indicator_string: Option<String>,
|
|
prompt_vi: (Option<String>, Option<String>),
|
|
render_right_prompt_on_last_line: bool,
|
|
) {
|
|
let (prompt_vi_insert_string, prompt_vi_normal_string) = prompt_vi;
|
|
|
|
self.left_prompt_string = left_prompt_string;
|
|
self.right_prompt_string = right_prompt_string;
|
|
self.default_prompt_indicator = prompt_indicator_string;
|
|
self.default_multiline_indicator = prompt_multiline_indicator_string;
|
|
|
|
self.default_vi_insert_prompt_indicator = prompt_vi_insert_string;
|
|
self.default_vi_normal_prompt_indicator = prompt_vi_normal_string;
|
|
|
|
self.render_right_prompt_on_last_line = render_right_prompt_on_last_line;
|
|
}
|
|
|
|
fn default_wrapped_custom_string(&self, str: String) -> String {
|
|
format!("({str})")
|
|
}
|
|
}
|
|
|
|
impl Prompt for NushellPrompt {
|
|
fn render_prompt_left(&self) -> Cow<str> {
|
|
#[cfg(windows)]
|
|
{
|
|
let _ = enable_vt_processing();
|
|
}
|
|
|
|
if let Some(prompt_string) = &self.left_prompt_string {
|
|
prompt_string.replace('\n', "\r\n").into()
|
|
} else {
|
|
let default = DefaultPrompt::default();
|
|
let prompt = default
|
|
.render_prompt_left()
|
|
.to_string()
|
|
.replace('\n', "\r\n");
|
|
|
|
if self.shell_integration_osc633 {
|
|
if self.stack.get_env_var(&self.engine_state, "TERM_PROGRAM")
|
|
== Some(Value::test_string("vscode"))
|
|
{
|
|
// We're in vscode and we have osc633 enabled
|
|
format!("{VSCODE_PRE_PROMPT_MARKER}{prompt}{VSCODE_POST_PROMPT_MARKER}").into()
|
|
} else {
|
|
// If we're in VSCode but we don't find the env var, just return the regular markers
|
|
format!("{PRE_PROMPT_MARKER}{prompt}{POST_PROMPT_MARKER}").into()
|
|
}
|
|
} else if self.shell_integration_osc133 {
|
|
format!("{PRE_PROMPT_MARKER}{prompt}{POST_PROMPT_MARKER}").into()
|
|
} else {
|
|
prompt.into()
|
|
}
|
|
}
|
|
}
|
|
|
|
fn render_prompt_right(&self) -> Cow<str> {
|
|
if let Some(prompt_string) = &self.right_prompt_string {
|
|
prompt_string.replace('\n', "\r\n").into()
|
|
} else {
|
|
let default = DefaultPrompt::default();
|
|
default
|
|
.render_prompt_right()
|
|
.to_string()
|
|
.replace('\n', "\r\n")
|
|
.into()
|
|
}
|
|
}
|
|
|
|
fn render_prompt_indicator(&self, edit_mode: PromptEditMode) -> Cow<str> {
|
|
match edit_mode {
|
|
PromptEditMode::Default => match &self.default_prompt_indicator {
|
|
Some(indicator) => indicator,
|
|
None => "> ",
|
|
}
|
|
.into(),
|
|
PromptEditMode::Emacs => match &self.default_prompt_indicator {
|
|
Some(indicator) => indicator,
|
|
None => "> ",
|
|
}
|
|
.into(),
|
|
PromptEditMode::Vi(vi_mode) => match vi_mode {
|
|
PromptViMode::Normal => match &self.default_vi_normal_prompt_indicator {
|
|
Some(indicator) => indicator,
|
|
None => "> ",
|
|
},
|
|
PromptViMode::Insert => match &self.default_vi_insert_prompt_indicator {
|
|
Some(indicator) => indicator,
|
|
None => ": ",
|
|
},
|
|
}
|
|
.into(),
|
|
PromptEditMode::Custom(str) => self.default_wrapped_custom_string(str).into(),
|
|
}
|
|
}
|
|
|
|
fn render_prompt_multiline_indicator(&self) -> Cow<str> {
|
|
match &self.default_multiline_indicator {
|
|
Some(indicator) => indicator,
|
|
None => "::: ",
|
|
}
|
|
.into()
|
|
}
|
|
|
|
fn render_prompt_history_search_indicator(
|
|
&self,
|
|
history_search: PromptHistorySearch,
|
|
) -> Cow<str> {
|
|
let prefix = match history_search.status {
|
|
PromptHistorySearchStatus::Passing => "",
|
|
PromptHistorySearchStatus::Failing => "failing ",
|
|
};
|
|
|
|
Cow::Owned(format!(
|
|
"({}reverse-search: {})",
|
|
prefix, history_search.term
|
|
))
|
|
}
|
|
|
|
fn right_prompt_on_last_line(&self) -> bool {
|
|
self.render_right_prompt_on_last_line
|
|
}
|
|
}
|