2017-06-06 13:23:50 +02:00
|
|
|
package chroma
|
|
|
|
|
2017-07-20 08:51:16 +02:00
|
|
|
import (
|
2017-09-21 04:52:08 +02:00
|
|
|
"fmt"
|
2017-07-20 08:51:16 +02:00
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
)
|
2017-06-06 13:23:50 +02:00
|
|
|
|
|
|
|
// A StyleEntry in the Style map.
|
|
|
|
type StyleEntry struct {
|
|
|
|
// Hex colours.
|
|
|
|
Colour Colour
|
|
|
|
Background Colour
|
|
|
|
Border Colour
|
|
|
|
|
|
|
|
Bold bool
|
|
|
|
Italic bool
|
|
|
|
Underline bool
|
|
|
|
}
|
|
|
|
|
2017-09-20 06:15:06 +02:00
|
|
|
// Clone this StyleEntry.
|
|
|
|
func (s *StyleEntry) Clone() *StyleEntry {
|
|
|
|
clone := &StyleEntry{}
|
|
|
|
*clone = *s
|
|
|
|
return clone
|
|
|
|
}
|
|
|
|
|
2017-07-20 08:51:16 +02:00
|
|
|
func (s *StyleEntry) String() string {
|
2017-06-06 13:23:50 +02:00
|
|
|
out := []string{}
|
2017-07-20 08:51:16 +02:00
|
|
|
if s.Bold {
|
2017-06-06 13:23:50 +02:00
|
|
|
out = append(out, "bold")
|
|
|
|
}
|
2017-07-20 08:51:16 +02:00
|
|
|
if s.Italic {
|
2017-06-06 13:23:50 +02:00
|
|
|
out = append(out, "italic")
|
|
|
|
}
|
2017-07-20 08:51:16 +02:00
|
|
|
if s.Underline {
|
2017-06-06 13:23:50 +02:00
|
|
|
out = append(out, "underline")
|
|
|
|
}
|
2017-07-20 08:51:16 +02:00
|
|
|
if s.Colour.IsSet() {
|
|
|
|
out = append(out, s.Colour.String())
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
2017-07-20 08:51:16 +02:00
|
|
|
if s.Background.IsSet() {
|
|
|
|
out = append(out, "bg:"+s.Background.String())
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
2017-07-20 08:51:16 +02:00
|
|
|
if s.Border.IsSet() {
|
|
|
|
out = append(out, "border:"+s.Border.String())
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
|
|
|
return strings.Join(out, " ")
|
|
|
|
}
|
|
|
|
|
2017-07-20 08:51:16 +02:00
|
|
|
func (s *StyleEntry) IsZero() bool {
|
|
|
|
return s.Colour == 0 && s.Background == 0 && s.Border == 0 && !s.Bold && !s.Italic && !s.Underline
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *StyleEntry) Sub(e *StyleEntry) *StyleEntry {
|
|
|
|
out := &StyleEntry{}
|
|
|
|
if e.Colour != s.Colour {
|
|
|
|
out.Colour = s.Colour
|
|
|
|
}
|
|
|
|
if e.Background != s.Background {
|
|
|
|
out.Background = s.Background
|
|
|
|
}
|
|
|
|
if e.Bold != s.Bold {
|
|
|
|
out.Bold = s.Bold
|
|
|
|
}
|
|
|
|
if e.Italic != s.Italic {
|
|
|
|
out.Italic = s.Italic
|
|
|
|
}
|
|
|
|
if e.Underline != s.Underline {
|
|
|
|
out.Underline = s.Underline
|
|
|
|
}
|
|
|
|
if e.Border != s.Border {
|
|
|
|
out.Border = s.Border
|
|
|
|
}
|
|
|
|
return out
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// StyleEntries mapping TokenType to colour definition.
|
|
|
|
type StyleEntries map[TokenType]string
|
|
|
|
|
|
|
|
// NewStyle creates a new style definition.
|
2017-09-21 04:52:08 +02:00
|
|
|
func NewStyle(name string, entries StyleEntries) (*Style, error) {
|
2017-06-06 13:23:50 +02:00
|
|
|
s := &Style{
|
2017-07-20 08:51:16 +02:00
|
|
|
Name: name,
|
|
|
|
Entries: map[TokenType]*StyleEntry{},
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
2017-09-21 04:52:08 +02:00
|
|
|
if err := s.Add(Background, ""); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := s.AddAll(entries); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MustNewStyle creates a new style or panics.
|
|
|
|
func MustNewStyle(name string, entries StyleEntries) *Style {
|
|
|
|
style, err := NewStyle(name, entries)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return style
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// A Style definition.
|
|
|
|
//
|
|
|
|
// See http://pygments.org/docs/styles/ for details. Semantics are intended to be identical.
|
|
|
|
type Style struct {
|
|
|
|
Name string
|
|
|
|
Entries map[TokenType]*StyleEntry
|
|
|
|
}
|
|
|
|
|
2017-09-20 06:15:06 +02:00
|
|
|
// Clone this style. The clone can then be safely modified.
|
|
|
|
func (s *Style) Clone() *Style {
|
|
|
|
clone := &Style{
|
|
|
|
Name: s.Name,
|
|
|
|
Entries: map[TokenType]*StyleEntry{},
|
|
|
|
}
|
|
|
|
for tt, e := range s.Entries {
|
|
|
|
clone.Entries[tt] = e.Clone()
|
|
|
|
}
|
|
|
|
return clone
|
|
|
|
}
|
|
|
|
|
2017-06-06 13:23:50 +02:00
|
|
|
// Get a style entry. Will try sub-category or category if an exact match is not found, and
|
|
|
|
// finally return the entry mapped to `InheritStyle`.
|
|
|
|
func (s *Style) Get(ttype TokenType) *StyleEntry {
|
|
|
|
out := s.Entries[ttype]
|
|
|
|
if out == nil {
|
|
|
|
out = s.Entries[ttype.SubCategory()]
|
|
|
|
if out == nil {
|
|
|
|
out = s.Entries[ttype.Category()]
|
|
|
|
if out == nil {
|
2017-07-20 08:51:16 +02:00
|
|
|
out = s.Entries[Background]
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2017-09-21 04:52:08 +02:00
|
|
|
func (s *Style) AddAll(entries StyleEntries) error {
|
2017-07-20 08:51:16 +02:00
|
|
|
tis := []int{}
|
|
|
|
for tt := range entries {
|
|
|
|
tis = append(tis, int(tt))
|
|
|
|
}
|
|
|
|
sort.Ints(tis)
|
|
|
|
for _, ti := range tis {
|
|
|
|
tt := TokenType(ti)
|
|
|
|
entry := entries[tt]
|
2017-09-21 04:52:08 +02:00
|
|
|
if err := s.Add(tt, entry); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-07-20 08:51:16 +02:00
|
|
|
}
|
2017-09-21 04:52:08 +02:00
|
|
|
return nil
|
2017-07-20 08:51:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add a StyleEntry to the Style map.
|
2017-06-06 13:23:50 +02:00
|
|
|
//
|
|
|
|
// See http://pygments.org/docs/styles/#style-rules for details.
|
2017-09-21 04:52:08 +02:00
|
|
|
func (s *Style) Add(ttype TokenType, entry string) error { // nolint: gocyclo
|
2017-06-06 13:23:50 +02:00
|
|
|
dupl := s.Entries[ttype.SubCategory()]
|
|
|
|
if dupl == nil {
|
|
|
|
dupl = s.Entries[ttype.Category()]
|
|
|
|
if dupl == nil {
|
2017-07-20 08:51:16 +02:00
|
|
|
dupl = s.Entries[Background]
|
|
|
|
if dupl == nil {
|
|
|
|
dupl = &StyleEntry{}
|
|
|
|
}
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
parent := &StyleEntry{}
|
|
|
|
// Duplicate ancestor node.
|
|
|
|
*parent = *dupl
|
2017-09-21 04:52:08 +02:00
|
|
|
se, err := ParseStyleEntry(parent, entry)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.Entries[ttype] = se
|
|
|
|
return nil
|
2017-07-20 08:51:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// ParseStyleEntry parses a Pygments style entry.
|
2017-09-21 04:52:08 +02:00
|
|
|
func ParseStyleEntry(parent *StyleEntry, entry string) (*StyleEntry, error) { // nolint: gocyclo
|
2017-07-20 08:51:16 +02:00
|
|
|
out := &StyleEntry{}
|
|
|
|
parts := strings.Fields(entry)
|
|
|
|
// Check if parent style should be inherited...
|
|
|
|
if parent != nil {
|
|
|
|
inherit := true
|
|
|
|
for _, part := range parts {
|
|
|
|
if part == "noinherit" {
|
|
|
|
inherit = false
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if inherit {
|
|
|
|
*out = *parent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, part := range parts {
|
2017-06-06 13:23:50 +02:00
|
|
|
switch {
|
|
|
|
case part == "italic":
|
|
|
|
out.Italic = true
|
|
|
|
case part == "noitalic":
|
|
|
|
out.Italic = false
|
|
|
|
case part == "bold":
|
|
|
|
out.Bold = true
|
|
|
|
case part == "nobold":
|
|
|
|
out.Bold = false
|
|
|
|
case part == "underline":
|
|
|
|
out.Underline = true
|
|
|
|
case part == "nounderline":
|
|
|
|
out.Underline = false
|
2017-07-20 08:51:16 +02:00
|
|
|
case part == "bg:":
|
|
|
|
out.Background = 0
|
2017-06-06 13:23:50 +02:00
|
|
|
case strings.HasPrefix(part, "bg:#"):
|
|
|
|
out.Background = ParseColour(part[3:])
|
2017-09-21 04:52:08 +02:00
|
|
|
if !out.Background.IsSet() {
|
|
|
|
return nil, fmt.Errorf("invalid background colour %q", part)
|
|
|
|
}
|
2017-06-06 13:23:50 +02:00
|
|
|
case strings.HasPrefix(part, "border:#"):
|
|
|
|
out.Border = ParseColour(part[7:])
|
2017-09-21 04:52:08 +02:00
|
|
|
if !out.Border.IsSet() {
|
|
|
|
return nil, fmt.Errorf("invalid border colour %q", part)
|
|
|
|
}
|
2017-06-06 13:23:50 +02:00
|
|
|
case strings.HasPrefix(part, "#"):
|
|
|
|
out.Colour = ParseColour(part)
|
2017-09-21 04:52:08 +02:00
|
|
|
if !out.Colour.IsSet() {
|
|
|
|
return nil, fmt.Errorf("invalid colour %q", part)
|
|
|
|
}
|
2017-06-06 13:23:50 +02:00
|
|
|
default:
|
2017-09-21 04:52:08 +02:00
|
|
|
return nil, fmt.Errorf("unknown style element %q", part)
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|
|
|
|
}
|
2017-09-21 04:52:08 +02:00
|
|
|
return out, nil
|
2017-06-06 13:23:50 +02:00
|
|
|
}
|