From 6208283f6d9cbeb6fda2915afa3a6158d030b01e Mon Sep 17 00:00:00 2001 From: Thomas Otto Date: Mon, 5 Jun 2023 22:21:07 +0200 Subject: [PATCH] Wrap --help output and use pager Set clap option `max_term_width`, so the help output is by default as wide as the terminal or this max value. Then manually wrap the `after_long_help()` text (only on demand) to the same width using `textwrap`. Also use matching ansi codes in this section. The help output is now paginated if output is to a terminal. All code paths flow back to main() so an invoked pager is properly waited for (by the OutputType drop impl). --- Cargo.lock | 7 + Cargo.toml | 1 + manual/src/full---help-output.md | 4 + src/ansi/mod.rs | 2 + src/cli.rs | 427 +++++++++++++++++++------------ src/main.rs | 9 + src/subcommands/show_colors.rs | 7 +- 7 files changed, 294 insertions(+), 163 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 78abb1376..bdf91e24f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -528,6 +528,7 @@ dependencies = [ "syntect", "sysinfo", "terminal-colorsaurus", + "textwrap", "unicode-segmentation", "unicode-width", "xdg", @@ -1286,6 +1287,12 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "textwrap" +version = "0.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23d434d3f8967a09480fb04132ebe0a3e088c173e6d0ee7897abbdf4eab0f8b9" + [[package]] name = "thiserror" version = "1.0.56" diff --git a/Cargo.toml b/Cargo.toml index 65bb72331..4cadaa26b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -55,6 +55,7 @@ unicode-width = "0.1.10" xdg = "2.4.1" clap_complete = "4.4.4" terminal-colorsaurus = "0.4.1" +textwrap = { version = "0.16.0", default-features = false, features = [] } [dependencies.git2] version = "0.18.2" diff --git a/manual/src/full---help-output.md b/manual/src/full---help-output.md index 16116a1dc..9741267b1 100644 --- a/manual/src/full---help-output.md +++ b/manual/src/full---help-output.md @@ -790,7 +790,11 @@ Similarly, the default value of --line-numbers-right-format is '{np:^4}│'. Thi Use '<' for left-align, '^' for center-align, and '>' for right-align. +SUPPORT +------- + If something isn't working correctly, or you have a feature request, please open an issue at https://github.com/dandavison/delta/issues. + For a short help summary, please use delta -h. ``` diff --git a/src/ansi/mod.rs b/src/ansi/mod.rs index 856c39022..5c3b11b88 100644 --- a/src/ansi/mod.rs +++ b/src/ansi/mod.rs @@ -12,8 +12,10 @@ use iterator::{AnsiElementIterator, Element}; pub const ANSI_CSI_CLEAR_TO_EOL: &str = "\x1b[0K"; pub const ANSI_CSI_CLEAR_TO_BOL: &str = "\x1b[1K"; +pub const ANSI_SGR_BOLD: &str = "\x1b[1m"; pub const ANSI_SGR_RESET: &str = "\x1b[0m"; pub const ANSI_SGR_REVERSE: &str = "\x1b[7m"; +pub const ANSI_SGR_UNDERLINE: &str = "\x1b[4m"; pub fn strip_ansi_codes(s: &str) -> String { strip_ansi_codes_from_strings_iterator(ansi_strings_iterator(s)) diff --git a/src/cli.rs b/src/cli.rs index 28d196280..e2b1d6a50 100644 --- a/src/cli.rs +++ b/src/cli.rs @@ -3,12 +3,14 @@ use std::ffi::OsString; use std::path::{Path, PathBuf}; use bat::assets::HighlightingAssets; -use clap::{ColorChoice, CommandFactory, FromArgMatches, Parser, ValueEnum, ValueHint}; +use clap::{ArgMatches, ColorChoice, CommandFactory, FromArgMatches, Parser, ValueEnum, ValueHint}; use clap_complete::Shell; +use console::Term; use lazy_static::lazy_static; use syntect::highlighting::Theme as SyntaxTheme; use syntect::parsing::SyntaxSet; +use crate::ansi::{ANSI_SGR_BOLD, ANSI_SGR_RESET, ANSI_SGR_UNDERLINE}; use crate::config::delta_unreachable; use crate::env::DeltaEnv; use crate::git_config::GitConfig; @@ -16,170 +18,15 @@ use crate::options; use crate::utils; use crate::utils::bat::output::PagingMode; +const TERM_FALLBACK_WIDTH: usize = 79; + #[derive(Parser)] #[command( name = "delta", about = "A viewer for git and diff output", version, color = ColorChoice::Always, - term_width(0), - after_long_help = "\ -GIT CONFIG ----------- - -By default, delta takes settings from a section named \"delta\" in git config files, if one is present. The git config file to use for delta options will usually be ~/.gitconfig, but delta follows the rules given in https://git-scm.com/docs/git-config#FILES. Most delta options can be given in a git config file, using the usual option names but without the initial '--'. An example is - -[delta] - line-numbers = true - zero-style = dim syntax - -FEATURES --------- -A feature is a named collection of delta options in git config. An example is: - -[delta \"my-delta-feature\"] - syntax-theme = Dracula - plus-style = bold syntax \"#002800\" - -To activate those options, you would use: - -delta --features my-delta-feature - -A feature name may not contain whitespace. You can activate multiple features: - -[delta] - features = my-highlight-styles-colors-feature my-line-number-styles-feature - -If more than one feature sets the same option, the last one wins. - -If an option is present in the [delta] section, then features are not considered at all. - -If you want an option to be fully overridable by a feature and also have a non default value when no features are used, then you need to define a \"default\" feature and include it in the main delta configuration. - -For instance: - -[delta] -feature = default-feature - -[delta \"default-feature\"] -width = 123 - -At this point, you can override features set in the command line or in the environment variables and the \"last one wins\" rules will apply as expected. - -STYLES ------- - -All options that have a name like --*-style work the same way. It is very similar to how colors/styles are specified in a gitconfig file: https://git-scm.com/docs/git-config#Documentation/git-config.txt-color - -Here is an example: - ---minus-style 'red bold ul \"#ffeeee\"' - -That means: For removed lines, set the foreground (text) color to 'red', make it bold and underlined, and set the background color to '#ffeeee'. - -See the COLORS section below for how to specify a color. In addition to real colors, there are 4 special color names: 'auto', 'normal', 'raw', and 'syntax'. - -Here is an example of using special color names together with a single attribute: - ---minus-style 'syntax bold auto' - -That means: For removed lines, syntax-highlight the text, and make it bold, and do whatever delta normally does for the background. - -The available attributes are: 'blink', 'bold', 'dim', 'hidden', 'italic', 'reverse', 'strike', and 'ul' (or 'underline'). - -The attribute 'omit' is supported by commit-style, file-style, and hunk-header-style, meaning to remove the element entirely from the output. - -A complete description of the style string syntax follows: - -- If the input that delta is receiving already has colors, and you want delta to output those colors unchanged, then use the special style string 'raw'. Otherwise, delta will strip any colors from its input. - -- A style string consists of 0, 1, or 2 colors, together with an arbitrary number of style attributes, all separated by spaces. - -- The first color is the foreground (text) color. The second color is the background color. Attributes can go in any position. - -- This means that in order to specify a background color you must also specify a foreground (text) color. - -- If you want delta to choose one of the colors automatically, then use the special color 'auto'. This can be used for both foreground and background. - -- If you want the foreground/background color to be your terminal's foreground/background color, then use the special color 'normal'. - -- If you want the foreground text to be syntax-highlighted according to its language, then use the special foreground color 'syntax'. This can only be used for the foreground (text). - -- The minimal style specification is the empty string ''. This means: do not apply any colors or styling to the element in question. - -COLORS ------- - -There are four ways to specify a color (this section applies to foreground and background colors within a style string): - -1. CSS color name - - Any of the 140 color names used in CSS: https://www.w3schools.com/colors/colors_groups.asp - -2. RGB hex code - - An example of using an RGB hex code is: - --file-style=\"#0e7c0e\" - -3. ANSI color name - - There are 8 ANSI color names: - black, red, green, yellow, blue, magenta, cyan, white. - - In addition, all of them have a bright form: - brightblack, brightred, brightgreen, brightyellow, brightblue, brightmagenta, brightcyan, brightwhite. - - An example of using an ANSI color name is: - --file-style=\"green\" - - Unlike RGB hex codes, ANSI color names are just names: you can choose the exact color that each name corresponds to in the settings of your terminal application (the application you use to enter commands at a shell prompt). This means that if you use ANSI color names, and you change the color theme used by your terminal, then delta's colors will respond automatically, without needing to change the delta command line. - - \"purple\" is accepted as a synonym for \"magenta\". Color names and codes are case-insensitive. - -4. ANSI color number - - An example of using an ANSI color number is: - --file-style=28 - - There are 256 ANSI color numbers: 0-255. The first 16 are the same as the colors described in the \"ANSI color name\" section above. See https://en.wikipedia.org/wiki/ANSI_escape_code#8-bit. Specifying colors like this is useful if your terminal only supports 256 colors (i.e. doesn\'t support 24-bit color). - - -LINE NUMBERS ------------- - -To display line numbers, use --line-numbers. - -Line numbers are displayed in two columns. Here's what it looks like by default: - - 1 ⋮ 1 │ unchanged line - 2 ⋮ │ removed line - ⋮ 2 │ added line - -In that output, the line numbers for the old (minus) version of the file appear in the left column, and the line numbers for the new (plus) version of the file appear in the right column. In an unchanged (zero) line, both columns contain a line number. - -The following options allow the line number display to be customized: - ---line-numbers-left-format: Change the contents of the left column ---line-numbers-right-format: Change the contents of the right column ---line-numbers-left-style: Change the style applied to the left column ---line-numbers-right-style: Change the style applied to the right column ---line-numbers-minus-style: Change the style applied to line numbers in minus lines ---line-numbers-zero-style: Change the style applied to line numbers in unchanged lines ---line-numbers-plus-style: Change the style applied to line numbers in plus lines - -Options --line-numbers-left-format and --line-numbers-right-format allow you to change the contents of the line number columns. Their values are arbitrary format strings, which are allowed to contain the placeholders {nm} for the line number associated with the old version of the file and {np} for the line number associated with the new version of the file. The placeholders support a subset of the string formatting syntax documented here: https://doc.rust-lang.org/std/fmt/#formatting-parameters. Specifically, you can use the alignment and width syntax. - -For example, the default value of --line-numbers-left-format is '{nm:^4}⋮'. This means that the left column should display the minus line number (nm), center-aligned, padded with spaces to a width of 4 characters, followed by a unicode dividing-line character (⋮). - -Similarly, the default value of --line-numbers-right-format is '{np:^4}│'. This means that the right column should display the plus line number (np), center-aligned, padded with spaces to a width of 4 characters, followed by a unicode dividing-line character (│). - -Use '<' for left-align, '^' for center-align, and '>' for right-align. - - -If something isn't working correctly, or you have a feature request, please open an issue at https://github.com/dandavison/delta/issues. - -For a short help summary, please use delta -h. -" + max_term_width(TERM_FALLBACK_WIDTH), )] pub struct Opt { #[arg(long = "blame-code-style", value_name = "STYLE")] @@ -1129,6 +976,196 @@ pub struct Opt { pub env: DeltaEnv, } +#[allow(non_snake_case)] +fn get_after_long_help(term: &Term) -> String { + let is_term = term.is_term(); + let term_width = if term.is_term() { + utils::workarounds::windows_msys2_width_fix(term.size(), term) + } else { + TERM_FALLBACK_WIDTH + }; + + let wrap = textwrap::Options::new(term_width) + .word_separator(textwrap::WordSeparator::AsciiSpace) + .word_splitter(textwrap::WordSplitter::NoHyphenation) + .wrap_algorithm(textwrap::WrapAlgorithm::FirstFit); + + // header and underline + let (H_, _H, u_, _u) = if is_term { + ( + format!("{ANSI_SGR_BOLD}{ANSI_SGR_UNDERLINE}"), + ANSI_SGR_RESET, + ANSI_SGR_UNDERLINE, + ANSI_SGR_RESET, + ) + } else { + ("".to_string(), "", "", "") + }; + + let after_long_help = format!( + r##" +{H_}Git config{_H} + +By default, delta takes settings from a section named "delta" in git config files, if one is present. The git config file to use for delta options will usually be ~/.gitconfig, but delta follows the rules given in https://git-scm.com/docs/git-config#FILES. Most delta options can be given in a git config file, using the usual option names but without the initial '--'. An example is + +[delta] + line-numbers = true + zero-style = dim syntax + + +{H_}Features{_H} + +A feature is a named collection of delta options in git config. An example is: + +[delta "my-delta-feature"] + syntax-theme = Dracula + plus-style = bold syntax "#002800" + +To activate those options, you would use: + +delta --features my-delta-feature + +A feature name may not contain whitespace. You can activate multiple features: + +[delta] + features = my-highlight-styles-colors-feature my-line-number-styles-feature + +If more than one feature sets the same option, the last one wins. + +If an option is present in the [delta] section, then features are not considered at all. + +If you want an option to be fully overridable by a feature and also have a non default value when no features are used, then you need to define a "default" feature and include it in the main delta configuration. + +For instance: + +[delta] + feature = default-feature + +[delta "default-feature"] + width = 123 + +At this point, you can override features set in the command line or in the environment variables and the "last one wins" rules will apply as expected. + + +{H_}Styles{_H} + +All options that have a name like --*-style work the same way. It is very similar to how colors/styles are specified in a gitconfig file: https://git-scm.com/docs/git-config#Documentation/git-config.txt-color + +Here is an example: + +--minus-style 'red bold ul "#ffeeee"' + +That means: For removed lines, set the foreground (text) color to 'red', make it bold and underlined, and set the background color to '#ffeeee'. + +See the {u_}Colors{_u} section below for how to specify a color. In addition to real colors, there are 4 special color names: 'auto', 'normal', 'raw', and 'syntax'. + +Here is an example of using special color names together with a single attribute: + +--minus-style 'syntax bold auto' + +That means: For removed lines, syntax-highlight the text, and make it bold, and do whatever delta normally does for the background. + +The available attributes are: 'blink', 'bold', 'dim', 'hidden', 'italic', 'reverse', 'strike', and 'ul' (or 'underline'). + +The attribute 'omit' is supported by commit-style, file-style, and hunk-header-style, meaning to remove the element entirely from the output. + +A complete description of the style string syntax follows: + +- If the input that delta is receiving already has colors, and you want delta to output those colors unchanged, then use the special style string 'raw'. Otherwise, delta will strip any colors from its input. + +- A style string consists of 0, 1, or 2 colors, together with an arbitrary number of style attributes, all separated by spaces. + +- The first color is the foreground (text) color. The second color is the background color. Attributes can go in any position. + +- This means that in order to specify a background color you must also specify a foreground (text) color. + +- If you want delta to choose one of the colors automatically, then use the special color 'auto'. This can be used for both foreground and background. + +- If you want the foreground/background color to be your terminal's foreground/background color, then use the special color 'normal'. + +- If you want the foreground text to be syntax-highlighted according to its language, then use the special foreground color 'syntax'. This can only be used for the foreground (text). + +- The minimal style specification is the empty string ''. This means: do not apply any colors or styling to the element in question. + + +{H_}Colors{_H} + +There are four ways to specify a color (this section applies to foreground and background colors within a style string): + +1. CSS color name + + Any of the 140 color names used in CSS: https://www.w3schools.com/colors/colors_groups.asp + +2. RGB hex code + + An example of using an RGB hex code is: + --file-style="#0e7c0e" + +3. ANSI color name + + There are 8 ANSI color names: + black, red, green, yellow, blue, magenta, cyan, white. + + In addition, all of them have a bright form: + brightblack, brightred, brightgreen, brightyellow, brightblue, brightmagenta, brightcyan, brightwhite. + + An example of using an ANSI color name is: + --file-style="green" + + Unlike RGB hex codes, ANSI color names are just names: you can choose the exact color that each name corresponds to in the settings of your terminal application (the application you use to enter commands at a shell prompt). This means that if you use ANSI color names, and you change the color theme used by your terminal, then delta's colors will respond automatically, without needing to change the delta command line. + + "purple" is accepted as a synonym for "magenta". Color names and codes are case-insensitive. + +4. ANSI color number + + An example of using an ANSI color number is: + --file-style=28 + + There are 256 ANSI color numbers: 0-255. The first 16 are the same as the colors described in the "ANSI color name" section above. See https://en.wikipedia.org/wiki/ANSI_escape_code#8-bit. Specifying colors like this is useful if your terminal only supports 256 colors (i.e. doesn't support 24-bit color). + + +{H_}Line Numbers{_H} + +To display line numbers, use --line-numbers. + +Line numbers are displayed in two columns. Here's what it looks like by default: + + 1 ⋮ 1 │ unchanged line + 2 ⋮ │ removed line + ⋮ 2 │ added line + +In that output, the line numbers for the old (minus) version of the file appear in the left column, and the line numbers for the new (plus) version of the file appear in the right column. In an unchanged (zero) line, both columns contain a line number. + +The following options allow the line number display to be customized: + +--line-numbers-left-format: Change the contents of the left column +--line-numbers-right-format: Change the contents of the right column +--line-numbers-left-style: Change the style applied to the left column +--line-numbers-right-style: Change the style applied to the right column +--line-numbers-minus-style: Change the style applied to line numbers in minus lines +--line-numbers-zero-style: Change the style applied to line numbers in unchanged lines +--line-numbers-plus-style: Change the style applied to line numbers in plus lines + +Options --line-numbers-left-format and --line-numbers-right-format allow you to change the contents of the line number columns. Their values are arbitrary format strings, which are allowed to contain the placeholders {{nm}} for the line number associated with the old version of the file and {{np}} for the line number associated with the new version of the file. The placeholders support a subset of the string formatting syntax documented here: https://doc.rust-lang.org/std/fmt/#formatting-parameters. Specifically, you can use the alignment and width syntax. + +For example, the default value of --line-numbers-left-format is '{{nm:^4}}⋮'. This means that the left column should display the minus line number (nm), center-aligned, padded with spaces to a width of 4 characters, followed by a unicode dividing-line character (⋮). + +Similarly, the default value of --line-numbers-right-format is '{{np:^4}}│'. This means that the right column should display the plus line number (np), center-aligned, padded with spaces to a width of 4 characters, followed by a unicode dividing-line character (│). + +Use '<' for left-align, '^' for center-align, and '>' for right-align. + + +{H_}Support{_H} + +If something isn't working correctly, or you have a feature request, please open an issue at https://github.com/dandavison/delta/issues. + +For a short help summary, please use delta -h. +"## + ); + + textwrap::wrap(&after_long_help, wrap).join("\n") +} + #[derive(Default, Clone, Debug)] pub struct ComputedValues { pub available_terminal_width: usize, @@ -1168,9 +1205,70 @@ pub enum DetectDarkLight { Never, } +// Which call path to take +#[derive(Debug)] +pub enum Call { + Delta(T), + Help(String), + Version(String), +} + +// Custom conversion because a) generic TryFrom is not possible and +// b) the Delta(T) variant can't be converted. +impl Call { + fn try_convert(self) -> Option> { + use Call::*; + match self { + Delta(_) => None, + Help(help) => Some(Help(help)), + Version(ver) => Some(Version(ver)), + } + } +} + impl Opt { - pub fn from_args_and_git_config(env: &DeltaEnv, assets: HighlightingAssets) -> Self { - let matches = Self::command().get_matches(); + fn handle_help_and_version(args: &[OsString]) -> Call { + match Self::command().try_get_matches_from(args) { + Err(e) if e.kind() == clap::error::ErrorKind::DisplayVersion => { + let version = Self::command().render_version(); + Call::Version(version) + } + Err(e) if e.kind() == clap::error::ErrorKind::DisplayHelp => { + let term = Term::stdout(); + // find out if short or long version of --help was used: + let help_text = if args.iter().any(|arg| arg == "-h") { + Self::command().render_help() + } else { + // generate long help on demand: + Self::command() + .after_long_help(get_after_long_help(&term)) + .render_long_help() + }; + + if term.is_term() { + Call::Help(format!("{}", help_text.ansi())) + } else { + Call::Help(format!("{}", help_text)) + } + } + Err(e) => { + e.exit(); + } + Ok(matches) => Call::Delta(matches), + } + } + + pub fn from_args_and_git_config(env: &DeltaEnv, assets: HighlightingAssets) -> Call { + let args = std::env::args_os().collect::>(); + + let matches = match Self::handle_help_and_version(&args) { + Call::Delta(t) => t, + msg => { + return msg + .try_convert() + .unwrap_or_else(|| panic!("Call<_> conversion failed")) + } + }; let mut final_config = if *matches.get_one::("no_gitconfig").unwrap_or(&false) { None @@ -1185,7 +1283,12 @@ impl Opt { } } - Self::from_clap_and_git_config(env, matches, final_config, assets) + Call::Delta(Self::from_clap_and_git_config( + env, + matches, + final_config, + assets, + )) } pub fn from_iter_and_git_config( diff --git a/src/main.rs b/src/main.rs index 78c1a25bf..a0df151ae 100644 --- a/src/main.rs +++ b/src/main.rs @@ -29,6 +29,7 @@ use std::process; use bytelines::ByteLinesReader; +use crate::cli::Call; use crate::delta::delta; use crate::utils::bat::assets::list_languages; use crate::utils::bat::output::OutputType; @@ -78,6 +79,14 @@ fn run_app() -> std::io::Result { let env = env::DeltaEnv::init(); let opt = cli::Opt::from_args_and_git_config(&env, assets); + let opt = match opt { + Call::Help(msg) | Call::Version(msg) => { + OutputType::oneshot_write(msg)?; + return Ok(0); + } + Call::Delta(opt) => opt, + }; + let subcommand_result = if let Some(shell) = opt.generate_completion { Some(subcommands::generate_completion::generate_completion_file( shell, diff --git a/src/subcommands/show_colors.rs b/src/subcommands/show_colors.rs index a7f5927ce..b67c22997 100644 --- a/src/subcommands/show_colors.rs +++ b/src/subcommands/show_colors.rs @@ -15,7 +15,12 @@ pub fn show_colors() -> std::io::Result<()> { let assets = utils::bat::assets::load_highlighting_assets(); let env = DeltaEnv::default(); - let opt = cli::Opt::from_args_and_git_config(&env, assets); + + let opt = match cli::Opt::from_args_and_git_config(&env, assets) { + cli::Call::Delta(opt) => opt, + _ => panic!("non-Delta Call variant should not occur here"), + }; + let config = config::Config::from(opt); let pagercfg = (&config).into();