Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for specifying both foreground and background colors. #49

Merged
merged 11 commits into from
Nov 15, 2019
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
*.swp
*~
mjarkk marked this conversation as resolved.
Show resolved Hide resolved
78 changes: 49 additions & 29 deletions escape.go
Original file line number Diff line number Diff line change
Expand Up @@ -191,39 +191,59 @@ func (ei *escapeInterpreter) output256() error {
return ei.outputNormal()
}

fgbg, err := strconv.Atoi(ei.csiParam[0])
if err != nil {
return errCSIParseError
}
color, err := strconv.Atoi(ei.csiParam[2])
if err != nil {
return errCSIParseError
}

switch fgbg {
case 38:
ei.curFgColor = Attribute(color + 1)
for _, param := range ei.splitFgBg() {
fgbg, err := strconv.Atoi(param[0])
if err != nil {
return errCSIParseError
}
color, err := strconv.Atoi(param[2])
if err != nil {
return errCSIParseError
}

for _, param := range ei.csiParam[3:] {
p, err := strconv.Atoi(param)
if err != nil {
return errCSIParseError
switch fgbg {
case 38:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we change this magic number stuff while we are at it

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I pushed a change that I think is what you want.

ei.curFgColor = Attribute(color + 1)

for _, s := range param[3:] {
p, err := strconv.Atoi(s)
if err != nil {
return errCSIParseError
}

switch {
case p == 1:
ei.curFgColor |= AttrBold
case p == 4:
ei.curFgColor |= AttrUnderline
case p == 7:
ei.curFgColor |= AttrReverse

}
}
case 48:
ei.curBgColor = Attribute(color + 1)
default:
return errCSIParseError
}
}
return nil
}

switch {
case p == 1:
ei.curFgColor |= AttrBold
case p == 4:
ei.curFgColor |= AttrUnderline
case p == 7:
ei.curFgColor |= AttrReverse
}
func (ei *escapeInterpreter) splitFgBg() [][]string {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

does this function need to be a method of ei, cant this just be a function we can call like

func splitFgBg() [][]string { ... return out}

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

spltFfBg is very specific to the escapeInterpreter, so to me it makes sense to keep it as is.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I get that, but it does not rely on ei, it just relies on data. It is not driven by the struct

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not a big deal to me. I'll change it. Just a sec.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

var out [][]string
var current []string
for _, p := range ei.csiParam {
if len(current) > 0 && (p == "48" || p == "38") {
out = append(out, current)
current = []string{}
}
case 48:
ei.curBgColor = Attribute(color + 1)
default:
return errCSIParseError
current = append(current, p)
}

return nil
if len(current) > 0 {
out = append(out, current)
}

return out
}
61 changes: 61 additions & 0 deletions escape_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
package gocui
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we don't want any tests in this library at this point in time.
Maybe in the further but for now this file is not needed an can be removed.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok, done


import (
"fmt"
"testing"
)

func TestEscape(t *testing.T) {
testCases := []struct {
input string
fg Attribute
bg Attribute
}{
{
input: "\033[48;5;200;38;5;100mHi!!",
fg: 101,
bg: 201,
},
{
input: "\033[38;5;100;48;5;200mHi!!",
fg: 101,
bg: 201,
},
{
input: "\033[38;5;100mHi!!",
fg: 101,
bg: ColorDefault,
},
{
input: "\033[48;5;100mHi!!",
fg: ColorDefault,
bg: 101,
},
{
input: "Hi!!",
fg: ColorDefault,
bg: ColorDefault,
},
}

for i, tc := range testCases {
t.Run(fmt.Sprintf("%02d", i), func(t *testing.T) {
ei := newEscapeInterpreter(Output256)

for _, r := range tc.input {
_, err := ei.parseOne(r)
if err != nil {
t.Fatal(err)
}
}

if ei.curFgColor != tc.fg {
t.Fatalf("foreground color is not %d: %v", tc.fg, ei.curFgColor)
}
if ei.curBgColor != tc.bg {
t.Fatalf("background color is not %d: %v", tc.bg, ei.curBgColor)
}
})
}

}