-
Notifications
You must be signed in to change notification settings - Fork 841
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: add color profiles and detect the terminal profile on program run
This defines `tea.Profile` and its consts. A color profile can downgrade `color.Color` based on terminal colors support. When the program starts, it will detect the color profile from the output and the program environment variables before sending the profile to the program model. This also adds a `startup` example that demonstrates messages a program receive during startup.
- Loading branch information
1 parent
45222df
commit 0d56e62
Showing
5 changed files
with
685 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,178 @@ | ||
package tea | ||
|
||
import ( | ||
"io" | ||
"strconv" | ||
"strings" | ||
|
||
"github.com/charmbracelet/x/term" | ||
"github.com/xo/terminfo" | ||
) | ||
|
||
// detectColorProfile returns the color profile based on the terminal output, | ||
// and environment variables. This respects NO_COLOR, CLICOLOR, and | ||
// CLICOLOR_FORCE environment variables. | ||
// | ||
// The rules as follows: | ||
// - TERM=dumb is always treated as NoTTY unless CLICOLOR_FORCE=1 is set. | ||
// - If COLORTERM=truecolor, and the profile is not NoTTY, it gest upgraded to TrueColor. | ||
// - Using any 256 color terminal (e.g. TERM=xterm-256color) will set the profile to ANSI256. | ||
// - Using any color terminal (e.g. TERM=xterm-color) will set the profile to ANSI. | ||
// - Using CLICOLOR=1 without TERM defined should be treated as ANSI if the | ||
// output is a terminal. | ||
// - NO_COLOR takes precedence over CLICOLOR/CLICOLOR_FORCE, and will disable | ||
// colors but not text decoration, i.e. bold, italic, faint, etc. | ||
// | ||
// See https://no-color.org/ and https://bixense.com/clicolors/ for more information. | ||
func detectColorProfile(output io.Writer, environ []string) (p Profile) { | ||
out, ok := output.(term.File) | ||
isatty := ok && term.IsTerminal(out.Fd()) | ||
return colorProfile(isatty, environ) | ||
} | ||
|
||
func colorProfile(isatty bool, environ []string) (p Profile) { | ||
env := environMap(environ) | ||
envProfile := envColorProfile(env) | ||
|
||
// Start with the environment profile. | ||
p = envProfile | ||
|
||
term := strings.ToLower(env["TERM"]) | ||
isDumb := term == "dumb" | ||
|
||
// Check if the output is a terminal. | ||
// Treat dumb terminals as NoTTY | ||
if !isatty || isDumb { | ||
p = NoTTY | ||
} | ||
|
||
if envNoColor(env) { | ||
if p < Ascii { | ||
p = Ascii | ||
} | ||
return | ||
} | ||
|
||
if cliColorForced(env) { | ||
if p > ANSI { | ||
p = ANSI | ||
} | ||
if envProfile < p { | ||
p = envProfile | ||
} | ||
|
||
return | ||
} | ||
|
||
if cliColor(env) { | ||
if isatty && !isDumb && p > ANSI { | ||
p = ANSI | ||
} | ||
} | ||
|
||
return p | ||
} | ||
|
||
// envNoColor returns true if the environment variables explicitly disable color output | ||
// by setting NO_COLOR (https://no-color.org/). | ||
func envNoColor(env map[string]string) bool { | ||
noColor, _ := strconv.ParseBool(env["NO_COLOR"]) | ||
return noColor | ||
} | ||
|
||
func cliColor(env map[string]string) bool { | ||
cliColor, _ := strconv.ParseBool(env["CLICOLOR"]) | ||
return cliColor | ||
} | ||
|
||
func cliColorForced(env map[string]string) bool { | ||
cliColorForce, _ := strconv.ParseBool(env["CLICOLOR_FORCE"]) | ||
return cliColorForce | ||
} | ||
|
||
func colorTerm(env map[string]string) bool { | ||
colorTerm := strings.ToLower(env["COLORTERM"]) | ||
return colorTerm == "truecolor" || colorTerm == "24bit" || | ||
colorTerm == "yes" || colorTerm == "true" | ||
} | ||
|
||
// envColorProfile returns infers the color profile from the environment. | ||
func envColorProfile(env map[string]string) (p Profile) { | ||
p = Ascii // Default to Ascii | ||
if isCloudShell, _ := strconv.ParseBool(env["GOOGLE_CLOUD_SHELL"]); isCloudShell { | ||
p = TrueColor | ||
return | ||
} | ||
|
||
term := strings.ToLower(env["TERM"]) | ||
switch term { | ||
case "", "dumb": | ||
p = NoTTY | ||
} | ||
|
||
if colorTerm(env) { | ||
p = TrueColor | ||
return | ||
} | ||
|
||
switch term { | ||
case "alacritty", "contour", "wezterm", "xterm-ghostty", "xterm-kitty": | ||
p = TrueColor | ||
return | ||
case "linux": | ||
if p > ANSI { | ||
p = ANSI | ||
} | ||
} | ||
|
||
if strings.Contains(term, "256color") && p > ANSI256 { | ||
p = ANSI256 | ||
} | ||
if strings.Contains(term, "color") && p > ANSI { | ||
p = ANSI | ||
} | ||
if strings.Contains(term, "ansi") && p > ANSI { | ||
p = ANSI | ||
} | ||
|
||
if ti, err := terminfo.Load(term); err == nil { | ||
Check failure on line 138 in env.go GitHub Actions / lint-soft
|
||
extbools := ti.ExtBoolCapsShort() | ||
if _, ok := extbools["RGB"]; ok { | ||
p = TrueColor | ||
return | ||
} | ||
|
||
if _, ok := extbools["Tc"]; ok { | ||
p = TrueColor | ||
return | ||
} | ||
|
||
nums := ti.NumCapsShort() | ||
if colors, ok := nums["colors"]; ok { | ||
if colors >= 0x1000000 { | ||
p = TrueColor | ||
return | ||
} else if colors >= 0x100 && p > ANSI256 { | ||
p = ANSI256 | ||
} else if colors >= 0x10 && p > ANSI { | ||
p = ANSI | ||
} | ||
} | ||
} | ||
|
||
return | ||
} | ||
|
||
// environMap converts an environment slice to a map. | ||
func environMap(environ []string) map[string]string { | ||
m := make(map[string]string, len(environ)) | ||
for _, e := range environ { | ||
parts := strings.SplitN(e, "=", 2) | ||
Check failure on line 170 in env.go GitHub Actions / lint-soft
|
||
var value string | ||
if len(parts) == 2 { | ||
value = parts[1] | ||
} | ||
m[parts[0]] = value | ||
} | ||
return m | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,39 @@ | ||
package main | ||
|
||
import ( | ||
"log" | ||
|
||
tea "github.com/charmbracelet/bubbletea" | ||
) | ||
|
||
type model struct{} | ||
|
||
var _ tea.Model = model{} | ||
|
||
// Init implements tea.Model. | ||
func (m model) Init() (tea.Model, tea.Cmd) { | ||
return m, tea.TerminalVersion | ||
} | ||
|
||
// Update implements tea.Model. | ||
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) { | ||
switch msg := msg.(type) { | ||
case tea.KeyMsg: | ||
return m, tea.Quit | ||
case tea.BackgroundColorMsg, tea.ForegroundColorMsg, tea.CursorColorMsg, tea.TerminalVersionMsg, tea.ColorProfileMsg: | ||
return m, tea.Printf("Received a terminal startup message: %T: %s", msg, msg) | ||
} | ||
return m, nil | ||
} | ||
|
||
// View implements tea.Model. | ||
func (m model) View() string { | ||
return "Press any key to exit." | ||
} | ||
|
||
func main() { | ||
p := tea.NewProgram(model{}) | ||
if _, err := p.Run(); err != nil { | ||
log.Fatal(err) | ||
} | ||
} |
Oops, something went wrong.