mirror of
https://github.com/issadarkthing/gomu.git
synced 2025-05-08 19:29:50 +08:00

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 ```
242 lines
4.8 KiB
Go
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
|
|
}
|