diff --git a/.gitignore b/.gitignore
index a3ea8cff..fbfe6ac6 100644
--- a/.gitignore
+++ b/.gitignore
@@ -2,6 +2,7 @@
**/*.rs.bk
# Generated files
+/assets/completions/_bat.ps1
/assets/completions/bat.bash
/assets/completions/bat.fish
/assets/completions/bat.zsh
diff --git a/CHANGELOG.md b/CHANGELOG.md
index d2ec2da9..fdbbf867 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -9,6 +9,8 @@
- Syntax highlighting for JavaScript files that start with `#!/usr/bin/env bun` #2913 (@sharunkumar)
- `bat --strip-ansi={never,always,auto}` to remove ANSI escape sequences from bat's input, see #2999 (@eth-p)
- Add or remove individual style components without replacing all styles #2929 (@eth-p)
+- Automatically choose theme based on the terminal's color scheme, see #2896 (@bash)
+- Add option `--binary=as-text` for printing binary content, see issue #2974 and PR #2976 (@einfachIrgendwer0815)
## Bugfixes
@@ -18,6 +20,7 @@
- Fix handling of inputs with combined ANSI color and attribute sequences, see #2185 and #2856 (@eth-p)
- Fix panel width when line 10000 wraps, see #2854 (@eth-p)
- Fix compile issue of `time` dependency caused by standard library regression #3045 (@cyqsimon)
+- Fix override behavior of --plain and --paging, see issue #2731 and PR #3108 (@einfachIrgendwer0815)
## Other
@@ -44,6 +47,7 @@
- Use bat's ANSI iterator during tab expansion, see #2998 (@eth-p)
- Support 'statically linked binary' for aarch64 in 'Release' page, see #2992 (@tzq0301)
- Update options in shell completions and the man page of `bat`, see #2995 (@akinomyoga)
+- Update nix dev-dependency to v0.29.0, see #3112 (@decathorpe)
## Syntaxes
@@ -60,6 +64,10 @@
- Associate Wireguard config `/etc/wireguard/*.conf`, see #2874 (@cyqsimon)
- Add support for [CFML](https://www.adobe.com/products/coldfusion-family.html), see #3031 (@brenton-at-pieces)
- Map `*.mkd` files to `Markdown` syntax, see issue #3060 and PR #3061 (@einfachIrgendwer0815)
+- Add syntax mapping for CITATION.cff, see #3103 (@Ugzuzg)
+- Add syntax mapping for kubernetes config files #3049 (@cyqsimon)
+- Adds support for pipe delimiter for CSV #3115 (@pratik-m)
+- Add syntax mapping for `/etc/pacman.conf` #2961 (@cyqsimon)
## Themes
@@ -71,6 +79,9 @@
- [BREAKING] `SyntaxMapping::mappings` is replaced by `SyntaxMapping::{builtin,custom,all}_mappings`
- Make `Controller::run_with_error_handler`'s error handler `FnMut`, see #2831 (@rhysd)
- Improve compile time by 20%, see #2815 (@dtolnay)
+- Add `theme::theme` for choosing an appropriate theme based on the
+ terminal's color scheme, see #2896 (@bash)
+ - [BREAKING] Remove `HighlightingAssets::default_theme`. Use `theme::default_theme` instead.
# v0.24.0
@@ -109,6 +120,7 @@
- Update `Julia` syntax, see #2553 (@dependabot)
- add `NSIS` support, see #2577 (@idleberg)
- Update `ssh-config`, see #2697 (@mrmeszaros)
+- Add syntax mapping `*.debdiff` => `diff`, see #2947 (@jacg)
## `bat` as a library
diff --git a/Cargo.lock b/Cargo.lock
index 42db6b5f..a6d47a19 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -149,6 +149,7 @@ dependencies = [
"shell-words",
"syntect",
"tempfile",
+ "terminal-colorsaurus",
"thiserror",
"toml",
"unicode-width",
@@ -243,6 +244,12 @@ version = "1.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
+[[package]]
+name = "cfg_aliases"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724"
+
[[package]]
name = "clap"
version = "4.4.12"
@@ -273,12 +280,11 @@ checksum = "702fc72eb24e5a1e48ce58027a675bc24edd52096d5397d4aea7c6dd9eca0bd1"
[[package]]
name = "clircle"
-version = "0.5.0"
+version = "0.6.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ec0b92245ea62a7a751db4b0e4a583f8978e508077ef6de24fcc0d0dc5311a8d"
+checksum = "e136d50bd652710f1d86259a8977263d46bef0ab782a8bfc3887e44338517015"
dependencies = [
"cfg-if",
- "libc",
"serde",
"serde_derive",
"winapi",
@@ -416,9 +422,9 @@ checksum = "a357d28ed41a50f9c765dbfe56cbc04a64e53e5fc58ba79fbc34c10ef3df831f"
[[package]]
name = "encoding_rs"
-version = "0.8.34"
+version = "0.8.35"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "b45de904aa0b010bce2ab45264d0631681847fa7b6f2eaa7dab7619943bc4f59"
+checksum = "75030f3c4f45dafd7586dd6780965a8c7e8e285a5ecb86713e63a79c5b2766f3"
dependencies = [
"cfg-if",
]
@@ -596,9 +602,9 @@ dependencies = [
[[package]]
name = "grep-cli"
-version = "0.1.10"
+version = "0.1.11"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "ea40788c059ab8b622c4d074732750bfb3bd2912e2dd58eabc11798a4d5ad725"
+checksum = "47f1288f0e06f279f84926fa4c17e3fcd2a22b357927a82f2777f7be26e4cec0"
dependencies = [
"bstr",
"globset",
@@ -620,6 +626,12 @@ version = "0.14.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7dfda62a12f55daeae5015f81b0baea145391cb4520f86c248fc615d72640d12"
+[[package]]
+name = "hermit-abi"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024"
+
[[package]]
name = "home"
version = "0.5.9"
@@ -688,9 +700,9 @@ checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646"
[[package]]
name = "libc"
-version = "0.2.149"
+version = "0.2.161"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "a08173bc88b7955d1b3145aa561539096c421ac8debde8cbc3612ec635fee29b"
+checksum = "8e9489c2807c139ffd9c1794f4af0ebe86a828db53ecdc7fea2111d0fed085d1"
[[package]]
name = "libgit2-sys"
@@ -746,9 +758,9 @@ checksum = "b5e6163cb8c49088c2c36f57875e58ccd8c87c7427f7fbd50ea6710b2f3f2e8f"
[[package]]
name = "memchr"
-version = "2.6.4"
+version = "2.7.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "f665ee40bc4a3c5590afb1e9677db74a508659dfd71e126420da8274909a0167"
+checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3"
[[package]]
name = "miniz_oxide"
@@ -760,13 +772,26 @@ dependencies = [
]
[[package]]
-name = "nix"
-version = "0.26.4"
+name = "mio"
+version = "1.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "598beaf3cc6fdd9a5dfb1630c2800c7acd31df7aaf0f565796fba2b53ca1af1b"
+checksum = "4929e1f84c5e54c3ec6141cd5d8b5a5c055f031f80cf78f2072920173cb4d880"
dependencies = [
- "bitflags 1.3.2",
+ "hermit-abi",
+ "libc",
+ "wasi",
+ "windows-sys 0.52.0",
+]
+
+[[package]]
+name = "nix"
+version = "0.29.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "71e2746dc3a24dd78b3cfcb7be93368c6de9963d30f43a6a73998a9cf4b17b46"
+dependencies = [
+ "bitflags 2.4.0",
"cfg-if",
+ "cfg_aliases",
"libc",
]
@@ -1140,9 +1165,9 @@ dependencies = [
[[package]]
name = "serde_spanned"
-version = "0.6.5"
+version = "0.6.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "eb3622f419d1296904700073ea6cc23ad690adbd66f13ea683df73298736f0c1"
+checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1"
dependencies = [
"serde",
]
@@ -1302,6 +1327,30 @@ dependencies = [
"winapi-util",
]
+[[package]]
+name = "terminal-colorsaurus"
+version = "0.4.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5f99bb1dc5cde9eada5a8f466641240f9d5b9f55291d675df4160b097fbfa42e"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "memchr",
+ "mio",
+ "terminal-trx",
+]
+
+[[package]]
+name = "terminal-trx"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6d4c86910e10c782a02d3b7606de43cf7ebd80e1fafdca8e49a0db2b0d4611f0"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "windows-sys 0.52.0",
+]
+
[[package]]
name = "terminal_size"
version = "0.3.0"
@@ -1386,9 +1435,9 @@ checksum = "cda74da7e1a664f795bb1f8a87ec406fb89a02522cf6e50620d016add6dbbf5c"
[[package]]
name = "toml"
-version = "0.8.9"
+version = "0.8.19"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "c6a4b9e8023eb94392d3dca65d717c53abc5dad49c07cb65bb8fcd87115fa325"
+checksum = "a1ed1f98e3fdc28d6d910e6737ae6ab1a93bf1985935a1193e68f93eeb68d24e"
dependencies = [
"indexmap",
"serde",
@@ -1399,18 +1448,18 @@ dependencies = [
[[package]]
name = "toml_datetime"
-version = "0.6.5"
+version = "0.6.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "3550f4e9685620ac18a50ed434eb3aec30db8ba93b0287467bca5826ea25baf1"
+checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41"
dependencies = [
"serde",
]
[[package]]
name = "toml_edit"
-version = "0.21.1"
+version = "0.22.22"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "6a8534fd7f78b5405e860340ad6575217ce99f38d4d5c8f2442cb5ecb50090e1"
+checksum = "4ae48d6208a266e853d946088ed816055e556cc6028c5e8e2b84d9fa5dd7c7f5"
dependencies = [
"indexmap",
"serde",
@@ -1740,9 +1789,9 @@ checksum = "dff9641d1cd4be8d1a070daf9e3773c5f67e78b4d9d42263020c057706765c04"
[[package]]
name = "winnow"
-version = "0.5.18"
+version = "0.6.20"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "176b6138793677221d420fd2f0aeeced263f197688b36484660da767bca2fa32"
+checksum = "36c1fec1a2bb5866f07c25f68c26e565c4c200aebb96d7e55710c19d3e8ac49b"
dependencies = [
"memchr",
]
diff --git a/Cargo.toml b/Cargo.toml
index d5b61566..cd59aef7 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -58,16 +58,17 @@ serde_derive = "1.0"
serde_yaml = "0.9.28"
semver = "1.0"
path_abs = { version = "0.5", default-features = false }
-clircle = "0.5"
+clircle = "0.6"
bugreport = { version = "0.5.0", optional = true }
etcetera = { version = "0.8.0", optional = true }
-grep-cli = { version = "0.1.10", optional = true }
+grep-cli = { version = "0.1.11", optional = true }
regex = { version = "1.10.2", optional = true }
walkdir = { version = "2.5", optional = true }
bytesize = { version = "1.3.0" }
-encoding_rs = "0.8.34"
+encoding_rs = "0.8.35"
os_str_bytes = { version = "~7.0", optional = true }
run_script = { version = "^0.10.1", optional = true}
+terminal-colorsaurus = "0.4"
[dependencies.git2]
version = "0.19"
@@ -98,7 +99,7 @@ tempfile = "3.8.1"
serde = { version = "1.0", features = ["derive"] }
[target.'cfg(unix)'.dev-dependencies]
-nix = { version = "0.26.4", default-features = false, features = ["term"] }
+nix = { version = "0.29", default-features = false, features = ["term"] }
[build-dependencies]
anyhow = "1.0.86"
@@ -109,7 +110,7 @@ regex = "1.10.2"
serde = "1.0"
serde_derive = "1.0"
serde_with = { version = "3.8.1", default-features = false, features = ["macros"] }
-toml = { version = "0.8.9", features = ["preserve_order"] }
+toml = { version = "0.8.19", features = ["preserve_order"] }
walkdir = "2.5"
[build-dependencies.clap]
diff --git a/README.md b/README.md
index 016fe834..c09d0e36 100644
--- a/README.md
+++ b/README.md
@@ -35,11 +35,11 @@ A special *thank you* goes to our biggest sponsors
- Warp is a modern, Rust-based terminal with AI built in
so you and your team can build great software, faster.
+ Warp, the intelligent terminal
- Feel more productive on the command line with parameterized commands,
+ Run commands like a power user with AI and your dev team’s
- autosuggestions, and an IDE-like text editor.
+ knowledge in one fast, intuitive terminal. For MacOS or Linux.
### Syntax highlighting
@@ -482,8 +482,10 @@ the following command (you need [`fzf`](https://github.com/junegunn/fzf) for thi
bat --list-themes | fzf --preview="bat --theme={} --color=always /path/to/file"
```
-`bat` looks good on a dark background by default. However, if your terminal uses a
-light background, some themes like `GitHub` or `OneHalfLight` will work better for you.
+`bat` automatically picks a fitting theme depending on your terminal's background color.
+You can use the `--theme-light` / `--theme-light` options or the `BAT_THEME_DARK` / `BAT_THEME_LIGHT` environment variables
+to customize the themes used. This is especially useful if you frequently switch between dark and light mode.
+
You can also use a custom theme by following the
['Adding new themes' section below](https://github.com/sharkdp/bat#adding-new-themes).
@@ -693,10 +695,11 @@ on your operating system. To get the default path for your system, call
bat --config-file
```
-Alternatively, you can use the `BAT_CONFIG_PATH` environment variable to point `bat` to a
-non-default location of the configuration file:
+Alternatively, you can use `BAT_CONFIG_PATH` or `BAT_CONFIG_DIR` environment variables to point `bat`
+to a non-default location of the configuration file or the configuration directory respectively:
```bash
-export BAT_CONFIG_PATH="/path/to/bat.conf"
+export BAT_CONFIG_PATH="/path/to/bat/bat.conf"
+export BAT_CONFIG_DIR="/path/to/bat"
```
A default configuration file can be created with the `--generate-config-file` option.
diff --git a/assets/completions/_bat.ps1.in b/assets/completions/_bat.ps1.in
index c0c151e1..b6f62aae 100644
--- a/assets/completions/_bat.ps1.in
+++ b/assets/completions/_bat.ps1.in
@@ -37,6 +37,8 @@ Register-ArgumentCompleter -Native -CommandName '{{PROJECT_EXECUTABLE}}' -Script
[CompletionResult]::new('-m', 'm', [CompletionResultType]::ParameterName, 'Use the specified syntax for files matching the glob pattern (''*.cpp:C++'').')
[CompletionResult]::new('--map-syntax', 'map-syntax', [CompletionResultType]::ParameterName, 'Use the specified syntax for files matching the glob pattern (''*.cpp:C++'').')
[CompletionResult]::new('--theme', 'theme', [CompletionResultType]::ParameterName, 'Set the color theme for syntax highlighting.')
+ [CompletionResult]::new('--theme-dark', 'theme', [CompletionResultType]::ParameterName, 'Set the color theme for syntax highlighting for dark backgrounds.')
+ [CompletionResult]::new('--theme-light', 'theme', [CompletionResultType]::ParameterName, 'Set the color theme for syntax highlighting for light backgrounds.')
[CompletionResult]::new('--style', 'style', [CompletionResultType]::ParameterName, 'Comma-separated list of style elements to display (*default*, auto, full, plain, changes, header, header-filename, header-filesize, grid, rule, numbers, snip).')
[CompletionResult]::new('-r', 'r', [CompletionResultType]::ParameterName, 'Only print the lines from N to M.')
[CompletionResult]::new('--line-range', 'line-range', [CompletionResultType]::ParameterName, 'Only print the lines from N to M.')
diff --git a/assets/completions/bat.bash.in b/assets/completions/bat.bash.in
index f314bb25..90931f24 100644
--- a/assets/completions/bat.bash.in
+++ b/assets/completions/bat.bash.in
@@ -113,6 +113,13 @@ _bat() {
return 0
;;
--theme)
+ local IFS=$'\n'
+ COMPREPLY=($(compgen -W "auto${IFS}auto:always${IFS}auto:system${IFS}dark${IFS}light${IFS}$("$1" --list-themes)" -- "$cur"))
+ __bat_escape_completions
+ return 0
+ ;;
+ --theme-dark | \
+ --theme-light)
local IFS=$'\n'
COMPREPLY=($(compgen -W "$("$1" --list-themes)" -- "$cur"))
__bat_escape_completions
@@ -170,6 +177,8 @@ _bat() {
--map-syntax
--ignored-suffix
--theme
+ --theme-dark
+ --theme-light
--list-themes
--squeeze-blank
--squeeze-limit
diff --git a/assets/completions/bat.fish.in b/assets/completions/bat.fish.in
index 788f71b0..e2712706 100644
--- a/assets/completions/bat.fish.in
+++ b/assets/completions/bat.fish.in
@@ -129,6 +129,14 @@ set -l tabs_opts '
8\t
'
+set -l special_themes '
+ auto\tdefault,\ Choose\ a\ theme\ based\ on\ dark\ or\ light\ mode
+ auto:always\tChoose\ a\ theme\ based\ on\ dark\ or\ light\ mode
+ auto:system\tChoose\ a\ theme\ based\ on\ dark\ or\ light\ mode
+ dark\tUse\ the\ theme\ specified\ by\ --theme-dark
+ light\tUse\ the\ theme\ specified\ by\ --theme-light
+'
+
# Completions:
complete -c $bat -l acknowledgements -d "Print acknowledgements" -n __fish_is_first_arg
@@ -203,7 +211,11 @@ complete -c $bat -l tabs -x -a "$tabs_opts" -d "Set tab width" -n __bat_no_excl_
complete -c $bat -l terminal-width -x -d "Set terminal , +, or -" -n __bat_no_excl_args
-complete -c $bat -l theme -x -a "(command $bat --list-themes | command cat)" -d "Set the syntax highlighting theme" -n __bat_no_excl_args
+complete -c $bat -l theme -x -a "$special_themes(command $bat --list-themes | command cat)" -d "Set the syntax highlighting theme" -n __bat_no_excl_args
+
+complete -c $bat -l theme-dark -x -a "(command $bat --list-themes | command cat)" -d "Set the syntax highlighting theme for dark backgrounds" -n __bat_no_excl_args
+
+complete -c $bat -l theme-light -x -a "(command $bat --list-themes | command cat)" -d "Set the syntax highlighting theme for light backgrounds" -n __bat_no_excl_args
complete -c $bat -s V -l version -f -d "Show version information" -n __fish_is_first_arg
diff --git a/assets/completions/bat.zsh.in b/assets/completions/bat.zsh.in
index 7d03abb3..76b981b6 100644
--- a/assets/completions/bat.zsh.in
+++ b/assets/completions/bat.zsh.in
@@ -42,7 +42,9 @@ _{{PROJECT_EXECUTABLE}}_main() {
--decorations='[specify when to show the decorations]:when:(auto never always)'
--paging='[specify when to use the pager]:when:(auto never always)'
'(-m --map-syntax)'{-m+,--map-syntax=}'[map a glob pattern to an existing syntax name]: :->syntax-maps'
- '(--theme)'--theme='[set the color theme for syntax highlighting]:theme:->themes'
+ '(--theme)'--theme='[set the color theme for syntax highlighting]:theme:->theme_preferences'
+ '(--theme-dark)'--theme-dark='[set the color theme for syntax highlighting for dark backgrounds]:theme:->themes'
+ '(--theme-light)'--theme-light='[set the color theme for syntax highlighting for light backgrounds]:theme:->themes'
'(: --list-themes --list-languages -L)'--list-themes'[show all supported highlighting themes]'
--style='[comma-separated list of style elements to display]: : _values "style [default]"
default auto full plain changes header header-filename header-filesize grid rule numbers snip'
@@ -82,7 +84,13 @@ _{{PROJECT_EXECUTABLE}}_main() {
themes)
local -a themes expl
- themes=( ${(f)"$(_call_program themes {{PROJECT_EXECUTABLE}} --list-themes)"} )
+ themes=(${(f)"$(_call_program themes {{PROJECT_EXECUTABLE}} --list-themes)"} )
+
+ _wanted themes expl 'theme' compadd -a themes && ret=0
+ ;;
+ theme_preferences)
+ local -a themes expl
+ themes=(auto dark light auto:always auto:system ${(f)"$(_call_program themes {{PROJECT_EXECUTABLE}} --list-themes)"} )
_wanted themes expl 'theme' compadd -a themes && ret=0
;;
diff --git a/assets/manual/bat.1.in b/assets/manual/bat.1.in
index 2bc0a3a5..ccc70629 100644
--- a/assets/manual/bat.1.in
+++ b/assets/manual/bat.1.in
@@ -152,9 +152,38 @@ will use JSON syntax, and ignore '.dev'
.HP
\fB\-\-theme\fR
.IP
-Set the theme for syntax highlighting. Use '\-\-list\-themes' to see all available themes.
-To set a default theme, add the '\-\-theme="..."' option to the configuration file or
-export the BAT_THEME environment variable (e.g.: export BAT_THEME="...").
+Set the theme for syntax highlighting. Use \fB\-\-list\-themes\fP to see all available themes.
+To set a default theme, add the \fB\-\-theme="..."\fP option to the configuration file or
+export the \fBBAT_THEME\fP environment variable (e.g.: \fBexport BAT_THEME="..."\fP).
+
+Special values:
+.RS
+.IP "auto (\fIdefault\fR)"
+Picks a dark or light theme depending on the terminal's colors.
+Use \fB-\-theme\-light\fR and \fB-\-theme\-dark\fR to customize the selected theme.
+.IP "auto:always"
+Variation of \fBauto\fR where where the terminal's colors are detected even when the output is redirected.
+.IP "auto:system (macOS only)"
+Variation of \fBauto\fR where the color scheme is detected from the system-wide preference instead.
+.IP "dark"
+Use the dark theme specified by \fB-\-theme-dark\fR.
+.IP "light"
+Use the light theme specified by \fB-\-theme-light\fR.
+.RE
+.HP
+\fB\-\-theme\-dark\fR
+.IP
+Sets the theme name for syntax highlighting used when the terminal uses a dark background.
+To set a default theme, add the \fB\-\-theme-dark="..."\fP option to the configuration file or
+export the \fBBAT_THEME_DARK\fP environment variable (e.g. \fBexport BAT_THEME_DARK="..."\fP).
+This option only has an effect when \fB\-\-theme\fP option is set to \fBauto\fR or \fBdark\fR.
+.HP
+\fB\-\-theme\-light\fR
+.IP
+Sets the theme name for syntax highlighting used when the terminal uses a dark background.
+To set a default theme, add the \fB\-\-theme-dark="..."\fP option to the configuration file or
+export the \fBBAT_THEME_LIGHT\fP environment variable (e.g. \fBexport BAT_THEME_LIGHT="..."\fP).
+This option only has an effect when \fB\-\-theme\fP option is set to \fBauto\fR or \fBlight\fR.
.HP
\fB\-\-list\-themes\fR
.IP
@@ -307,7 +336,7 @@ To use the preprocessor, call:
\fB{{PROJECT_EXECUTABLE}} --lessopen\fR
-Alternatively, the preprocessor may be enabled by default by adding the '\-\-lessopen' option to the configuration file.
+Alternatively, the preprocessor may be enabled by default by adding the '\-\-lessopen' option to the configuration file.
To temporarily disable the preprocessor if it is enabled by default, call:
@@ -323,7 +352,7 @@ Enable the $LESSOPEN preprocessor.
.IP
Disable the $LESSOPEN preprocessor if enabled (overrides --lessopen)
.PP
-For more information, see the "INPUT PREPROCESSOR" section of less(1).
+For more information, see the "INPUT PREPROCESSOR" section of less(1).
.SH "MORE INFORMATION"
diff --git a/assets/syntaxes/02_Extra/CSV.sublime-syntax b/assets/syntaxes/02_Extra/CSV.sublime-syntax
index cca7cd2c..0ad17834 100644
--- a/assets/syntaxes/02_Extra/CSV.sublime-syntax
+++ b/assets/syntaxes/02_Extra/CSV.sublime-syntax
@@ -7,14 +7,14 @@ file_extensions:
- tsv
scope: text.csv
variables:
- field_separator: (?:[,;\t])
+ field_separator: (?:[,;|\t])
record_separator: (?:$\n?)
contexts:
prototype:
- match: (?={{record_separator}})
pop: true
fields:
- - match: ''
+ - match: ""
push:
- field_or_record_separator
- field4
@@ -26,15 +26,15 @@ contexts:
- field1
main:
- meta_include_prototype: false
- - match: '^'
+ - match: "^"
set: fields
field_or_record_separator:
- meta_include_prototype: false
- - match: '{{record_separator}}'
+ - match: "{{record_separator}}"
scope: punctuation.terminator.record.csv
pop: true
- - match: '{{field_separator}}'
+ - match: "{{field_separator}}"
scope: punctuation.separator.sequence.csv
pop: true
@@ -56,23 +56,22 @@ contexts:
pop: true
field1:
- - match: ''
+ - match: ""
set:
- meta_content_scope: meta.field-1.csv support.type
- include: field_contents
field2:
- - match: ''
+ - match: ""
set:
- meta_content_scope: meta.field-2.csv support.function
- include: field_contents
field3:
- - match: ''
+ - match: ""
set:
- meta_content_scope: meta.field-3.csv constant.numeric
- include: field_contents
field4:
- - match: ''
+ - match: ""
set:
- meta_content_scope: meta.field-4.csv keyword.operator
- include: field_contents
-
diff --git a/doc/long-help.txt b/doc/long-help.txt
index 2b03490f..85d595b9 100644
--- a/doc/long-help.txt
+++ b/doc/long-help.txt
@@ -20,6 +20,13 @@ Options:
* unicode (␇, ␊, ␀, ..)
* caret (^G, ^J, ^@, ..)
+ --binary
+ How to treat binary content. (default: no-printing)
+
+ Possible values:
+ * no-printing: do not print any binary content
+ * as-text: treat binary content as normal text
+
-p, --plain...
Only show plain style, no decorations. This is an alias for '--style=plain'. When '-p' is
used twice ('-pp'), it also disables automatic paging (alias for '--style=plain
@@ -112,6 +119,27 @@ Options:
Set the theme for syntax highlighting. Use '--list-themes' to see all available themes. To
set a default theme, add the '--theme="..."' option to the configuration file or export
the BAT_THEME environment variable (e.g.: export BAT_THEME="...").
+
+ Special values:
+
+ * auto: Picks a dark or light theme depending on the terminal's colors (default).
+ Use '--theme-light' and '--theme-dark' to customize the selected theme.
+ * auto:always: Detect the terminal's colors even when the output is redirected.
+ * auto:system: Detect the color scheme from the system-wide preference (macOS only).
+ * dark: Use the dark theme specified by '--theme-dark'.
+ * light: Use the light theme specified by '--theme-light'.
+
+ --theme-light
+ Sets the theme name for syntax highlighting used when the terminal uses a light
+ background. Use '--list-themes' to see all available themes. To set a default theme, add
+ the '--theme-light="..." option to the configuration file or export the BAT_THEME_LIGHT
+ environment variable (e.g. export BAT_THEME_LIGHT="...").
+
+ --theme-dark
+ Sets the theme name for syntax highlighting used when the terminal uses a dark background.
+ Use '--list-themes' to see all available themes. To set a default theme, add the
+ '--theme-dark="..." option to the configuration file or export the BAT_THEME_DARK
+ environment variable (e.g. export BAT_THEME_DARK="...").
--list-themes
Display a list of supported themes for syntax highlighting.
diff --git a/doc/short-help.txt b/doc/short-help.txt
index 305bbf3d..ba06ef30 100644
--- a/doc/short-help.txt
+++ b/doc/short-help.txt
@@ -11,6 +11,8 @@ Options:
Show non-printable characters (space, tab, newline, ..).
--nonprintable-notation
Set notation for non-printable characters.
+ --binary
+ How to treat binary content. (default: no-printing)
-p, --plain...
Show plain style (alias for '--style=plain').
-l, --language
@@ -41,6 +43,10 @@ Options:
Use the specified syntax for files matching the glob pattern ('*.cpp:C++').
--theme
Set the color theme for syntax highlighting.
+ --theme-light
+ Sets the color theme for syntax highlighting used for light backgrounds.
+ --theme-dark
+ Sets the color theme for syntax highlighting used for dark backgrounds.
--list-themes
Display all supported highlighting themes.
-s, --squeeze-blank
diff --git a/doc/sponsors/warp-logo.png b/doc/sponsors/warp-logo.png
index 4795a2b9..f99dd38c 100644
Binary files a/doc/sponsors/warp-logo.png and b/doc/sponsors/warp-logo.png differ
diff --git a/src/assets.rs b/src/assets.rs
index 9655553d..d32ccbd4 100644
--- a/src/assets.rs
+++ b/src/assets.rs
@@ -13,6 +13,7 @@ use crate::error::*;
use crate::input::{InputReader, OpenedInput};
use crate::syntax_mapping::ignored_suffixes::IgnoredSuffixes;
use crate::syntax_mapping::MappingTarget;
+use crate::theme::{default_theme, ColorScheme};
use crate::{bat_warning, SyntaxMapping};
use lazy_theme_set::LazyThemeSet;
@@ -69,57 +70,6 @@ impl HighlightingAssets {
}
}
- /// The default theme.
- ///
- /// ### Windows and Linux
- ///
- /// Windows and most Linux distributions has a dark terminal theme by
- /// default. On these platforms, this function always returns a theme that
- /// looks good on a dark background.
- ///
- /// ### macOS
- ///
- /// On macOS the default terminal background is light, but it is common that
- /// Dark Mode is active, which makes the terminal background dark. On this
- /// platform, the default theme depends on
- /// ```bash
- /// defaults read -globalDomain AppleInterfaceStyle
- /// ```
- /// To avoid the overhead of the check on macOS, simply specify a theme
- /// explicitly via `--theme`, `BAT_THEME`, or `~/.config/bat`.
- ///
- /// See and
- /// for more context.
- pub fn default_theme() -> &'static str {
- #[cfg(not(target_os = "macos"))]
- {
- Self::default_dark_theme()
- }
- #[cfg(target_os = "macos")]
- {
- if macos_dark_mode_active() {
- Self::default_dark_theme()
- } else {
- Self::default_light_theme()
- }
- }
- }
-
- /**
- * The default theme that looks good on a dark background.
- */
- fn default_dark_theme() -> &'static str {
- "Monokai Extended"
- }
-
- /**
- * The default theme that looks good on a light background.
- */
- #[cfg(target_os = "macos")]
- fn default_light_theme() -> &'static str {
- "Monokai Extended Light"
- }
-
pub fn from_cache(cache_path: &Path) -> Result {
Ok(HighlightingAssets::new(
SerializedSyntaxSet::FromFile(cache_path.join("syntaxes.bin")),
@@ -248,7 +198,10 @@ impl HighlightingAssets {
bat_warning!("Unknown theme '{}', using default.", theme)
}
self.get_theme_set()
- .get(self.fallback_theme.unwrap_or_else(Self::default_theme))
+ .get(
+ self.fallback_theme
+ .unwrap_or_else(|| default_theme(ColorScheme::Dark)),
+ )
.expect("something is very wrong if the default theme is missing")
}
}
@@ -399,26 +352,6 @@ fn asset_from_cache(
.map_err(|_| format!("Could not parse cached {description}").into())
}
-#[cfg(target_os = "macos")]
-fn macos_dark_mode_active() -> bool {
- const PREFERENCES_FILE: &str = "Library/Preferences/.GlobalPreferences.plist";
- const STYLE_KEY: &str = "AppleInterfaceStyle";
-
- let preferences_file = home::home_dir()
- .map(|home| home.join(PREFERENCES_FILE))
- .expect("Could not get home directory");
-
- match plist::Value::from_file(preferences_file).map(|file| file.into_dictionary()) {
- Ok(Some(preferences)) => match preferences.get(STYLE_KEY).and_then(|val| val.as_string()) {
- Some(value) => value == "Dark",
- // If the key does not exist, then light theme is currently in use.
- None => false,
- },
- // Unreachable, in theory. All macOS users have a home directory and preferences file setup.
- Ok(None) | Err(_) => true,
- }
-}
-
#[cfg(test)]
mod tests {
use super::*;
diff --git a/src/bin/bat/app.rs b/src/bin/bat/app.rs
index d6628668..8f69870f 100644
--- a/src/bin/bat/app.rs
+++ b/src/bin/bat/app.rs
@@ -9,6 +9,8 @@ use crate::{
config::{get_args_from_config_file, get_args_from_env_opts_var, get_args_from_env_vars},
};
use bat::style::StyleComponentList;
+use bat::theme::{theme, ThemeName, ThemeOptions, ThemePreference};
+use bat::BinaryBehavior;
use bat::StripAnsiMode;
use clap::ArgMatches;
@@ -16,7 +18,6 @@ use console::Term;
use crate::input::{new_file_input, new_stdin_input};
use bat::{
- assets::HighlightingAssets,
bat_warning,
config::{Config, VisibleLines},
error::*,
@@ -97,12 +98,30 @@ impl App {
pub fn config(&self, inputs: &[Input]) -> Result {
let style_components = self.style_components()?;
+ let extra_plain = self.matches.get_count("plain") > 1;
+ let plain_last_index = self
+ .matches
+ .indices_of("plain")
+ .and_then(Iterator::max)
+ .unwrap_or_default();
+ let paging_last_index = self
+ .matches
+ .indices_of("paging")
+ .and_then(Iterator::max)
+ .unwrap_or_default();
+
let paging_mode = match self.matches.get_one::("paging").map(|s| s.as_str()) {
- Some("always") => PagingMode::Always,
+ Some("always") => {
+ // Disable paging if the second -p (or -pp) is specified after --paging=always
+ if extra_plain && plain_last_index > paging_last_index {
+ PagingMode::Never
+ } else {
+ PagingMode::Always
+ }
+ }
Some("never") => PagingMode::Never,
Some("auto") | None => {
// If we have -pp as an option when in auto mode, the pager should be disabled.
- let extra_plain = self.matches.get_count("plain") > 1;
if extra_plain || self.matches.get_flag("no-paging") {
PagingMode::Never
} else if inputs.iter().any(Input::is_stdin) {
@@ -193,6 +212,11 @@ impl App {
Some("caret") => NonprintableNotation::Caret,
_ => unreachable!("other values for --nonprintable-notation are not allowed"),
},
+ binary: match self.matches.get_one::("binary").map(|s| s.as_str()) {
+ Some("as-text") => BinaryBehavior::AsText,
+ Some("no-printing") => BinaryBehavior::NoPrinting,
+ _ => unreachable!("other values for --binary are not allowed"),
+ },
wrapping_mode: if self.interactive_output || maybe_term_width.is_some() {
if !self.matches.get_flag("chop-long-lines") {
match self.matches.get_one::("wrap").map(|s| s.as_str()) {
@@ -254,18 +278,7 @@ impl App {
Some("auto") => StripAnsiMode::Auto,
_ => unreachable!("other values for --strip-ansi are not allowed"),
},
- theme: self
- .matches
- .get_one::("theme")
- .map(String::from)
- .map(|s| {
- if s == "default" {
- String::from(HighlightingAssets::default_theme())
- } else {
- s
- }
- })
- .unwrap_or_else(|| String::from(HighlightingAssets::default_theme())),
+ theme: theme(self.theme_options()).to_string(),
visible_lines: match self.matches.try_contains_id("diff").unwrap_or_default()
&& self.matches.get_flag("diff")
{
@@ -424,4 +437,25 @@ impl App {
Ok(styled_components)
}
+
+ fn theme_options(&self) -> ThemeOptions {
+ let theme = self
+ .matches
+ .get_one::("theme")
+ .map(|t| ThemePreference::from_str(t).unwrap())
+ .unwrap_or_default();
+ let theme_dark = self
+ .matches
+ .get_one::("theme-dark")
+ .map(|t| ThemeName::from_str(t).unwrap());
+ let theme_light = self
+ .matches
+ .get_one::("theme-light")
+ .map(|t| ThemeName::from_str(t).unwrap());
+ ThemeOptions {
+ theme,
+ theme_dark,
+ theme_light,
+ }
+ }
}
diff --git a/src/bin/bat/clap_app.rs b/src/bin/bat/clap_app.rs
index 33dde980..f5e3948e 100644
--- a/src/bin/bat/clap_app.rs
+++ b/src/bin/bat/clap_app.rs
@@ -77,11 +77,26 @@ pub fn build_app(interactive_output: bool) -> Command {
* caret (^G, ^J, ^@, ..)",
),
)
+ .arg(
+ Arg::new("binary")
+ .long("binary")
+ .action(ArgAction::Set)
+ .default_value("no-printing")
+ .value_parser(["no-printing", "as-text"])
+ .value_name("behavior")
+ .hide_default_value(true)
+ .help("How to treat binary content. (default: no-printing)")
+ .long_help(
+ "How to treat binary content. (default: no-printing)\n\n\
+ Possible values:\n \
+ * no-printing: do not print any binary content\n \
+ * as-text: treat binary content as normal text",
+ ),
+ )
.arg(
Arg::new("plain")
.overrides_with("plain")
.overrides_with("number")
- .overrides_with("paging")
.short('p')
.long("plain")
.action(ArgAction::Count)
@@ -306,7 +321,6 @@ pub fn build_app(interactive_output: bool) -> Command {
.long("paging")
.overrides_with("paging")
.overrides_with("no-paging")
- .overrides_with("plain")
.value_name("when")
.value_parser(["auto", "never", "always"])
.default_value("auto")
@@ -379,9 +393,40 @@ pub fn build_app(interactive_output: bool) -> Command {
see all available themes. To set a default theme, add the \
'--theme=\"...\"' option to the configuration file or export the \
BAT_THEME environment variable (e.g.: export \
- BAT_THEME=\"...\").",
+ BAT_THEME=\"...\").\n\n\
+ Special values:\n\n \
+ * auto: Picks a dark or light theme depending on the terminal's colors (default).\n \
+ Use '--theme-light' and '--theme-dark' to customize the selected theme.\n \
+ * auto:always: Detect the terminal's colors even when the output is redirected.\n \
+ * auto:system: Detect the color scheme from the system-wide preference (macOS only).\n \
+ * dark: Use the dark theme specified by '--theme-dark'.\n \
+ * light: Use the light theme specified by '--theme-light'.",
),
)
+ .arg(
+ Arg::new("theme-light")
+ .long("theme-light")
+ .overrides_with("theme-light")
+ .value_name("theme")
+ .help("Sets the color theme for syntax highlighting used for light backgrounds.")
+ .long_help(
+ "Sets the theme name for syntax highlighting used when the terminal uses a light background. \
+ Use '--list-themes' to see all available themes. To set a default theme, add the \
+ '--theme-light=\"...\" option to the configuration file or export the BAT_THEME_LIGHT \
+ environment variable (e.g. export BAT_THEME_LIGHT=\"...\")."),
+ )
+ .arg(
+ Arg::new("theme-dark")
+ .long("theme-dark")
+ .overrides_with("theme-dark")
+ .value_name("theme")
+ .help("Sets the color theme for syntax highlighting used for dark backgrounds.")
+ .long_help(
+ "Sets the theme name for syntax highlighting used when the terminal uses a dark background. \
+ Use '--list-themes' to see all available themes. To set a default theme, add the \
+ '--theme-dark=\"...\" option to the configuration file or export the BAT_THEME_DARK \
+ environment variable (e.g. export BAT_THEME_DARK=\"...\")."),
+ )
.arg(
Arg::new("list-themes")
.long("list-themes")
diff --git a/src/bin/bat/config.rs b/src/bin/bat/config.rs
index 6fa18f09..a0ee7ba3 100644
--- a/src/bin/bat/config.rs
+++ b/src/bin/bat/config.rs
@@ -140,7 +140,9 @@ fn get_args_from_str(content: &str) -> Result, shell_words::ParseE
pub fn get_args_from_env_vars() -> Vec {
[
("--tabs", "BAT_TABS"),
- ("--theme", "BAT_THEME"),
+ ("--theme", bat::theme::env::BAT_THEME),
+ ("--theme-dark", bat::theme::env::BAT_THEME_DARK),
+ ("--theme-light", bat::theme::env::BAT_THEME_LIGHT),
("--pager", "BAT_PAGER"),
("--paging", "BAT_PAGING"),
("--style", "BAT_STYLE"),
diff --git a/src/bin/bat/main.rs b/src/bin/bat/main.rs
index 3b74ec75..7b7bafe6 100644
--- a/src/bin/bat/main.rs
+++ b/src/bin/bat/main.rs
@@ -14,6 +14,7 @@ use std::io::{BufReader, Write};
use std::path::Path;
use std::process;
+use bat::theme::DetectColorScheme;
use nu_ansi_term::Color::Green;
use nu_ansi_term::Style;
@@ -30,12 +31,12 @@ use directories::PROJECT_DIRS;
use globset::GlobMatcher;
use bat::{
- assets::HighlightingAssets,
config::Config,
controller::Controller,
error::*,
input::Input,
style::{StyleComponent, StyleComponents},
+ theme::{color_scheme, default_theme, ColorScheme},
MappingTarget, PagingMode,
};
@@ -189,7 +190,12 @@ fn theme_preview_file<'a>() -> Input<'a> {
Input::from_reader(Box::new(BufReader::new(THEME_PREVIEW_DATA)))
}
-pub fn list_themes(cfg: &Config, config_dir: &Path, cache_dir: &Path) -> Result<()> {
+pub fn list_themes(
+ cfg: &Config,
+ config_dir: &Path,
+ cache_dir: &Path,
+ detect_color_scheme: DetectColorScheme,
+) -> Result<()> {
let assets = assets_from_cache_or_binary(cfg.use_custom_assets, cache_dir)?;
let mut config = cfg.clone();
let mut style = HashSet::new();
@@ -200,10 +206,14 @@ pub fn list_themes(cfg: &Config, config_dir: &Path, cache_dir: &Path) -> Result<
let stdout = io::stdout();
let mut stdout = stdout.lock();
- let default_theme = HighlightingAssets::default_theme();
+ let default_theme_name = default_theme(color_scheme(detect_color_scheme).unwrap_or_default());
for theme in assets.themes() {
- let default_theme_info = if !config.loop_through && default_theme == theme {
+ let default_theme_info = if !config.loop_through && default_theme_name == theme {
" (default)"
+ } else if default_theme(ColorScheme::Dark) == theme {
+ " (default dark)"
+ } else if default_theme(ColorScheme::Light) == theme {
+ " (default light)"
} else {
""
};
@@ -371,7 +381,7 @@ fn run() -> Result {
};
run_controller(inputs, &plain_config, cache_dir)
} else if app.matches.get_flag("list-themes") {
- list_themes(&config, config_dir, cache_dir)?;
+ list_themes(&config, config_dir, cache_dir, DetectColorScheme::default())?;
Ok(true)
} else if app.matches.get_flag("config-file") {
println!("{}", config_file().to_string_lossy());
diff --git a/src/config.rs b/src/config.rs
index eb7df8ee..eb44281c 100644
--- a/src/config.rs
+++ b/src/config.rs
@@ -1,5 +1,5 @@
use crate::line_range::{HighlightedLineRanges, LineRanges};
-use crate::nonprintable_notation::NonprintableNotation;
+use crate::nonprintable_notation::{BinaryBehavior, NonprintableNotation};
#[cfg(feature = "paging")]
use crate::paging::PagingMode;
use crate::style::StyleComponents;
@@ -44,6 +44,9 @@ pub struct Config<'a> {
/// The configured notation for non-printable characters
pub nonprintable_notation: NonprintableNotation,
+ /// How to treat binary content
+ pub binary: BinaryBehavior,
+
/// The character width of the terminal
pub term_width: usize,
diff --git a/src/lib.rs b/src/lib.rs
index 23c4a800..502427a7 100644
--- a/src/lib.rs
+++ b/src/lib.rs
@@ -49,10 +49,11 @@ pub(crate) mod printer;
pub mod style;
pub(crate) mod syntax_mapping;
mod terminal;
+pub mod theme;
mod vscreen;
pub(crate) mod wrapping;
-pub use nonprintable_notation::NonprintableNotation;
+pub use nonprintable_notation::{BinaryBehavior, NonprintableNotation};
pub use preprocessor::StripAnsiMode;
pub use pretty_printer::{Input, PrettyPrinter, Syntax};
pub use syntax_mapping::{MappingTarget, SyntaxMapping};
diff --git a/src/nonprintable_notation.rs b/src/nonprintable_notation.rs
index ff09aca6..9f8d7cb8 100644
--- a/src/nonprintable_notation.rs
+++ b/src/nonprintable_notation.rs
@@ -10,3 +10,15 @@ pub enum NonprintableNotation {
#[default]
Unicode,
}
+
+/// How to treat binary content
+#[derive(Debug, Default, Clone, Copy, PartialEq, Eq)]
+#[non_exhaustive]
+pub enum BinaryBehavior {
+ /// Do not print any binary content
+ #[default]
+ NoPrinting,
+
+ /// Treat binary content as normal text
+ AsText,
+}
diff --git a/src/pretty_printer.rs b/src/pretty_printer.rs
index eb123ea3..51c9af80 100644
--- a/src/pretty_printer.rs
+++ b/src/pretty_printer.rs
@@ -245,7 +245,9 @@ impl<'a> PrettyPrinter<'a> {
self
}
- /// Specify the highlighting theme
+ /// Specify the highlighting theme.
+ /// You can use [`crate::theme::theme`] to pick a theme based on user preferences
+ /// and the terminal's background color.
pub fn theme(&mut self, theme: impl AsRef) -> &mut Self {
self.config.theme = theme.as_ref().to_owned();
self
diff --git a/src/printer.rs b/src/printer.rs
index e9bea3fd..95017188 100644
--- a/src/printer.rs
+++ b/src/printer.rs
@@ -35,6 +35,7 @@ use crate::style::StyleComponent;
use crate::terminal::{as_terminal_escaped, to_ansi_color};
use crate::vscreen::{AnsiStyle, EscapeSequence, EscapeSequenceIterator};
use crate::wrapping::WrappingMode;
+use crate::BinaryBehavior;
use crate::StripAnsiMode;
const ANSI_UNDERLINE_ENABLE: EscapeSequence = EscapeSequence::CSI {
@@ -268,7 +269,8 @@ impl<'a> InteractivePrinter<'a> {
.content_type
.map_or(false, |c| c.is_binary() && !config.show_nonprintable);
- let needs_to_match_syntax = !is_printing_binary
+ let needs_to_match_syntax = (!is_printing_binary
+ || matches!(config.binary, BinaryBehavior::AsText))
&& (config.colored_output || config.strip_ansi == StripAnsiMode::Auto);
let (is_plain_text, highlighter_from_set) = if needs_to_match_syntax {
@@ -458,7 +460,10 @@ impl<'a> Printer for InteractivePrinter<'a> {
}
if !self.config.style_components.header() {
- if Some(ContentType::BINARY) == self.content_type && !self.config.show_nonprintable {
+ if Some(ContentType::BINARY) == self.content_type
+ && !self.config.show_nonprintable
+ && !matches!(self.config.binary, BinaryBehavior::AsText)
+ {
writeln!(
handle,
"{}: Binary content from {} will not be printed to the terminal \
@@ -539,7 +544,10 @@ impl<'a> Printer for InteractivePrinter<'a> {
})?;
if self.config.style_components.grid() {
- if self.content_type.map_or(false, |c| c.is_text()) || self.config.show_nonprintable {
+ if self.content_type.map_or(false, |c| c.is_text())
+ || self.config.show_nonprintable
+ || matches!(self.config.binary, BinaryBehavior::AsText)
+ {
self.print_horizontal_line(handle, '┼')?;
} else {
self.print_horizontal_line(handle, '┴')?;
@@ -551,7 +559,9 @@ impl<'a> Printer for InteractivePrinter<'a> {
fn print_footer(&mut self, handle: &mut OutputHandle, _input: &OpenedInput) -> Result<()> {
if self.config.style_components.grid()
- && (self.content_type.map_or(false, |c| c.is_text()) || self.config.show_nonprintable)
+ && (self.content_type.map_or(false, |c| c.is_text())
+ || self.config.show_nonprintable
+ || matches!(self.config.binary, BinaryBehavior::AsText))
{
self.print_horizontal_line(handle, '┴')
} else {
@@ -599,7 +609,9 @@ impl<'a> Printer for InteractivePrinter<'a> {
.into()
} else {
let mut line = match self.content_type {
- Some(ContentType::BINARY) | None => {
+ Some(ContentType::BINARY) | None
+ if !matches!(self.config.binary, BinaryBehavior::AsText) =>
+ {
return Ok(());
}
Some(ContentType::UTF_16LE) => UTF_16LE.decode_with_bom_removal(line_buffer).0,
diff --git a/src/syntax_mapping/builtins/common/50-citation.toml b/src/syntax_mapping/builtins/common/50-citation.toml
new file mode 100644
index 00000000..aa06b5b9
--- /dev/null
+++ b/src/syntax_mapping/builtins/common/50-citation.toml
@@ -0,0 +1,2 @@
+[mappings]
+"YAML" = ["CITATION.cff"]
diff --git a/src/syntax_mapping/builtins/common/50-diff.toml b/src/syntax_mapping/builtins/common/50-diff.toml
new file mode 100644
index 00000000..2998d9c5
--- /dev/null
+++ b/src/syntax_mapping/builtins/common/50-diff.toml
@@ -0,0 +1,3 @@
+# .debdiff is the extension used for diffs in Debian packaging
+[mappings]
+"Diff" = ["*.debdiff"]
diff --git a/src/syntax_mapping/builtins/linux/50-kubernetes.toml b/src/syntax_mapping/builtins/linux/50-kubernetes.toml
new file mode 100644
index 00000000..6a81a35a
--- /dev/null
+++ b/src/syntax_mapping/builtins/linux/50-kubernetes.toml
@@ -0,0 +1,2 @@
+[mappings]
+"YAML" = ["/etc/kubernetes/*.conf"]
diff --git a/src/syntax_mapping/builtins/linux/50-pacman.toml b/src/syntax_mapping/builtins/linux/50-pacman.toml
index 655118c5..2f4ee71f 100644
--- a/src/syntax_mapping/builtins/linux/50-pacman.toml
+++ b/src/syntax_mapping/builtins/linux/50-pacman.toml
@@ -1,3 +1,8 @@
[mappings]
-# pacman hooks
-"INI" = ["/usr/share/libalpm/hooks/*.hook", "/etc/pacman.d/hooks/*.hook"]
+"INI" = [
+ # config
+ "/etc/pacman.conf",
+ # hooks
+ "/usr/share/libalpm/hooks/*.hook",
+ "/etc/pacman.d/hooks/*.hook",
+]
diff --git a/src/theme.rs b/src/theme.rs
new file mode 100644
index 00000000..9fbef238
--- /dev/null
+++ b/src/theme.rs
@@ -0,0 +1,571 @@
+//! Utilities for choosing an appropriate theme for syntax highlighting.
+
+use std::convert::Infallible;
+use std::fmt;
+use std::io::IsTerminal as _;
+use std::str::FromStr;
+
+/// Environment variable names.
+pub mod env {
+ /// See [`crate::theme::ThemeOptions::theme`].
+ pub const BAT_THEME: &str = "BAT_THEME";
+ /// See [`crate::theme::ThemeOptions::theme_dark`].
+ pub const BAT_THEME_DARK: &str = "BAT_THEME";
+ /// See [`crate::theme::ThemeOptions::theme_light`].
+ pub const BAT_THEME_LIGHT: &str = "BAT_THEME";
+}
+
+/// Chooses an appropriate theme or falls back to a default theme
+/// based on the user-provided options and the color scheme of the terminal.
+///
+/// Intentionally returns a [`ThemeResult`] instead of a simple string so
+/// that downstream consumers such as `delta` can easily apply their own
+/// default theme and can use the detected color scheme elsewhere.
+pub fn theme(options: ThemeOptions) -> ThemeResult {
+ theme_impl(options, &TerminalColorSchemeDetector)
+}
+
+/// The default theme, suitable for the given color scheme.
+/// Use [`theme`] if you want to automatically detect the color scheme from the terminal.
+pub const fn default_theme(color_scheme: ColorScheme) -> &'static str {
+ match color_scheme {
+ ColorScheme::Dark => "Monokai Extended",
+ ColorScheme::Light => "Monokai Extended Light",
+ }
+}
+
+/// Detects the color scheme from the terminal.
+pub fn color_scheme(when: DetectColorScheme) -> Option {
+ color_scheme_impl(when, &TerminalColorSchemeDetector)
+}
+
+/// Options for configuring the theme used for syntax highlighting.
+/// Used together with [`theme`].
+#[derive(Debug, Clone, Default, PartialEq, Eq)]
+pub struct ThemeOptions {
+ /// Configures how the theme is chosen. If set to a [`ThemePreference::Fixed`] value,
+ /// then the given theme is used regardless of the terminal's background color.
+ /// This corresponds with the `BAT_THEME` environment variable and the `--theme` option.
+ pub theme: ThemePreference,
+ /// The theme to use in case the terminal uses a dark background with light text.
+ /// This corresponds with the `BAT_THEME_DARK` environment variable and the `--theme-dark` option.
+ pub theme_dark: Option,
+ /// The theme to use in case the terminal uses a light background with dark text.
+ /// This corresponds with the `BAT_THEME_LIGHT` environment variable and the `--theme-light` option.
+ pub theme_light: Option,
+}
+
+/// What theme should `bat` use?
+///
+/// The easiest way to construct this is from a string:
+/// ```
+/// # use bat::theme::{ThemePreference, DetectColorScheme};
+/// let preference = ThemePreference::new("auto:system");
+/// assert_eq!(ThemePreference::Auto(DetectColorScheme::System), preference);
+/// ```
+#[derive(Debug, Clone, PartialEq, Eq, Hash)]
+pub enum ThemePreference {
+ /// Choose between [`ThemeOptions::theme_dark`] and [`ThemeOptions::theme_light`]
+ /// based on the terminal's color scheme.
+ Auto(DetectColorScheme),
+ /// Always use the same theme regardless of the terminal's color scheme.
+ Fixed(ThemeName),
+ /// Use a dark theme.
+ Dark,
+ /// Use a light theme.
+ Light,
+}
+
+impl Default for ThemePreference {
+ fn default() -> Self {
+ ThemePreference::Auto(Default::default())
+ }
+}
+
+impl ThemePreference {
+ /// Creates a theme preference from a string.
+ pub fn new(s: impl Into) -> Self {
+ use ThemePreference::*;
+ let s = s.into();
+ match s.as_str() {
+ "auto" => Auto(Default::default()),
+ "auto:always" => Auto(DetectColorScheme::Always),
+ "auto:system" => Auto(DetectColorScheme::System),
+ "dark" => Dark,
+ "light" => Light,
+ _ => Fixed(ThemeName::new(s)),
+ }
+ }
+}
+
+impl FromStr for ThemePreference {
+ type Err = Infallible;
+
+ fn from_str(s: &str) -> Result {
+ Ok(ThemePreference::new(s))
+ }
+}
+
+impl fmt::Display for ThemePreference {
+ fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
+ use ThemePreference::*;
+ match self {
+ Auto(DetectColorScheme::Auto) => f.write_str("auto"),
+ Auto(DetectColorScheme::Always) => f.write_str("auto:always"),
+ Auto(DetectColorScheme::System) => f.write_str("auto:system"),
+ Fixed(theme) => theme.fmt(f),
+ Dark => f.write_str("dark"),
+ Light => f.write_str("light"),
+ }
+ }
+}
+
+/// The name of a theme or the default theme.
+///
+/// ```
+/// # use bat::theme::ThemeName;
+/// assert_eq!(ThemeName::Default, ThemeName::new("default"));
+/// assert_eq!(ThemeName::Named("example".to_string()), ThemeName::new("example"));
+/// ```
+#[derive(Debug, Clone, PartialEq, Eq, Hash)]
+pub enum ThemeName {
+ Named(String),
+ Default,
+}
+
+impl ThemeName {
+ /// Creates a theme name from a string.
+ pub fn new(s: impl Into) -> Self {
+ let s = s.into();
+ if s == "default" {
+ ThemeName::Default
+ } else {
+ ThemeName::Named(s)
+ }
+ }
+}
+
+impl FromStr for ThemeName {
+ type Err = Infallible;
+
+ fn from_str(s: &str) -> Result {
+ Ok(ThemeName::new(s))
+ }
+}
+
+impl fmt::Display for ThemeName {
+ fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
+ match self {
+ ThemeName::Named(t) => f.write_str(t),
+ ThemeName::Default => f.write_str("default"),
+ }
+ }
+}
+
+#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, Hash)]
+pub enum DetectColorScheme {
+ /// Only query the terminal for its colors when appropriate (i.e. when the the output is not redirected).
+ #[default]
+ Auto,
+ /// Always query the terminal for its colors.
+ Always,
+ /// Detect the system-wide dark/light preference (macOS only).
+ System,
+}
+
+/// The color scheme used to pick a fitting theme. Defaults to [`ColorScheme::Dark`].
+#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, Hash)]
+pub enum ColorScheme {
+ #[default]
+ Dark,
+ Light,
+}
+
+/// The resolved theme and the color scheme as determined from
+/// the terminal, OS or fallback.
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub struct ThemeResult {
+ /// The theme selected according to the [`ThemeOptions`].
+ pub theme: ThemeName,
+ /// Either the user's chosen color scheme, the terminal's color scheme, the OS's
+ /// color scheme or `None` if the color scheme was not detected because the user chose a fixed theme.
+ pub color_scheme: Option,
+}
+
+impl fmt::Display for ThemeResult {
+ fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
+ match &self.theme {
+ ThemeName::Named(name) => f.write_str(name),
+ ThemeName::Default => f.write_str(default_theme(self.color_scheme.unwrap_or_default())),
+ }
+ }
+}
+
+fn theme_impl(options: ThemeOptions, detector: &dyn ColorSchemeDetector) -> ThemeResult {
+ // Implementation note: This function is mostly pure (i.e. it has no side effects) for the sake of testing.
+ // All the side effects (e.g. querying the terminal for its colors) are performed in the detector.
+ match options.theme {
+ ThemePreference::Fixed(theme) => ThemeResult {
+ theme,
+ color_scheme: None,
+ },
+ ThemePreference::Dark => choose_theme_opt(Some(ColorScheme::Dark), options),
+ ThemePreference::Light => choose_theme_opt(Some(ColorScheme::Light), options),
+ ThemePreference::Auto(when) => choose_theme_opt(color_scheme_impl(when, detector), options),
+ }
+}
+
+fn choose_theme_opt(color_scheme: Option, options: ThemeOptions) -> ThemeResult {
+ ThemeResult {
+ color_scheme,
+ theme: color_scheme
+ .and_then(|c| choose_theme(options, c))
+ .unwrap_or(ThemeName::Default),
+ }
+}
+
+fn choose_theme(options: ThemeOptions, color_scheme: ColorScheme) -> Option {
+ match color_scheme {
+ ColorScheme::Dark => options.theme_dark,
+ ColorScheme::Light => options.theme_light,
+ }
+}
+
+fn color_scheme_impl(
+ when: DetectColorScheme,
+ detector: &dyn ColorSchemeDetector,
+) -> Option {
+ let should_detect = match when {
+ DetectColorScheme::Auto => detector.should_detect(),
+ DetectColorScheme::Always => true,
+ DetectColorScheme::System => return color_scheme_from_system(),
+ };
+ should_detect.then(|| detector.detect()).flatten()
+}
+
+trait ColorSchemeDetector {
+ fn should_detect(&self) -> bool;
+
+ fn detect(&self) -> Option;
+}
+
+struct TerminalColorSchemeDetector;
+
+impl ColorSchemeDetector for TerminalColorSchemeDetector {
+ fn should_detect(&self) -> bool {
+ // Querying the terminal for its colors via OSC 10 / OSC 11 requires "exclusive" access
+ // since we read/write from the terminal and enable/disable raw mode.
+ // This causes race conditions with pagers such as less when they are attached to the
+ // same terminal as us.
+ //
+ // This is usually only an issue when the output is manually piped to a pager.
+ // For example: `bat Cargo.toml | less`.
+ // Otherwise, if we start the pager ourselves, then there's no race condition
+ // since the pager is started *after* the color is detected.
+ std::io::stdout().is_terminal()
+ }
+
+ fn detect(&self) -> Option {
+ use terminal_colorsaurus::{color_scheme, ColorScheme as ColorsaurusScheme, QueryOptions};
+ match color_scheme(QueryOptions::default()).ok()? {
+ ColorsaurusScheme::Dark => Some(ColorScheme::Dark),
+ ColorsaurusScheme::Light => Some(ColorScheme::Light),
+ }
+ }
+}
+
+#[cfg(not(target_os = "macos"))]
+fn color_scheme_from_system() -> Option {
+ crate::bat_warning!(
+ "Theme 'auto:system' is only supported on macOS, \
+ using default."
+ );
+ None
+}
+
+#[cfg(target_os = "macos")]
+fn color_scheme_from_system() -> Option {
+ const PREFERENCES_FILE: &str = "Library/Preferences/.GlobalPreferences.plist";
+ const STYLE_KEY: &str = "AppleInterfaceStyle";
+
+ let preferences_file = home::home_dir()
+ .map(|home| home.join(PREFERENCES_FILE))
+ .expect("Could not get home directory");
+
+ match plist::Value::from_file(preferences_file).map(|file| file.into_dictionary()) {
+ Ok(Some(preferences)) => match preferences.get(STYLE_KEY).and_then(|val| val.as_string()) {
+ Some("Dark") => Some(ColorScheme::Dark),
+ // If the key does not exist, then light theme is currently in use.
+ Some(_) | None => Some(ColorScheme::Light),
+ },
+ // Unreachable, in theory. All macOS users have a home directory and preferences file setup.
+ Ok(None) | Err(_) => None,
+ }
+}
+
+#[cfg(test)]
+impl ColorSchemeDetector for Option {
+ fn should_detect(&self) -> bool {
+ true
+ }
+
+ fn detect(&self) -> Option {
+ *self
+ }
+}
+
+#[cfg(test)]
+mod tests {
+ use super::ColorScheme::*;
+ use super::*;
+ use std::cell::Cell;
+ use std::iter;
+
+ mod color_scheme_detection {
+ use super::*;
+
+ #[test]
+ fn not_called_for_dark_or_light() {
+ for theme in [ThemePreference::Dark, ThemePreference::Light] {
+ let detector = DetectorStub::should_detect(Some(Dark));
+ let options = ThemeOptions {
+ theme,
+ ..Default::default()
+ };
+ _ = theme_impl(options, &detector);
+ assert!(!detector.was_called.get());
+ }
+ }
+
+ #[test]
+ fn called_for_always() {
+ let detectors = [
+ DetectorStub::should_detect(Some(Dark)),
+ DetectorStub::should_not_detect(),
+ ];
+ for detector in detectors {
+ let options = ThemeOptions {
+ theme: ThemePreference::Auto(DetectColorScheme::Always),
+ ..Default::default()
+ };
+ _ = theme_impl(options, &detector);
+ assert!(detector.was_called.get());
+ }
+ }
+
+ #[test]
+ fn called_for_auto_if_should_detect() {
+ let detector = DetectorStub::should_detect(Some(Dark));
+ _ = theme_impl(ThemeOptions::default(), &detector);
+ assert!(detector.was_called.get());
+ }
+
+ #[test]
+ fn not_called_for_auto_if_not_should_detect() {
+ let detector = DetectorStub::should_not_detect();
+ _ = theme_impl(ThemeOptions::default(), &detector);
+ assert!(!detector.was_called.get());
+ }
+ }
+
+ mod precedence {
+ use super::*;
+
+ #[test]
+ fn theme_is_preferred_over_light_or_dark_themes() {
+ for color_scheme in optional(color_schemes()) {
+ for options in [
+ ThemeOptions {
+ theme: ThemePreference::Fixed(ThemeName::Named("Theme".to_string())),
+ ..Default::default()
+ },
+ ThemeOptions {
+ theme: ThemePreference::Fixed(ThemeName::Named("Theme".to_string())),
+ theme_dark: Some(ThemeName::Named("Dark Theme".to_string())),
+ theme_light: Some(ThemeName::Named("Light Theme".to_string())),
+ ..Default::default()
+ },
+ ] {
+ let detector = ConstantDetector(color_scheme);
+ assert_eq!("Theme", theme_impl(options, &detector).to_string());
+ }
+ }
+ }
+
+ #[test]
+ fn detector_is_not_called_if_theme_is_present() {
+ let options = ThemeOptions {
+ theme: ThemePreference::Fixed(ThemeName::Named("Theme".to_string())),
+ ..Default::default()
+ };
+ let detector = DetectorStub::should_detect(Some(Dark));
+ _ = theme_impl(options, &detector);
+ assert!(!detector.was_called.get());
+ }
+ }
+
+ mod default_theme {
+ use super::*;
+
+ #[test]
+ fn default_dark_if_unable_to_detect_color_scheme() {
+ let detector = ConstantDetector(None);
+ assert_eq!(
+ default_theme(ColorScheme::Dark),
+ theme_impl(ThemeOptions::default(), &detector).to_string()
+ );
+ }
+
+ // For backwards compatibility, if the default theme is requested
+ // explicitly through BAT_THEME, we always pick the default dark theme.
+ #[test]
+ fn default_dark_if_requested_explicitly_through_theme() {
+ for color_scheme in optional(color_schemes()) {
+ let options = ThemeOptions {
+ theme: ThemePreference::Fixed(ThemeName::Default),
+ ..Default::default()
+ };
+ let detector = ConstantDetector(color_scheme);
+ assert_eq!(
+ default_theme(ColorScheme::Dark),
+ theme_impl(options, &detector).to_string()
+ );
+ }
+ }
+
+ #[test]
+ fn varies_depending_on_color_scheme() {
+ for color_scheme in color_schemes() {
+ for options in [
+ ThemeOptions::default(),
+ ThemeOptions {
+ theme_dark: Some(ThemeName::Default),
+ theme_light: Some(ThemeName::Default),
+ ..Default::default()
+ },
+ ] {
+ let detector = ConstantDetector(Some(color_scheme));
+ assert_eq!(
+ default_theme(color_scheme),
+ theme_impl(options, &detector).to_string()
+ );
+ }
+ }
+ }
+ }
+
+ mod choosing {
+ use super::*;
+
+ #[test]
+ fn chooses_default_theme_if_unknown() {
+ let options = ThemeOptions {
+ theme_dark: Some(ThemeName::Named("Dark".to_string())),
+ theme_light: Some(ThemeName::Named("Light".to_string())),
+ ..Default::default()
+ };
+ let detector = ConstantDetector(None);
+ assert_eq!(
+ default_theme(ColorScheme::default()),
+ theme_impl(options, &detector).to_string()
+ );
+ }
+
+ #[test]
+ fn chooses_dark_theme_if_dark_or_unknown() {
+ let options = ThemeOptions {
+ theme_dark: Some(ThemeName::Named("Dark".to_string())),
+ theme_light: Some(ThemeName::Named("Light".to_string())),
+ ..Default::default()
+ };
+ let detector = ConstantDetector(Some(ColorScheme::Dark));
+ assert_eq!("Dark", theme_impl(options, &detector).to_string());
+ }
+
+ #[test]
+ fn chooses_light_theme_if_light() {
+ let options = ThemeOptions {
+ theme_dark: Some(ThemeName::Named("Dark".to_string())),
+ theme_light: Some(ThemeName::Named("Light".to_string())),
+ ..Default::default()
+ };
+ let detector = ConstantDetector(Some(ColorScheme::Light));
+ assert_eq!("Light", theme_impl(options, &detector).to_string());
+ }
+ }
+
+ mod theme_preference {
+ use super::*;
+
+ #[test]
+ fn values_roundtrip_via_display() {
+ let prefs = [
+ ThemePreference::Auto(DetectColorScheme::Auto),
+ ThemePreference::Auto(DetectColorScheme::Always),
+ ThemePreference::Auto(DetectColorScheme::System),
+ ThemePreference::Fixed(ThemeName::Default),
+ ThemePreference::Fixed(ThemeName::new("foo")),
+ ThemePreference::Dark,
+ ThemePreference::Light,
+ ];
+ for pref in prefs {
+ assert_eq!(pref, ThemePreference::new(&pref.to_string()));
+ }
+ }
+ }
+
+ struct DetectorStub {
+ should_detect: bool,
+ color_scheme: Option,
+ was_called: Cell,
+ }
+
+ impl DetectorStub {
+ fn should_detect(color_scheme: Option) -> Self {
+ DetectorStub {
+ should_detect: true,
+ color_scheme,
+ was_called: Cell::default(),
+ }
+ }
+
+ fn should_not_detect() -> Self {
+ DetectorStub {
+ should_detect: false,
+ color_scheme: None,
+ was_called: Cell::default(),
+ }
+ }
+ }
+
+ impl ColorSchemeDetector for DetectorStub {
+ fn should_detect(&self) -> bool {
+ self.should_detect
+ }
+
+ fn detect(&self) -> Option {
+ self.was_called.set(true);
+ self.color_scheme
+ }
+ }
+
+ struct ConstantDetector(Option);
+
+ impl ColorSchemeDetector for ConstantDetector {
+ fn should_detect(&self) -> bool {
+ true
+ }
+
+ fn detect(&self) -> Option {
+ self.0
+ }
+ }
+
+ fn optional(value: impl Iterator- ) -> impl Iterator
- > {
+ value.map(Some).chain(iter::once(None))
+ }
+
+ fn color_schemes() -> impl Iterator
- {
+ [Dark, Light].into_iter()
+ }
+}
diff --git a/tests/integration_tests.rs b/tests/integration_tests.rs
index c083a941..cd5c0846 100644
--- a/tests/integration_tests.rs
+++ b/tests/integration_tests.rs
@@ -9,7 +9,6 @@ use tempfile::tempdir;
mod unix {
pub use std::fs::File;
pub use std::io::{self, Write};
- pub use std::os::unix::io::FromRawFd;
pub use std::path::PathBuf;
pub use std::process::Stdio;
pub use std::thread;
@@ -274,11 +273,8 @@ fn squeeze_limit_line_numbers() {
#[test]
fn list_themes_with_colors() {
- #[cfg(target_os = "macos")]
- let default_theme_chunk = "Monokai Extended Light\x1B[0m (default)";
-
- #[cfg(not(target_os = "macos"))]
let default_theme_chunk = "Monokai Extended\x1B[0m (default)";
+ let default_light_theme_chunk = "Monokai Extended Light\x1B[0m (default light)";
bat()
.arg("--color=always")
@@ -287,16 +283,14 @@ fn list_themes_with_colors() {
.success()
.stdout(predicate::str::contains("DarkNeon").normalize())
.stdout(predicate::str::contains(default_theme_chunk).normalize())
+ .stdout(predicate::str::contains(default_light_theme_chunk).normalize())
.stdout(predicate::str::contains("Output the square of a number.").normalize());
}
#[test]
fn list_themes_without_colors() {
- #[cfg(target_os = "macos")]
- let default_theme_chunk = "Monokai Extended Light (default)";
-
- #[cfg(not(target_os = "macos"))]
let default_theme_chunk = "Monokai Extended (default)";
+ let default_light_theme_chunk = "Monokai Extended Light (default light)";
bat()
.arg("--color=never")
@@ -305,7 +299,8 @@ fn list_themes_without_colors() {
.assert()
.success()
.stdout(predicate::str::contains("DarkNeon").normalize())
- .stdout(predicate::str::contains(default_theme_chunk).normalize());
+ .stdout(predicate::str::contains(default_theme_chunk).normalize())
+ .stdout(predicate::str::contains(default_light_theme_chunk).normalize());
}
#[test]
@@ -415,9 +410,10 @@ fn no_args_doesnt_break() {
// as the slave end of a pseudo terminal. Although both point to the same "file", bat should
// not exit, because in this case it is safe to read and write to the same fd, which is why
// this test exists.
+
let OpenptyResult { master, slave } = openpty(None, None).expect("Couldn't open pty.");
- let mut master = unsafe { File::from_raw_fd(master) };
- let stdin_file = unsafe { File::from_raw_fd(slave) };
+ let mut master = File::from(master);
+ let stdin_file = File::from(slave);
let stdout_file = stdin_file.try_clone().unwrap();
let stdin = Stdio::from(stdin_file);
let stdout = Stdio::from(stdout_file);
@@ -425,6 +421,7 @@ fn no_args_doesnt_break() {
let mut child = bat_raw_command()
.stdin(stdin)
.stdout(stdout)
+ .env("TERM", "dumb") // Suppresses color detection
.spawn()
.expect("Failed to start.");
@@ -1020,6 +1017,31 @@ fn enable_pager_if_pp_flag_comes_before_paging() {
.stdout(predicate::eq("pager-output\n").normalize());
}
+#[test]
+fn paging_does_not_override_simple_plain() {
+ bat()
+ .env("PAGER", "echo pager-output")
+ .arg("--decorations=always")
+ .arg("--plain")
+ .arg("--paging=never")
+ .arg("test.txt")
+ .assert()
+ .success()
+ .stdout(predicate::eq("hello world\n"));
+}
+
+#[test]
+fn simple_plain_does_not_override_paging() {
+ bat()
+ .env("PAGER", "echo pager-output")
+ .arg("--paging=always")
+ .arg("--plain")
+ .arg("test.txt")
+ .assert()
+ .success()
+ .stdout(predicate::eq("pager-output\n"));
+}
+
#[test]
fn pager_failed_to_parse() {
bat()
@@ -1939,6 +1961,16 @@ fn show_all_with_unicode() {
.stderr("");
}
+#[test]
+fn binary_as_text() {
+ bat()
+ .arg("--binary=as-text")
+ .arg("control_characters.txt")
+ .assert()
+ .stdout("\x00\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0A\x0B\x0C\x0D\x0E\x0F\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1A\x1B\x1C\x1D\x1E\x1F\x7F")
+ .stderr("");
+}
+
#[test]
fn no_paging_arg() {
bat()