mirror of
https://github.com/JanDeDobbeleer/oh-my-posh.git
synced 2025-01-03 15:27:26 -08:00
151 lines
5 KiB
Go
151 lines
5 KiB
Go
|
package main
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/gookit/color"
|
||
|
)
|
||
|
|
||
|
type colorFormats struct {
|
||
|
single string
|
||
|
full string
|
||
|
transparent string
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
// Map for color names and their respective foreground [0] or background [1] color codes
|
||
|
colorMap map[string][2]string = map[string][2]string{
|
||
|
"black": {"30", "40"},
|
||
|
"red": {"31", "41"},
|
||
|
"green": {"32", "42"},
|
||
|
"yellow": {"33", "43"},
|
||
|
"blue": {"34", "44"},
|
||
|
"magenta": {"35", "45"},
|
||
|
"cyan": {"36", "46"},
|
||
|
"white": {"37", "47"},
|
||
|
"default": {"39", "49"},
|
||
|
"darkGray": {"90", "100"},
|
||
|
"lightRed": {"91", "101"},
|
||
|
"lightGreen": {"92", "102"},
|
||
|
"lightYellow": {"93", "103"},
|
||
|
"lightBlue": {"94", "104"},
|
||
|
"lightMagenta": {"95", "105"},
|
||
|
"lightCyan": {"96", "106"},
|
||
|
"lightWhite": {"97", "107"},
|
||
|
}
|
||
|
)
|
||
|
|
||
|
// Returns the color code for a given color name
|
||
|
func getColorFromName(colorName string, isBackground bool) (string, error) {
|
||
|
colorMapOffset := 0
|
||
|
if isBackground {
|
||
|
colorMapOffset = 1
|
||
|
}
|
||
|
if colorCodes, found := colorMap[colorName]; found {
|
||
|
return colorCodes[colorMapOffset], nil
|
||
|
}
|
||
|
return "", errors.New("color name does not exist")
|
||
|
}
|
||
|
|
||
|
// AnsiColor writes colorized strings
|
||
|
type AnsiColor struct {
|
||
|
buffer *bytes.Buffer
|
||
|
formats *colorFormats
|
||
|
}
|
||
|
|
||
|
const (
|
||
|
// Transparent implies a transparent color
|
||
|
Transparent = "transparent"
|
||
|
)
|
||
|
|
||
|
func (a *AnsiColor) init(shell string) {
|
||
|
a.formats = &colorFormats{}
|
||
|
switch shell {
|
||
|
case zsh:
|
||
|
a.formats.single = "%%{\x1b[%sm%%}%s%%{\x1b[0m%%}"
|
||
|
a.formats.full = "%%{\x1b[%sm\x1b[%sm%%}%s%%{\x1b[0m%%}"
|
||
|
a.formats.transparent = "%%{\x1b[%s;49m\x1b[7m%%}%s%%{\x1b[m\x1b[0m%%}"
|
||
|
case bash:
|
||
|
a.formats.single = "\\[\x1b[%sm\\]%s\\[\x1b[0m\\]"
|
||
|
a.formats.full = "\\[\x1b[%sm\x1b[%sm\\]%s\\[\x1b[0m\\]"
|
||
|
a.formats.transparent = "\\[\x1b[%s;49m\x1b[7m\\]%s\\[\x1b[m\x1b[0m\\]"
|
||
|
default:
|
||
|
a.formats.single = "\x1b[%sm%s\x1b[0m"
|
||
|
a.formats.full = "\x1b[%sm\x1b[%sm%s\x1b[0m"
|
||
|
a.formats.transparent = "\x1b[%s;49m\x1b[7m%s\x1b[m\x1b[0m"
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Gets the ANSI color code for a given color string.
|
||
|
// This can include a valid hex color in the format `#FFFFFF`,
|
||
|
// but also a name of one of the first 16 ANSI colors like `lightBlue`.
|
||
|
func (a *AnsiColor) getAnsiFromColorString(colorString string, isBackground bool) string {
|
||
|
colorFromName, err := getColorFromName(colorString, isBackground)
|
||
|
if err == nil {
|
||
|
return colorFromName
|
||
|
}
|
||
|
style := color.HEX(colorString, isBackground)
|
||
|
return style.Code()
|
||
|
}
|
||
|
|
||
|
func (a *AnsiColor) writeColoredText(background, foreground, text string) {
|
||
|
var coloredText string
|
||
|
if foreground == Transparent && background != "" {
|
||
|
ansiColor := a.getAnsiFromColorString(background, false)
|
||
|
coloredText = fmt.Sprintf(a.formats.transparent, ansiColor, text)
|
||
|
} else if background == "" || background == Transparent {
|
||
|
ansiColor := a.getAnsiFromColorString(foreground, false)
|
||
|
coloredText = fmt.Sprintf(a.formats.single, ansiColor, text)
|
||
|
} else if foreground != "" && background != "" {
|
||
|
bgAnsiColor := a.getAnsiFromColorString(background, true)
|
||
|
fgAnsiColor := a.getAnsiFromColorString(foreground, false)
|
||
|
coloredText = fmt.Sprintf(a.formats.full, bgAnsiColor, fgAnsiColor, text)
|
||
|
}
|
||
|
a.buffer.WriteString(coloredText)
|
||
|
}
|
||
|
|
||
|
func (a *AnsiColor) writeAndRemoveText(background, foreground, text, textToRemove, parentText string) string {
|
||
|
a.writeColoredText(background, foreground, text)
|
||
|
return strings.Replace(parentText, textToRemove, "", 1)
|
||
|
}
|
||
|
|
||
|
func (a *AnsiColor) write(background, foreground, text string) {
|
||
|
// first we match for any potentially valid colors enclosed in <>
|
||
|
match := findAllNamedRegexMatch(`<(?P<foreground>[^,>]+)?,?(?P<background>[^>]+)?>(?P<content>[^<]*)<\/>`, text)
|
||
|
for i := range match {
|
||
|
extractedForegroundColor := match[i]["foreground"]
|
||
|
extractedBackgroundColor := match[i]["background"]
|
||
|
if col := a.getAnsiFromColorString(extractedForegroundColor, false); col == "" && extractedForegroundColor != Transparent && len(extractedBackgroundColor) == 0 {
|
||
|
continue // we skip invalid colors
|
||
|
}
|
||
|
if col := a.getAnsiFromColorString(extractedBackgroundColor, false); col == "" && extractedBackgroundColor != Transparent && len(extractedForegroundColor) == 0 {
|
||
|
continue // we skip invalid colors
|
||
|
}
|
||
|
// reuse function colors if only one was specified
|
||
|
if len(extractedBackgroundColor) == 0 {
|
||
|
extractedBackgroundColor = background
|
||
|
}
|
||
|
if len(extractedForegroundColor) == 0 {
|
||
|
extractedForegroundColor = foreground
|
||
|
}
|
||
|
escapedTextSegment := match[i]["text"]
|
||
|
innerText := match[i]["content"]
|
||
|
textBeforeColorOverride := strings.Split(text, escapedTextSegment)[0]
|
||
|
text = a.writeAndRemoveText(background, foreground, textBeforeColorOverride, textBeforeColorOverride, text)
|
||
|
text = a.writeAndRemoveText(extractedBackgroundColor, extractedForegroundColor, innerText, escapedTextSegment, text)
|
||
|
}
|
||
|
// color the remaining part of text with background and foreground
|
||
|
a.writeColoredText(background, foreground, text)
|
||
|
}
|
||
|
|
||
|
func (a *AnsiColor) string() string {
|
||
|
return a.buffer.String()
|
||
|
}
|
||
|
|
||
|
func (a *AnsiColor) reset() {
|
||
|
a.buffer.Reset()
|
||
|
}
|