package main import ( "fmt" "github.com/pelletier/go-toml" "regexp" "strings" ) // Begin Colors type color int const ( black color = iota red green yellow purple magenta cyan grey normal color = -1 ) func colorFromString(s string) color { s = strings.ToLower(s) switch s { case "black": return black case "red": return red case "green": return green case "yellow": return yellow case "purple": return purple case "magenta": return magenta case "cyan": return cyan case "grey": return grey case "normal": return normal default: printError(fmt.Sprintf("color `%s` cannot be parsed.", s)) } return normal } // Style struct for specializing the style/color of a stylize type Style struct { foregroundColor color backgroundColor color bold bool italic bool // Currently not supported by the UI library underline bool strikethrough bool // Currently not supported by the UI library inverse bool } var basicStyle = Style{normal, normal, false, false, false, false, false} func styleFromConfig(config *toml.Tree, key string) Style { key = "Colors." + key + "." style := basicStyle if config.Has(key + "foreground") { style = style.withForeground(colorFromString(config.Get(key + "foreground").(string))) } if config.Has(key + "background") { style = style.withForeground(colorFromString(config.Get(key + "background").(string))) } if config.GetDefault(key+"bold", false).(bool) { style = style.withBold() } if config.GetDefault(key+"italic", false).(bool) { style = style.withItalic() } if config.GetDefault(key+"underline", false).(bool) { style = style.withUnderline() } if config.GetDefault(key+"strikethrough", false).(bool) { style = style.withStrikethrough() } if config.GetDefault(key+"inverse", false).(bool) { style = style.withInverse() } return style } func (s Style) withForeground(f color) Style { s.foregroundColor = f return s } func (s Style) withBackground(f color) Style { s.backgroundColor = f 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 colorless { return "" } output := "\x1b[0m\x1b[0" if s.foregroundColor != normal { output += fmt.Sprintf(";%d", 30+s.foregroundColor) } if s.backgroundColor != normal { output += fmt.Sprintf(";%d", 40+s.backgroundColor) } if s.bold { output += ";1" } if s.italic { output += ";3" } if s.underline { output += ";4" } if s.inverse { output += ";7" } if s.strikethrough { output += ";9" } return output + "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 } // 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 (t StyledString) stringFollowedByStyle(style Style) string { return t.style.toANSI() + t.message + style.toANSI() } func (t StyledString) string() string { return t.stringFollowedByStyle(basicStyle) } func (t StyledString) replace(match string, value StyledString) StyledString { return t.replaceN(match, value, -1) } func (t StyledString) replaceN(match string, value StyledString, n int) StyledString { t.message = strings.Replace(t.message, match, value.stringFollowedByStyle(t.style), n) return t } func (t StyledString) replaceString(match string, value string) StyledString { t.message = strings.Replace(t.message, match, value, -1) return t } // Overrides current formatting func (t StyledString) colorRegex(match string, style Style) StyledString { re := regexp.MustCompile("(" + match + ")") locations := re.FindAllStringIndex(t.message, -1) var newMessage string var prevIndex int for _, loc := range locations { cleanSubstring := style.stylize(removeFormatting(string(t.message[loc[0]:loc[1]]))) newMessage += t.message[prevIndex:loc[0]] newMessage += cleanSubstring.stringFollowedByStyle(t.style) prevIndex = loc[1] } // Append any string after the final match newMessage += t.message[prevIndex:len(t.message)] t.message = newMessage return t } // Appends the other stylize at the end, but retains same style func (t StyledString) append(other StyledString) StyledString { t.message = t.message + other.stringFollowedByStyle(t.style) return t } func (t StyledString) appendString(other string) StyledString { t.message += other return t } // Begin Formatting func removeFormatting(s string) string { reFormatting := regexp.MustCompile(`(?m)\x1b\[(\d*;?)*m`) return reFormatting.ReplaceAllString(s, "") }