242 lines
5.2 KiB
Go
242 lines
5.2 KiB
Go
package main
|
|
|
|
import (
|
|
"log"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/nsf/termbox-go"
|
|
)
|
|
|
|
const gAnsiColorResetMask = termbox.AttrBold | termbox.AttrUnderline | termbox.AttrReverse
|
|
|
|
var gAnsiCodes = map[int]termbox.Attribute{
|
|
0: termbox.ColorDefault,
|
|
1: termbox.AttrBold,
|
|
4: termbox.AttrUnderline,
|
|
7: termbox.AttrReverse,
|
|
30: termbox.ColorBlack,
|
|
31: termbox.ColorRed,
|
|
32: termbox.ColorGreen,
|
|
33: termbox.ColorYellow,
|
|
34: termbox.ColorBlue,
|
|
35: termbox.ColorMagenta,
|
|
36: termbox.ColorCyan,
|
|
37: termbox.ColorWhite,
|
|
40: termbox.ColorBlack,
|
|
41: termbox.ColorRed,
|
|
42: termbox.ColorGreen,
|
|
43: termbox.ColorYellow,
|
|
44: termbox.ColorBlue,
|
|
45: termbox.ColorMagenta,
|
|
46: termbox.ColorCyan,
|
|
47: termbox.ColorWhite,
|
|
}
|
|
|
|
type colorEntry struct {
|
|
fg termbox.Attribute
|
|
bg termbox.Attribute
|
|
}
|
|
|
|
type colorMap map[string]colorEntry
|
|
|
|
func parseColors() colorMap {
|
|
if env := os.Getenv("LS_COLORS"); env != "" {
|
|
return parseColorsGNU(env)
|
|
}
|
|
|
|
if env := os.Getenv("LSCOLORS"); env != "" {
|
|
return parseColorsBSD(env)
|
|
}
|
|
|
|
// default values from dircolors
|
|
defaultColors := []string{
|
|
// "rs=0",
|
|
"di=01;34",
|
|
"ln=01;36",
|
|
// "mh=00",
|
|
"pi=40;33",
|
|
"so=01;35",
|
|
"do=01;35",
|
|
"bd=40;33;01",
|
|
"cd=40;33;01",
|
|
"or=40;31;01",
|
|
// "mi=00",
|
|
"su=37;41",
|
|
"sg=30;43",
|
|
"ca=30;41",
|
|
"tw=30;42",
|
|
"ow=34;42",
|
|
"st=37;44",
|
|
"ex=01;32",
|
|
}
|
|
|
|
return parseColorsGNU(strings.Join(defaultColors, ":"))
|
|
}
|
|
|
|
func applyAnsiCodes(s string, fg, bg termbox.Attribute) (termbox.Attribute, termbox.Attribute) {
|
|
toks := strings.Split(s, ";")
|
|
|
|
var nums []int
|
|
for _, tok := range toks {
|
|
if tok == "" {
|
|
nums = append(nums, 0)
|
|
continue
|
|
}
|
|
n, err := strconv.Atoi(tok)
|
|
if err != nil {
|
|
log.Printf("converting escape code: %s", err)
|
|
continue
|
|
}
|
|
nums = append(nums, n)
|
|
}
|
|
|
|
for i := 0; i < len(nums); i++ {
|
|
n := nums[i]
|
|
switch {
|
|
case n == 38 && i+2 < len(nums) && nums[i+1] == 5:
|
|
fg &= gAnsiColorResetMask
|
|
fg |= termbox.Attribute(nums[i+2] + 1)
|
|
i += 2
|
|
continue
|
|
case n == 48 && i+2 < len(nums) && nums[i+1] == 5:
|
|
bg = termbox.Attribute(nums[i+2] + 1)
|
|
i += 2
|
|
continue
|
|
}
|
|
attr, ok := gAnsiCodes[n]
|
|
if !ok {
|
|
log.Printf("unknown ansi code: %d", n)
|
|
continue
|
|
}
|
|
switch {
|
|
case n == 0:
|
|
fg, bg = attr, attr
|
|
case n == 1 || n == 4 || n == 7:
|
|
fg |= attr
|
|
case 30 <= n && n <= 37:
|
|
fg &= gAnsiColorResetMask
|
|
fg |= attr
|
|
case 40 <= n && n <= 47:
|
|
bg = attr
|
|
}
|
|
}
|
|
|
|
return fg, bg
|
|
}
|
|
|
|
// This function parses $LS_COLORS environment variable.
|
|
func parseColorsGNU(env string) colorMap {
|
|
colors := make(colorMap)
|
|
|
|
entries := strings.Split(env, ":")
|
|
for _, entry := range entries {
|
|
if entry == "" {
|
|
continue
|
|
}
|
|
pair := strings.Split(entry, "=")
|
|
if len(pair) != 2 {
|
|
log.Printf("invalid $LS_COLORS entry: %s", entry)
|
|
return colors
|
|
}
|
|
key, val := pair[0], pair[1]
|
|
fg, bg := applyAnsiCodes(val, termbox.ColorDefault, termbox.ColorDefault)
|
|
colors[key] = colorEntry{fg: fg, bg: bg}
|
|
}
|
|
|
|
return colors
|
|
}
|
|
|
|
// This function parses $LSCOLORS environment variable.
|
|
func parseColorsBSD(env string) colorMap {
|
|
colors := make(colorMap)
|
|
|
|
if len(env) != 22 {
|
|
log.Printf("invalid $LSCOLORS variable: %s", env)
|
|
return colors
|
|
}
|
|
|
|
colorNames := []string{"di", "ln", "so", "pi", "ex", "bd", "cd", "su", "sg", "tw", "ow"}
|
|
colorCodes := map[byte]termbox.Attribute{
|
|
'a': termbox.ColorBlack,
|
|
'b': termbox.ColorRed,
|
|
'c': termbox.ColorGreen,
|
|
'd': termbox.ColorYellow, // brown
|
|
'e': termbox.ColorBlue,
|
|
'f': termbox.ColorMagenta,
|
|
'g': termbox.ColorCyan,
|
|
'h': termbox.ColorWhite, // light grey
|
|
'A': termbox.AttrBold | termbox.ColorBlack,
|
|
'B': termbox.AttrBold | termbox.ColorRed,
|
|
'C': termbox.AttrBold | termbox.ColorGreen,
|
|
'D': termbox.AttrBold | termbox.ColorYellow, // brown
|
|
'E': termbox.AttrBold | termbox.ColorBlue,
|
|
'F': termbox.AttrBold | termbox.ColorMagenta,
|
|
'G': termbox.AttrBold | termbox.ColorCyan,
|
|
'H': termbox.AttrBold | termbox.ColorWhite, // light grey
|
|
'x': termbox.ColorDefault,
|
|
}
|
|
|
|
getColor := func(r byte) termbox.Attribute {
|
|
if color, ok := colorCodes[r]; ok {
|
|
return color
|
|
}
|
|
|
|
log.Printf("invalid $LSCOLORS entry: %c", r)
|
|
return termbox.ColorDefault
|
|
}
|
|
|
|
for i, key := range colorNames {
|
|
colors[key] = colorEntry{fg: getColor(env[i*2]), bg: getColor(env[i*2+1])}
|
|
}
|
|
|
|
return colors
|
|
}
|
|
|
|
func (cm colorMap) get(f *file) (termbox.Attribute, termbox.Attribute) {
|
|
var key string
|
|
|
|
switch {
|
|
case f.IsDir() && f.Mode()&os.ModeSticky != 0 && f.Mode()&0002 != 0:
|
|
key = "tw"
|
|
case f.IsDir() && f.Mode()&os.ModeSticky != 0:
|
|
key = "st"
|
|
case f.IsDir() && f.Mode()&0002 != 0:
|
|
key = "ow"
|
|
case f.IsDir():
|
|
key = "di"
|
|
case f.linkState == working:
|
|
key = "ln"
|
|
case f.linkState == broken:
|
|
key = "or"
|
|
case f.Mode()&os.ModeNamedPipe != 0:
|
|
key = "pi"
|
|
case f.Mode()&os.ModeSocket != 0:
|
|
key = "so"
|
|
case f.Mode()&os.ModeCharDevice != 0:
|
|
key = "cd"
|
|
case f.Mode()&os.ModeDevice != 0:
|
|
key = "bd"
|
|
case f.Mode()&os.ModeSetuid != 0:
|
|
key = "su"
|
|
case f.Mode()&os.ModeSetgid != 0:
|
|
key = "sg"
|
|
case f.Mode().IsRegular() && f.Mode()&0111 != 0:
|
|
key = "ex"
|
|
default:
|
|
key = "*" + filepath.Ext(f.Name())
|
|
}
|
|
|
|
if val, ok := cm[key]; ok {
|
|
return val.fg, val.bg
|
|
}
|
|
|
|
if val, ok := cm["fi"]; ok {
|
|
return val.fg, val.bg
|
|
}
|
|
|
|
return termbox.ColorDefault, termbox.ColorDefault
|
|
}
|