Skip to content

Commit

Permalink
feat: add color profiles and detect the terminal profile on program run
Browse files Browse the repository at this point in the history
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
aymanbagabas committed Sep 12, 2024
1 parent d1827e4 commit 1751088
Show file tree
Hide file tree
Showing 5 changed files with 685 additions and 0 deletions.
178 changes: 178 additions & 0 deletions env.go
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

View workflow job for this annotation

GitHub Actions / lint-soft

`if err == nil` has complex nested blocks (complexity: 7) (nestif)

Check failure on line 138 in env.go

View workflow job for this annotation

GitHub Actions / lint-soft

`if err == nil` has complex nested blocks (complexity: 7) (nestif)
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

View workflow job for this annotation

GitHub Actions / lint-soft

Magic number: 2, in <argument> detected (gomnd)

Check failure on line 170 in env.go

View workflow job for this annotation

GitHub Actions / lint-soft

Magic number: 2, in <argument> detected (gomnd)
var value string
if len(parts) == 2 {
value = parts[1]
}
m[parts[0]] = value
}
return m
}
39 changes: 39 additions & 0 deletions examples/startup/main.go
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)
}
}
Loading

0 comments on commit 1751088

Please sign in to comment.