gomu/anko/anko.go
augustus 51d55bb102 Fix panic on Ctrl+Arrow
I often hit fn+arrow up / down to page up / down. If I accidentally
hit ctrl gomu panics.

Investigated it today and the issue is the rune extraction -- it assumes
we'll always have Ctrl+<rune> but in the case of an arrow key we have
Ctrl+Up or Ctrl+Down.

Essentially this simply checks if the regex found any matches, if not we
return not result akin to a failed map lookup. I considered raising an
error, but that would be just as annoying as a panic when I fat finger
Ctrl :)

```golang
panic: runtime error: index out of range [0] with length 0 [recovered]
	panic: runtime error: index out of range [0] with length 0

goroutine 1 [running]:
github.com/rivo/tview.(*Application).Run.func1()
	/home/augustus/go/pkg/mod/github.com/rivo/tview@v0.0.0-20210312174852-ae9464cc3598/application.go:243 +0x4d
panic({0xadb840, 0xc00076a000})
	/usr/local/go/src/runtime/panic.go:1038 +0x215
github.com/issadarkthing/gomu/anko.extractCtrlRune({0xc00003cc90, 0x9})
	/home/augustus/go/src/github.com/issadarkthing/gomu/anko/anko.go:216 +0x75
github.com/issadarkthing/gomu/anko.(*Anko).KeybindExists(0xb23243, {0xb2591b, 0x6}, 0xc00007e040)
	/home/augustus/go/src/github.com/issadarkthing/gomu/anko/anko.go:158 +0x78
main.start.func4(0xc00007e040)
	/home/augustus/go/src/github.com/issadarkthing/gomu/start.go:516 +0x14b
github.com/rivo/tview.(*Application).Run(0xc00023e2a0)
	/home/augustus/go/pkg/mod/github.com/rivo/tview@v0.0.0-20210312174852-ae9464cc3598/application.go:318 +0x6d2
main.start(0xc00023e2a0, {0xc00028e840, 0xc00003d4b6, 0xc00028e850, 0xc00003d4b7})
	/home/augustus/go/src/github.com/issadarkthing/gomu/start.go:546 +0xceb
main.main()
	/home/augustus/go/src/github.com/issadarkthing/gomu/main.go:21 +0x10e
```
2022-07-02 17:07:55 +01:00

242 lines
4.8 KiB
Go

