19 releases (9 stable)

1.7.0 Oct 15, 2024
1.4.0 Mar 11, 2024
1.1.1 Feb 19, 2023
1.0.1 Sep 26, 2022
0.6.1 Nov 11, 2021

#61 in Command-line interface

Download history 1270/week @ 2024-09-23 1909/week @ 2024-09-30 2123/week @ 2024-10-07 2874/week @ 2024-10-14 2339/week @ 2024-10-21 1897/week @ 2024-10-28 1898/week @ 2024-11-04 1521/week @ 2024-11-11 1558/week @ 2024-11-18 2365/week @ 2024-11-25 2411/week @ 2024-12-02 1833/week @ 2024-12-09 1824/week @ 2024-12-16 1392/week @ 2024-12-23 1461/week @ 2024-12-30 1714/week @ 2025-01-06

6,612 downloads per month
Used in 25 crates (14 directly)

MIT license

40KB
101 lines

MIT Latest Version docs Chat on Miaou

terminal-light

This crate answers the question "Is the terminal dark or light?".

It provides

  • the background color, either as RGB or ANSI
  • the background color's luma, which varies from 0 (black) to 1 (white)

A use case in a TUI is to determine what set of colors would be most suitable depending on the terminal's background:

let should_use_light_skin = terminal_light::luma()
    .map_or(false, |luma| luma > 0.6);

If you have very specialized skins, you may choose a more precise switch:

match terminal_light::luma() {
    Ok(luma) if luma > 0.85 => {
        // Use a "light mode" skin.
    }
    Ok(luma) if luma < 0.2 => {
        // Use a "dark mode" skin.
    }
    _ => {
        // Either we couldn't determine the mode or it's kind of medium.
        // We should use an itermediate skin, or one defining the background.
    }
}

See the included example:

dark

light

Strategies

Here are the various strategies automatically used by terminal-light to answer the big question:

$COLORFGBG strategy

This environment variable is set by some terminals, like konsole or the rxvt family. It can also be set by users. Its value is like 15;0 where the second number is the ANSI code for the background color.

Upsides:

  • querying an env variable is a fast operation

Downsides:

  • this env variable isn't always immediately updated when you change the color of the terminal
  • the value isn't precise: 0 is "dark" and 15 is "light" but the real RGB color is uncertain as the low ANSI codes are often modified by the user

"Dynamic colors" OSC escape sequence strategy

Modern terminals implement this xterm extension: a query making it possible to know the background color as RGB.

Terminal-light sends the query on stdout, waits for the answer on stdin with a timeout of 20ms, then parses this answer.

Upsides:

  • this works well on all tested unix terminals, including on MacOs
  • the value is precise (RGB)
  • the value is up to date when it's available

Downsides:

  • waiting for stdin with a timeout isn't implemented on Windows in this crate (help welcome)
  • this isn't instant, a delay of 10 ms to get the answer isn't unusual
  • if a not compatible terminal doesn't answer at all, we're waiting for 20ms
  • it may fail on some terminal multiplexers

Global strategy used by Terminal-light

  1. if we're on a unix-like platform, we try the escape sequence strategy
  2. if it failed or we're not on unix, we try the $COLORFGBG strategy
  3. without a solution, we return a TlError::Unsupported error

Dependencies

~4–13MB
~183K SLoC