|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
black int = iota
|
|
|
|
red
|
|
|
|
green
|
|
|
|
yellow
|
|
|
|
purple
|
|
|
|
magenta
|
|
|
|
cyan
|
|
|
|
grey
|
|
|
|
normal int = -1
|
|
|
|
)
|
|
|
|
|
|
|
|
var colorMapString = map[string]int{
|
|
|
|
"black": black,
|
|
|
|
"red": red,
|
|
|
|
"green": green,
|
|
|
|
"yellow": yellow,
|
|
|
|
"purple": purple,
|
|
|
|
"magenta": magenta,
|
|
|
|
"cyan": cyan,
|
|
|
|
"grey": grey,
|
|
|
|
"normal": normal,
|
|
|
|
}
|
|
|
|
|
|
|
|
var colorMapInt = map[int]string{
|
|
|
|
black: "black",
|
|
|
|
red: "red",
|
|
|
|
green: "green",
|
|
|
|
yellow: "yellow",
|
|
|
|
purple: "purple",
|
|
|
|
magenta: "magenta",
|
|
|
|
cyan: "cyan",
|
|
|
|
grey: "grey",
|
|
|
|
normal: "normal",
|
|
|
|
}
|
|
|
|
|
|
|
|
func colorFromString(color string) int {
|
|
|
|
var result int
|
|
|
|
color = strings.ToLower(color)
|
|
|
|
result, ok := colorMapString[color]
|
|
|
|
if !ok {
|
|
|
|
return normal
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func colorFromInt(color int) string {
|
|
|
|
var result string
|
|
|
|
result, ok := colorMapInt[color]
|
|
|
|
if !ok {
|
|
|
|
return "normal"
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
var basicStyle = Style{
|
|
|
|
Foreground: colorMapInt[normal],
|
|
|
|
Background: colorMapInt[normal],
|
|
|
|
Italic: false,
|
|
|
|
Bold: false,
|
|
|
|
Underline: false,
|
|
|
|
Strikethrough: false,
|
|
|
|
Inverse: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s Style) withForeground(color int) Style {
|
|
|
|
s.Foreground = colorFromInt(color)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
func (s Style) withBackground(color int) Style {
|
|
|
|
s.Background = colorFromInt(color)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s Style) withBold() Style {
|
|
|
|
s.Bold = true
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
func (s Style) withInverse() Style {
|
|
|
|
s.Inverse = true
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
func (s Style) withItalic() Style {
|
|
|
|
s.Italic = true
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
func (s Style) withStrikethrough() Style {
|
|
|
|
s.Strikethrough = true
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
func (s Style) withUnderline() Style {
|
|
|
|
s.Underline = true
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO create both as `reset` (which it is now) as well as `append`
|
|
|
|
// which essentially just adds on top. that is relevant in the case of
|
|
|
|
// bold/italic etc - it should add style - not clear.
|
|
|
|
func (s Style) toANSI() string {
|
|
|
|
if config.Basics.Colorless {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
styleSlice := []string{"0"}
|
|
|
|
|
|
|
|
if colorFromString(s.Foreground) != normal {
|
|
|
|
styleSlice = append(styleSlice, fmt.Sprintf("%d", 30+colorFromString(s.Foreground)))
|
|
|
|
}
|
|
|
|
if colorFromString(s.Background) != normal {
|
|
|
|
styleSlice = append(styleSlice, fmt.Sprintf("%d", 40+colorFromString(s.Background)))
|
|
|
|
}
|
|
|
|
if s.Bold {
|
|
|
|
styleSlice = append(styleSlice, "1")
|
|
|
|
}
|
|
|
|
if s.Italic {
|
|
|
|
styleSlice = append(styleSlice, "3")
|
|
|
|
}
|
|
|
|
if s.Underline {
|
|
|
|
styleSlice = append(styleSlice, "4")
|
|
|
|
}
|
|
|
|
if s.Inverse {
|
|
|
|
styleSlice = append(styleSlice, "7")
|
|
|
|
}
|
|
|
|
if s.Strikethrough {
|
|
|
|
styleSlice = append(styleSlice, "9")
|
|
|
|
}
|
|
|
|
|
|
|
|
return "\x1b[" + strings.Join(styleSlice, ";") + "m"
|
|
|
|
}
|
|
|
|
|
|
|
|
// End Colors
|
|
|
|
// Begin StyledString
|
|
|
|
|
|
|
|
// StyledString is used to save a message with a style, which can then later be rendered to a string
|
|
|
|
type StyledString struct {
|
|
|
|
message string
|
|
|
|
style Style
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ss StyledString) withStyle(style Style) StyledString {
|
|
|
|
return StyledString{ss.message, style}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO change StyledString to have styles at start-end indexes.
|
|
|
|
|
|
|
|
// TODO handle all formatting types
|
|
|
|
func (s Style) sprintf(base string, parts ...StyledString) StyledString {
|
|
|
|
text := s.stylize(removeFormatting(base))
|
|
|
|
//TODO handle posibility to escape
|
|
|
|
re := regexp.MustCompile(`\$TEXT`)
|
|
|
|
for len(re.FindAllString(text.message, 1)) > 0 {
|
|
|
|
part := parts[0]
|
|
|
|
parts = parts[1:]
|
|
|
|
text = text.replaceN("$TEXT", part, 1)
|
|
|
|
}
|
|
|
|
return text
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s Style) stylize(msg string) StyledString {
|
|
|
|
return StyledString{msg, s}
|
|
|
|
}
|
|
|
|
func (ss StyledString) stringFollowedByStyle(style Style) string {
|
|
|
|
return ss.style.toANSI() + ss.message + style.toANSI()
|
|
|
|
}
|
|
|
|
func (ss StyledString) string() string {
|
|
|
|
return ss.stringFollowedByStyle(basicStyle)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ss StyledString) replace(match string, value StyledString) StyledString {
|
|
|
|
return ss.replaceN(match, value, -1)
|
|
|
|
}
|
|
|
|
func (ss StyledString) replaceN(match string, value StyledString, n int) StyledString {
|
|
|
|
ss.message = strings.Replace(ss.message, match, value.stringFollowedByStyle(ss.style), n)
|
|
|
|
return ss
|
|
|
|
}
|
|
|
|
func (ss StyledString) replaceString(match string, value string) StyledString {
|
|
|
|
ss.message = strings.Replace(ss.message, match, value, -1)
|
|
|
|
return ss
|
|
|
|
}
|
|
|
|
|
|
|
|
// Overrides current formatting
|
|
|
|
func (ss StyledString) colorRegex(match string, style Style) StyledString {
|
|
|
|
return ss.regexReplaceFunc(match, func(subString string) string {
|
|
|
|
return style.stylize(removeFormatting(subString)).stringFollowedByStyle(ss.style)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Replacer function takes the current match as input and should return how the match should be preseneted instead
|
|
|
|
func (ss StyledString) regexReplaceFunc(match string, replacer func(string) string) StyledString {
|
|
|
|
re := regexp.MustCompile(match)
|
|
|
|
locations := re.FindAllStringIndex(ss.message, -1)
|
|
|
|
var newMessage string
|
|
|
|
var prevIndex int
|
|
|
|
for _, loc := range locations {
|
|
|
|
newSubstring := replacer(ss.message[loc[0]:loc[1]])
|
|
|
|
newMessage += ss.message[prevIndex:loc[0]]
|
|
|
|
newMessage += newSubstring
|
|
|
|
prevIndex = loc[1]
|
|
|
|
}
|
|
|
|
// Append any string after the final match
|
|
|
|
newMessage += ss.message[prevIndex:len(ss.message)]
|
|
|
|
ss.message = newMessage
|
|
|
|
return ss
|
|
|
|
}
|
|
|
|
|
|
|
|
// Appends the other stylize at the end, but retains same style
|
|
|
|
func (ss StyledString) append(other StyledString) StyledString {
|
|
|
|
ss.message = ss.message + other.stringFollowedByStyle(ss.style)
|
|
|
|
return ss
|
|
|
|
}
|
|
|
|
func (ss StyledString) appendString(other string) StyledString {
|
|
|
|
ss.message += other
|
|
|
|
return ss
|
|
|
|
}
|
|
|
|
|
|
|
|
// Begin Formatting
|
|
|
|
|
|
|
|
func removeFormatting(s string) string {
|
|
|
|
reFormatting := regexp.MustCompile(`(?m)\x1b\[(\d*;?)*m`)
|
|
|
|
return reFormatting.ReplaceAllString(s, "")
|
|
|
|
}
|