2022-01-26 04:09:21 -08:00
|
|
|
package color
|
2020-12-17 23:59:45 -08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2022-01-26 01:23:18 -08:00
|
|
|
"oh-my-posh/regex"
|
2020-12-17 23:59:45 -08:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2021-11-02 06:39:51 -07:00
|
|
|
const (
|
|
|
|
colorRegex = `<(?P<foreground>[^,>]+)?,?(?P<background>[^>]+)?>(?P<content>[^<]*)<\/>`
|
|
|
|
)
|
|
|
|
|
2022-01-26 04:09:21 -08:00
|
|
|
type Writer interface {
|
|
|
|
Write(background, foreground, text string)
|
2022-02-03 08:36:37 -08:00
|
|
|
String() (string, int)
|
2022-01-26 04:09:21 -08:00
|
|
|
Reset()
|
|
|
|
SetColors(background, foreground string)
|
|
|
|
SetParentColors(background, foreground string)
|
|
|
|
ClearParentColors()
|
2021-04-20 12:30:46 -07:00
|
|
|
}
|
|
|
|
|
2022-01-26 04:09:21 -08:00
|
|
|
// AnsiWriter writes colorized ANSI strings
|
2021-11-02 06:39:51 -07:00
|
|
|
type AnsiWriter struct {
|
2022-01-26 04:09:21 -08:00
|
|
|
Ansi *Ansi
|
|
|
|
TerminalBackground string
|
2021-10-25 10:22:58 -07:00
|
|
|
Colors *Color
|
2021-12-16 01:57:01 -08:00
|
|
|
ParentColors []*Color
|
2022-01-26 04:09:21 -08:00
|
|
|
AnsiColors AnsiColors
|
|
|
|
|
|
|
|
builder strings.Builder
|
2022-02-03 08:36:37 -08:00
|
|
|
length int
|
2021-09-25 11:49:32 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
type Color struct {
|
|
|
|
Background string
|
|
|
|
Foreground string
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
|
|
|
|
2021-11-22 06:25:56 -08:00
|
|
|
// AnsiColors is the interface that wraps AnsiColorFromString method.
|
|
|
|
//
|
|
|
|
// AnsiColorFromString 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`.
|
|
|
|
type AnsiColors interface {
|
|
|
|
AnsiColorFromString(colorString string, isBackground bool) AnsiColor
|
|
|
|
}
|
|
|
|
|
|
|
|
// AnsiColor is an ANSI color code ready to be printed to the console.
|
|
|
|
// Example: "38;2;255;255;255", "48;2;255;255;255", "31", "95".
|
|
|
|
type AnsiColor string
|
|
|
|
|
|
|
|
const (
|
|
|
|
emptyAnsiColor = AnsiColor("")
|
|
|
|
transparentAnsiColor = AnsiColor(Transparent)
|
|
|
|
)
|
|
|
|
|
|
|
|
func (c AnsiColor) IsEmpty() bool {
|
|
|
|
return c == emptyAnsiColor
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c AnsiColor) IsTransparent() bool {
|
|
|
|
return c == transparentAnsiColor
|
|
|
|
}
|
|
|
|
|
2020-12-17 23:59:45 -08:00
|
|
|
const (
|
|
|
|
// Transparent implies a transparent color
|
|
|
|
Transparent = "transparent"
|
2021-11-06 05:13:45 -07:00
|
|
|
// ParentBackground takes the previous segment's background color
|
|
|
|
ParentBackground = "parentBackground"
|
|
|
|
// ParentForeground takes the previous segment's color
|
|
|
|
ParentForeground = "parentForeground"
|
2021-10-25 10:22:58 -07:00
|
|
|
// Background takes the current segment's background color
|
|
|
|
Background = "background"
|
|
|
|
// Foreground takes the current segment's foreground color
|
|
|
|
Foreground = "foreground"
|
2020-12-17 23:59:45 -08:00
|
|
|
)
|
|
|
|
|
2022-01-26 04:09:21 -08:00
|
|
|
func (a *AnsiWriter) SetColors(background, foreground string) {
|
2021-10-25 10:22:58 -07:00
|
|
|
a.Colors = &Color{
|
|
|
|
Background: background,
|
|
|
|
Foreground: foreground,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-26 04:09:21 -08:00
|
|
|
func (a *AnsiWriter) SetParentColors(background, foreground string) {
|
2021-12-16 01:57:01 -08:00
|
|
|
if a.ParentColors == nil {
|
|
|
|
a.ParentColors = make([]*Color, 0)
|
|
|
|
}
|
|
|
|
a.ParentColors = append([]*Color{{
|
2021-09-25 11:49:32 -07:00
|
|
|
Background: background,
|
|
|
|
Foreground: foreground,
|
2021-12-16 01:57:01 -08:00
|
|
|
}}, a.ParentColors...)
|
2021-09-25 11:49:32 -07:00
|
|
|
}
|
|
|
|
|
2022-01-26 04:09:21 -08:00
|
|
|
func (a *AnsiWriter) ClearParentColors() {
|
2021-10-30 02:39:08 -07:00
|
|
|
a.ParentColors = nil
|
|
|
|
}
|
|
|
|
|
2021-11-22 06:25:56 -08:00
|
|
|
func (a *AnsiWriter) getAnsiFromColorString(colorString string, isBackground bool) AnsiColor {
|
2022-01-26 04:09:21 -08:00
|
|
|
return a.AnsiColors.AnsiColorFromString(colorString, isBackground)
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
|
|
|
|
2021-11-22 06:25:56 -08:00
|
|
|
func (a *AnsiWriter) writeColoredText(background, foreground AnsiColor, text string) {
|
2020-12-27 02:21:03 -08:00
|
|
|
// Avoid emitting empty strings with color codes
|
2021-11-22 06:25:56 -08:00
|
|
|
if text == "" || (foreground.IsTransparent() && background.IsTransparent()) {
|
2020-12-27 02:21:03 -08:00
|
|
|
return
|
|
|
|
}
|
2022-04-03 04:04:56 -07:00
|
|
|
a.length += a.Ansi.MeasureText(text)
|
2021-09-25 01:52:12 -07:00
|
|
|
// default to white fg if empty, empty backgrond is supported
|
2021-11-22 06:25:56 -08:00
|
|
|
if foreground.IsEmpty() {
|
2021-09-25 01:52:12 -07:00
|
|
|
foreground = a.getAnsiFromColorString("white", false)
|
|
|
|
}
|
2022-01-26 04:09:21 -08:00
|
|
|
if foreground.IsTransparent() && !background.IsEmpty() && len(a.TerminalBackground) != 0 {
|
|
|
|
fgAnsiColor := a.getAnsiFromColorString(a.TerminalBackground, false)
|
|
|
|
coloredText := fmt.Sprintf(a.Ansi.colorFull, background, fgAnsiColor, text)
|
2021-03-14 06:14:08 -07:00
|
|
|
a.builder.WriteString(coloredText)
|
|
|
|
return
|
|
|
|
}
|
2021-11-22 06:25:56 -08:00
|
|
|
if foreground.IsTransparent() && !background.IsEmpty() {
|
2022-01-26 04:09:21 -08:00
|
|
|
coloredText := fmt.Sprintf(a.Ansi.colorTransparent, background, text)
|
2021-03-14 06:14:08 -07:00
|
|
|
a.builder.WriteString(coloredText)
|
|
|
|
return
|
2021-11-22 06:25:56 -08:00
|
|
|
} else if background.IsEmpty() || background.IsTransparent() {
|
2022-01-26 04:09:21 -08:00
|
|
|
coloredText := fmt.Sprintf(a.Ansi.colorSingle, foreground, text)
|
2021-03-14 06:14:08 -07:00
|
|
|
a.builder.WriteString(coloredText)
|
|
|
|
return
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
2022-01-26 04:09:21 -08:00
|
|
|
coloredText := fmt.Sprintf(a.Ansi.colorFull, background, foreground, text)
|
2021-01-05 23:52:43 -08:00
|
|
|
a.builder.WriteString(coloredText)
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
|
|
|
|
2021-11-22 06:25:56 -08:00
|
|
|
func (a *AnsiWriter) writeAndRemoveText(background, foreground AnsiColor, text, textToRemove, parentText string) string {
|
2020-12-17 23:59:45 -08:00
|
|
|
a.writeColoredText(background, foreground, text)
|
|
|
|
return strings.Replace(parentText, textToRemove, "", 1)
|
|
|
|
}
|
|
|
|
|
2022-01-26 04:09:21 -08:00
|
|
|
func (a *AnsiWriter) Write(background, foreground, text string) {
|
2021-06-08 10:29:09 -07:00
|
|
|
if len(text) == 0 {
|
|
|
|
return
|
|
|
|
}
|
2021-09-25 01:52:12 -07:00
|
|
|
|
2021-11-22 06:25:56 -08:00
|
|
|
bgAnsi, fgAnsi := a.asAnsiColors(background, foreground)
|
2022-01-26 04:09:21 -08:00
|
|
|
text = a.Ansi.formatText(text)
|
2022-04-03 11:38:02 -07:00
|
|
|
text = a.Ansi.GenerateHyperlink(text)
|
2022-04-03 04:04:56 -07:00
|
|
|
text = a.Ansi.EscapeText(text)
|
2021-05-26 03:03:11 -07:00
|
|
|
|
2020-12-17 23:59:45 -08:00
|
|
|
// first we match for any potentially valid colors enclosed in <>
|
2021-11-22 06:25:56 -08:00
|
|
|
// i.e., find color overrides
|
2022-01-26 01:23:18 -08:00
|
|
|
overrides := regex.FindAllNamedRegexMatch(colorRegex, text)
|
2021-11-22 06:25:56 -08:00
|
|
|
for _, override := range overrides {
|
|
|
|
fgOverride := override["foreground"]
|
|
|
|
bgOverride := override["background"]
|
|
|
|
if fgOverride == Transparent && len(bgOverride) == 0 {
|
|
|
|
bgOverride = background
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
2021-11-22 06:25:56 -08:00
|
|
|
bgOverrideAnsi, fgOverrideAnsi := a.asAnsiColors(bgOverride, fgOverride)
|
2021-09-25 01:52:12 -07:00
|
|
|
// set colors if they are empty
|
2021-11-22 06:25:56 -08:00
|
|
|
if bgOverrideAnsi.IsEmpty() {
|
|
|
|
bgOverrideAnsi = bgAnsi
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
2021-11-22 06:25:56 -08:00
|
|
|
if fgOverrideAnsi.IsEmpty() {
|
|
|
|
fgOverrideAnsi = fgAnsi
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
2021-11-22 06:25:56 -08:00
|
|
|
escapedTextSegment := override["text"]
|
|
|
|
innerText := override["content"]
|
2020-12-17 23:59:45 -08:00
|
|
|
textBeforeColorOverride := strings.Split(text, escapedTextSegment)[0]
|
2021-09-25 01:52:12 -07:00
|
|
|
text = a.writeAndRemoveText(bgAnsi, fgAnsi, textBeforeColorOverride, textBeforeColorOverride, text)
|
2021-11-22 06:25:56 -08:00
|
|
|
text = a.writeAndRemoveText(bgOverrideAnsi, fgOverrideAnsi, innerText, escapedTextSegment, text)
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
|
|
|
// color the remaining part of text with background and foreground
|
2021-09-25 01:52:12 -07:00
|
|
|
a.writeColoredText(bgAnsi, fgAnsi, text)
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
|
|
|
|
2021-11-22 06:25:56 -08:00
|
|
|
func (a *AnsiWriter) asAnsiColors(background, foreground string) (AnsiColor, AnsiColor) {
|
2021-12-15 11:09:17 -08:00
|
|
|
background = a.expandKeyword(background)
|
|
|
|
foreground = a.expandKeyword(foreground)
|
2021-11-22 06:25:56 -08:00
|
|
|
inverted := foreground == Transparent && len(background) != 0
|
|
|
|
backgroundAnsi := a.getAnsiFromColorString(background, !inverted)
|
|
|
|
foregroundAnsi := a.getAnsiFromColorString(foreground, false)
|
|
|
|
return backgroundAnsi, foregroundAnsi
|
|
|
|
}
|
|
|
|
|
2021-12-15 11:09:17 -08:00
|
|
|
func (a *AnsiWriter) isKeyword(color string) bool {
|
|
|
|
switch color {
|
|
|
|
case Transparent, ParentBackground, ParentForeground, Background, Foreground:
|
|
|
|
return true
|
2021-11-22 06:25:56 -08:00
|
|
|
default:
|
2021-12-15 11:09:17 -08:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *AnsiWriter) expandKeyword(keyword string) string {
|
2021-12-16 01:57:01 -08:00
|
|
|
resolveParentColor := func(keyword string) string {
|
|
|
|
for _, color := range a.ParentColors {
|
|
|
|
if color == nil {
|
|
|
|
return Transparent
|
|
|
|
}
|
|
|
|
switch keyword {
|
|
|
|
case ParentBackground:
|
|
|
|
keyword = color.Background
|
|
|
|
case ParentForeground:
|
|
|
|
keyword = color.Foreground
|
|
|
|
default:
|
|
|
|
return keyword
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return keyword
|
|
|
|
}
|
2021-12-15 11:09:17 -08:00
|
|
|
resolveKeyword := func(keyword string) string {
|
|
|
|
switch {
|
|
|
|
case keyword == Background && a.Colors != nil:
|
|
|
|
return a.Colors.Background
|
|
|
|
case keyword == Foreground && a.Colors != nil:
|
|
|
|
return a.Colors.Foreground
|
2021-12-16 01:57:01 -08:00
|
|
|
case (keyword == ParentBackground || keyword == ParentForeground) && a.ParentColors != nil:
|
|
|
|
return resolveParentColor(keyword)
|
2021-12-15 11:09:17 -08:00
|
|
|
default:
|
|
|
|
return Transparent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for ok := a.isKeyword(keyword); ok; ok = a.isKeyword(keyword) {
|
|
|
|
resolved := resolveKeyword(keyword)
|
|
|
|
if resolved == keyword {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
keyword = resolved
|
2021-11-22 06:25:56 -08:00
|
|
|
}
|
2021-12-15 11:09:17 -08:00
|
|
|
return keyword
|
2021-11-22 06:25:56 -08:00
|
|
|
}
|
|
|
|
|
2022-02-03 08:36:37 -08:00
|
|
|
func (a *AnsiWriter) String() (string, int) {
|
|
|
|
return a.builder.String(), a.length
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
|
|
|
|
2022-01-26 04:09:21 -08:00
|
|
|
func (a *AnsiWriter) Reset() {
|
2022-02-03 08:36:37 -08:00
|
|
|
a.length = 0
|
2021-01-05 23:52:43 -08:00
|
|
|
a.builder.Reset()
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|