package anko
import (
"fmt"
"regexp"
"strings"
"github.com/gdamore/tcell/v2"
"github.com/mattn/anko/core"
"github.com/mattn/anko/env"
_ "github.com/mattn/anko/packages"
"github.com/mattn/anko/parser"
"github.com/mattn/anko/vm"
)
type Anko struct {
env *env.Env
}
func NewAnko() *Anko {
env := core.Import(env.NewEnv())
importToX(env)
t, err := env.Get("typeOf")
if err != nil {
panic(err)
}
k, err := env.Get("kindOf")
if err != nil {
panic(err)
}
env.DeleteGlobal("typeOf")
env.DeleteGlobal("kindOf")
err = env.Define("type_of", t)
if err != nil {
panic(err)
}
err = env.Define("kind_of", k)
if err != nil {
panic(err)
}
return &Anko{env}
}
// DefineGlobal defines new symbol and value to the Anko env.
func (a *Anko) DefineGlobal(symbol string, value interface{}) error {
return a.env.DefineGlobal(symbol, value)
}
func (a *Anko) NewModule(name string) (*Anko, error) {
env, err := a.env.NewModule(name)
if err != nil {
return nil, err
}
return &Anko{env}, nil
}
func (a *Anko) Define(name string, value interface{}) error {
return a.env.Define(name, value)
}
// Set sets new value to existing symbol. Use this when change value under an
// existing symbol.
func (a *Anko) Set(symbol string, value interface{}) error {
return a.env.Set(symbol, value)
}
// Get gets value from anko env, returns error if symbol is not found.
func (a *Anko) Get(symbol string) (interface{}, error) {
return a.env.Get(symbol)
}
// GetInt gets int value from symbol, returns golang default value if not found.
func (a *Anko) GetInt(symbol string) int {
v, err := a.Execute(symbol)
if err != nil {
return 0
}
switch val := v.(type) {
case int:
return val
case int64:
return int(val)
}
return 0
}
// GetString gets string value from symbol, returns golang default value if not
// found.
func (a *Anko) GetString(symbol string) string {
v, err := a.Execute(symbol)
if err != nil {
return ""
}
val, ok := v.(string)
if !ok {
return ""
}
return val
}
// GetBool gets bool value from symbol, returns golang default value if not
// found.
func (a *Anko) GetBool(symbol string) bool {
v, err := a.Execute(symbol)
if err != nil {
return false
}
val, ok := v.(bool)
if !ok {
return false
}
return val
}
// Execute executes anko script.
func (a *Anko) Execute(src string) (interface{}, error) {
parser.EnableErrorVerbose()
stmts, err := parser.ParseSrc(src)
if err != nil {
return nil, err
}
val, err := vm.Run(a.env, nil, stmts)
if err != nil {
if e, ok := err.(*vm.Error); ok {
err = fmt.Errorf("error on line %d column %d: %s\n",
e.Pos.Line, e.Pos.Column, err)
} else if e, ok := err.(*parser.Error); ok {
err = fmt.Errorf("error on line %d column %d: %s\n",
e.Pos.Line, e.Pos.Column, err)
}
return nil, err
}
return val, nil
}
// KeybindExists checks if keybinding is defined.
func (a *Anko) KeybindExists(panel string, eventKey *tcell.EventKey) bool {
var src string
name := eventKey.Name()
if strings.Contains(name, "Ctrl") {
key, ok := extractCtrlRune(name)
if !ok {
return false
}
src = fmt.Sprintf("Keybinds.%s[\"ctrl_%s\"]",
panel, strings.ToLower(string(key)))
} else if strings.Contains(name, "Alt") {
key, ok := extractAltRune(name)
if !ok {
return false
}
src = fmt.Sprintf("Keybinds.%s[\"alt_%c\"]", panel, key)
} else if strings.Contains(name, "Rune") {
src = fmt.Sprintf("Keybinds.%s[\"%c\"]", panel, eventKey.Rune())
} else {
src = fmt.Sprintf("Keybinds.%s[\"%s\"]", panel, strings.ToLower(name))
}
val, err := a.Execute(src)
if err != nil {
return false
}
return val != nil
}
// ExecKeybind executes function bounded by the keybinding.
func (a *Anko) ExecKeybind(panel string, eventKey *tcell.EventKey) error {
var src string
name := eventKey.Name()
if strings.Contains(name, "Ctrl") {
key, ok := extractCtrlRune(name)
if !ok {
return nil
}
src = fmt.Sprintf("Keybinds.%s[\"ctrl_%s\"]()",
panel, strings.ToLower(string(key)))
} else if strings.Contains(name, "Alt") {
key, ok := extractAltRune(name)
if !ok {
return nil
}
src = fmt.Sprintf("Keybinds.%s[\"alt_%c\"]()", panel, key)
} else if strings.Contains(name, "Rune") {
src = fmt.Sprintf("Keybinds.%s[\"%c\"]()", panel, eventKey.Rune())
} else {
src = fmt.Sprintf("Keybinds.%s[\"%s\"]()", panel, strings.ToLower(name))
}
_, err := a.Execute(src)
if err != nil {
return err
}
return nil
}
func extractCtrlRune(str string) (rune, bool) {
re := regexp.MustCompile(`\+(.)$`)
x := re.FindStringSubmatch(str)
if len(x) == 0 {
return rune(' '), false
}
return rune(x[0][1]), true
}
func extractAltRune(str string) (rune, bool) {
re := regexp.MustCompile(`\[(.)\]`)
x := re.FindStringSubmatch(str)
if len(x) == 0 {
return rune(' '), false
}
return rune(x[0][1]), true
}