implement asynchronous read commands

This commit changes previous reading command implementation to an
asynchronous implementation. By the nature of this change, this commit
touches many places in the ui and evaluator. Aim is to fix the following
problems:

- There is no race condition anymore when reading commands and other
commands update the ui at the same time.

- Autocompletion and keymenu is now drawn in the main draw event. This
should fix some ui glitches when a new menu is smaller than the previous
one.

- Window resize event when reading a command is now properly handled.

- Readline actions are now regular commands. This should make it
possible to change the default keybindings for these actions in the
future.

Mentioned in #36.
This commit is contained in:
Gokcehan 2016-12-15 12:26:06 +03:00
parent f66a4a4a2e
commit 5f87cb2542
3 changed files with 343 additions and 341 deletions

140
app.go
View File

@ -8,9 +8,6 @@ import (
"os/exec" "os/exec"
"strconv" "strconv"
"strings" "strings"
"unicode"
"github.com/nsf/termbox-go"
) )
type App struct { type App struct {
@ -55,144 +52,11 @@ func waitKey() error {
return nil return nil
} }
// This function is used to read expressions on the client side. Prompting
// commands (e.g. "read") are recognized and evaluated while being read here.
// Digits are interpreted as command counts but this is only done for digits
// preceding any non-digit characters (e.g. "42y2k" as 42 times "y2k").
func (app *App) readExpr() chan MultiExpr {
ch := make(chan MultiExpr)
renew := &CallExpr{"renew", nil}
count := 1
var acc []rune
var cnt []rune
go func() {
for {
switch ev := app.ui.pollEvent(); ev.Type {
case termbox.EventKey:
if ev.Ch != 0 {
switch {
case ev.Ch == '<':
acc = append(acc, '<', 'l', 't', '>')
case ev.Ch == '>':
acc = append(acc, '<', 'g', 't', '>')
case unicode.IsDigit(ev.Ch) && len(acc) == 0:
cnt = append(cnt, ev.Ch)
default:
acc = append(acc, ev.Ch)
}
} else {
val := gKeyVal[ev.Key]
if string(val) == "<esc>" {
ch <- MultiExpr{renew, 1}
acc = nil
cnt = nil
}
acc = append(acc, val...)
}
binds, ok := findBinds(gOpts.keys, string(acc))
switch len(binds) {
case 0:
app.ui.message = fmt.Sprintf("unknown mapping: %s", string(acc))
ch <- MultiExpr{renew, 1}
acc = nil
cnt = nil
case 1:
if ok {
if len(cnt) > 0 {
c, err := strconv.Atoi(string(cnt))
if err != nil {
log.Printf("converting command count: %s", err)
}
count = c
} else {
count = 1
}
expr := gOpts.keys[string(acc)]
switch expr.(type) {
case *CallExpr:
switch expr.(*CallExpr).name {
case "read",
"read-shell",
"read-shell-wait",
"read-shell-async",
"search",
"search-back",
"push":
expr.eval(app, nil)
app.ui.loadFile(app.nav)
app.ui.draw(app.nav)
default:
ch <- MultiExpr{expr, count}
}
default:
ch <- MultiExpr{expr, count}
}
acc = nil
cnt = nil
}
if len(acc) > 0 {
app.ui.listBinds(binds)
}
default:
if ok {
// TODO: use a delay
if len(cnt) > 0 {
c, err := strconv.Atoi(string(cnt))
if err != nil {
log.Printf("converting command count: %s", err)
}
count = c
} else {
count = 1
}
expr := gOpts.keys[string(acc)]
switch expr.(type) {
case *CallExpr:
switch expr.(*CallExpr).name {
case "read",
"read-shell",
"read-shell-wait",
"read-shell-async",
"search",
"search-back",
"push":
expr.eval(app, nil)
app.ui.loadFile(app.nav)
app.ui.draw(app.nav)
default:
ch <- MultiExpr{expr, count}
}
default:
ch <- MultiExpr{expr, count}
}
acc = nil
cnt = nil
}
if len(acc) > 0 {
app.ui.listBinds(binds)
}
}
case termbox.EventResize:
ch <- MultiExpr{renew, 1}
default:
// TODO: handle other events
}
}
}()
return ch
}
// This is the main event loop of the application. There are two channels to // This is the main event loop of the application. There are two channels to
// read expressions from client and server. Reading and evaluation are done on // read expressions from client and server. Reading and evaluation are done on
// different goroutines except for prompting commands (e.g. "read"). // separate goroutines.
func (app *App) handleInp() { func (app *App) handleInp() {
clientChan := app.readExpr() clientChan := app.ui.readExpr()
var serverChan chan Expr var serverChan chan Expr

166
eval.go
View File

@ -242,56 +242,17 @@ func (e *CallExpr) eval(app *App, args []string) {
app.ui.loadFile(app.nav) app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav) app.ui.loadFileInfo(app.nav)
case "read": case "read":
s := app.ui.prompt(app.nav, ":") app.ui.cmdpref = ":"
if len(s) == 0 {
return
}
log.Printf("command: %s", s)
p := newParser(strings.NewReader(s))
for p.parse() {
p.expr.eval(app, nil)
}
if p.err != nil {
app.ui.message = p.err.Error()
log.Print(p.err)
}
case "read-shell": case "read-shell":
s := app.ui.prompt(app.nav, "$") app.ui.cmdpref = "$"
if len(s) == 0 {
return
}
log.Printf("shell: %s", s)
app.runShell(s, nil, false, false)
case "read-shell-wait": case "read-shell-wait":
s := app.ui.prompt(app.nav, "!") app.ui.cmdpref = "!"
if len(s) == 0 {
return
}
log.Printf("shell-wait: %s", s)
app.runShell(s, nil, true, false)
case "read-shell-async": case "read-shell-async":
s := app.ui.prompt(app.nav, "&") app.ui.cmdpref = "&"
if len(s) == 0 {
return
}
log.Printf("shell-async: %s", s)
app.runShell(s, nil, false, true)
case "search": case "search":
s := app.ui.prompt(app.nav, "/") app.ui.cmdpref = "/"
if len(s) == 0 {
return
}
log.Printf("search: %s", s)
app.ui.message = "sorry, search is not implemented yet!"
// TODO: implement
case "search-back": case "search-back":
s := app.ui.prompt(app.nav, "?") app.ui.cmdpref = "?"
if len(s) == 0 {
return
}
log.Printf("search-back: %s", s)
app.ui.message = "sorry, search-back is not implemented yet!"
// TODO: implement
case "toggle": case "toggle":
app.nav.toggle() app.nav.toggle()
case "invert": case "invert":
@ -359,7 +320,118 @@ func (e *CallExpr) eval(app *App, args []string) {
app.ui.loadFileInfo(app.nav) app.ui.loadFileInfo(app.nav)
case "push": case "push":
if len(e.args) > 0 { if len(e.args) > 0 {
app.ui.keysbuf = append(app.ui.keysbuf, splitKeys(e.args[0])...) log.Println("pushing keys", e.args[0])
for _, key := range splitKeys(e.args[0]) {
app.ui.keychan <- key
}
}
case "cmd-insert":
if len(e.args) > 0 {
app.ui.cmdlacc = append(app.ui.cmdlacc, []rune(e.args[0])...)
}
case "cmd-escape":
app.ui.menubuf = nil
app.ui.cmdbuf = nil
app.ui.cmdlacc = nil
app.ui.cmdracc = nil
app.ui.cmdpref = ""
case "cmd-comp":
var matches []string
if app.ui.cmdpref == ":" {
matches, app.ui.cmdlacc = compCmd(app.ui.cmdlacc)
} else {
matches, app.ui.cmdlacc = compShell(app.ui.cmdlacc)
}
app.ui.draw(app.nav)
if len(matches) > 1 {
app.ui.menubuf = listMatches(matches)
} else {
app.ui.menubuf = nil
}
case "cmd-enter":
s := string(append(app.ui.cmdlacc, app.ui.cmdracc...))
if len(s) == 0 {
return
}
app.ui.menubuf = nil
app.ui.cmdbuf = nil
app.ui.cmdlacc = nil
app.ui.cmdracc = nil
switch app.ui.cmdpref {
case ":":
log.Printf("command: %s", s)
p := newParser(strings.NewReader(s))
for p.parse() {
p.expr.eval(app, nil)
}
if p.err != nil {
app.ui.message = p.err.Error()
log.Print(p.err)
}
case "$":
log.Printf("shell: %s", s)
app.runShell(s, nil, false, false)
case "!":
log.Printf("shell-wait: %s", s)
app.runShell(s, nil, true, false)
case "&":
log.Printf("shell-async: %s", s)
app.runShell(s, nil, false, true)
case "/":
log.Printf("search: %s", s)
app.ui.message = "sorry, search is not implemented yet!"
// TODO: implement
case "?":
log.Printf("search-back: %s", s)
app.ui.message = "sorry, search-back is not implemented yet!"
// TODO: implement
default:
log.Printf("entering unknown execution prefix: %q", app.ui.cmdpref)
}
app.ui.cmdpref = ""
case "cmd-delete-back":
if len(app.ui.cmdlacc) > 0 {
app.ui.cmdlacc = app.ui.cmdlacc[:len(app.ui.cmdlacc)-1]
}
case "cmd-delete":
if len(app.ui.cmdracc) > 0 {
app.ui.cmdracc = app.ui.cmdracc[1:]
}
case "cmd-left":
if len(app.ui.cmdlacc) > 0 {
app.ui.cmdracc = append([]rune{app.ui.cmdlacc[len(app.ui.cmdlacc)-1]}, app.ui.cmdracc...)
app.ui.cmdlacc = app.ui.cmdlacc[:len(app.ui.cmdlacc)-1]
}
case "cmd-right":
if len(app.ui.cmdracc) > 0 {
app.ui.cmdlacc = append(app.ui.cmdlacc, app.ui.cmdracc[0])
app.ui.cmdracc = app.ui.cmdracc[1:]
}
case "cmd-beg":
app.ui.cmdracc = append(app.ui.cmdlacc, app.ui.cmdracc...)
app.ui.cmdlacc = nil
case "cmd-end":
app.ui.cmdlacc = append(app.ui.cmdlacc, app.ui.cmdracc...)
app.ui.cmdracc = nil
case "cmd-delete-end":
if len(app.ui.cmdracc) > 0 {
app.ui.cmdbuf = app.ui.cmdracc
app.ui.cmdracc = nil
}
case "cmd-delete-beg":
if len(app.ui.cmdlacc) > 0 {
app.ui.cmdbuf = app.ui.cmdlacc
app.ui.cmdlacc = nil
}
case "cmd-delete-word":
ind := strings.LastIndex(strings.TrimRight(string(app.ui.cmdlacc), " "), " ") + 1
app.ui.cmdbuf = app.ui.cmdlacc[ind:]
app.ui.cmdlacc = app.ui.cmdlacc[:ind]
case "cmd-put":
app.ui.cmdlacc = append(app.ui.cmdlacc, app.ui.cmdbuf...)
case "cmd-transpose":
if len(app.ui.cmdlacc) > 1 {
app.ui.cmdlacc[len(app.ui.cmdlacc)-1], app.ui.cmdlacc[len(app.ui.cmdlacc)-2] = app.ui.cmdlacc[len(app.ui.cmdlacc)-2], app.ui.cmdlacc[len(app.ui.cmdlacc)-1]
} }
default: default:
cmd, ok := gOpts.cmds[e.name] cmd, ok := gOpts.cmds[e.name]
@ -391,7 +463,7 @@ func (e *ExecExpr) eval(app *App, args []string) {
log.Printf("search-back: %s -- %s", e, args) log.Printf("search-back: %s -- %s", e, args)
// TODO: implement // TODO: implement
default: default:
log.Printf("unknown execution prefix: %q", e.pref) log.Printf("evaluating unknown execution prefix: %q", e.pref)
} }
} }

334
ui.go
View File

@ -13,6 +13,7 @@ import (
"strconv" "strconv"
"strings" "strings"
"text/tabwriter" "text/tabwriter"
"unicode"
"unicode/utf8" "unicode/utf8"
"github.com/nsf/termbox-go" "github.com/nsf/termbox-go"
@ -340,7 +341,13 @@ type UI struct {
message string message string
regprev []string regprev []string
dirprev *Dir dirprev *Dir
keysbuf []string keychan chan string
evschan chan termbox.Event
cmdpref string
cmdlacc []rune
cmdracc []rune
cmdbuf []rune
menubuf *bytes.Buffer
} }
func getWidths(wtot int) []int { func getWidths(wtot int) []int {
@ -376,11 +383,22 @@ func newUI() *UI {
wacc += widths[i] wacc += widths[i]
} }
key := make(chan string, 1000)
evs := make(chan termbox.Event)
go func() {
for {
evs <- termbox.PollEvent()
}
}()
return &UI{ return &UI{
wins: wins, wins: wins,
pwdwin: newWin(wtot, 1, 0, 0), pwdwin: newWin(wtot, 1, 0, 0),
msgwin: newWin(wtot, 1, 0, htot-1), msgwin: newWin(wtot, 1, 0, htot-1),
menuwin: newWin(wtot, 1, 0, htot-2), menuwin: newWin(wtot, 1, 0, htot-2),
keychan: key,
evschan: evs,
} }
} }
@ -478,26 +496,11 @@ func (ui *UI) loadFile(nav *Nav) {
} }
} }
func (ui *UI) clearMsg() {
fg, bg := termbox.ColorDefault, termbox.ColorDefault
win := ui.msgwin
win.printl(0, 0, fg, bg, "")
termbox.Flush()
}
func (ui *UI) draw(nav *Nav) { func (ui *UI) draw(nav *Nav) {
fg, bg := termbox.ColorDefault, termbox.ColorDefault fg, bg := termbox.ColorDefault, termbox.ColorDefault
termbox.Clear(fg, bg) termbox.Clear(fg, bg)
// leave the cursor at the beginning of the current file for screen readers
var length, woff, doff int
defer func() {
fmt.Printf("[%d;%dH", ui.wins[woff+length-1].y+nav.dirs[doff+length-1].pos+1, ui.wins[woff+length-1].x+1)
}()
defer termbox.Flush()
dir := nav.currDir() dir := nav.currDir()
path := strings.Replace(dir.path, envHome, "~", -1) path := strings.Replace(dir.path, envHome, "~", -1)
@ -507,20 +510,28 @@ func (ui *UI) draw(nav *Nav) {
ui.pwdwin.printf(len(envUser)+len(envHost)+1, 0, fg, bg, ":") ui.pwdwin.printf(len(envUser)+len(envHost)+1, 0, fg, bg, ":")
ui.pwdwin.printf(len(envUser)+len(envHost)+2, 0, termbox.AttrBold|termbox.ColorBlue, bg, "%s", path) ui.pwdwin.printf(len(envUser)+len(envHost)+2, 0, termbox.AttrBold|termbox.ColorBlue, bg, "%s", path)
length = min(len(ui.wins), len(nav.dirs)) length := min(len(ui.wins), len(nav.dirs))
woff = len(ui.wins) - length woff := len(ui.wins) - length
if gOpts.preview { if gOpts.preview {
length = min(len(ui.wins)-1, len(nav.dirs)) length = min(len(ui.wins)-1, len(nav.dirs))
woff = len(ui.wins) - 1 - length woff = len(ui.wins) - 1 - length
} }
doff = len(nav.dirs) - length doff := len(nav.dirs) - length
for i := 0; i < length; i++ { for i := 0; i < length; i++ {
ui.wins[woff+i].printd(nav.dirs[doff+i], nav.marks, nav.saves) ui.wins[woff+i].printd(nav.dirs[doff+i], nav.marks, nav.saves)
} }
defer ui.msgwin.print(0, 0, fg, bg, ui.message) if ui.cmdpref != "" {
ui.msgwin.printl(0, 0, fg, bg, ui.cmdpref)
ui.msgwin.print(len(ui.cmdpref), 0, fg, bg, string(ui.cmdlacc))
ui.msgwin.print(len(ui.cmdpref)+runeSliceWidth(ui.cmdlacc), 0, fg, bg, string(ui.cmdracc))
termbox.SetCursor(ui.msgwin.x+len(ui.cmdpref)+runeSliceWidth(ui.cmdlacc), ui.msgwin.y)
} else {
ui.msgwin.print(0, 0, fg, bg, ui.message)
termbox.HideCursor()
}
if gOpts.preview { if gOpts.preview {
f, err := nav.currFile() f, err := nav.currFile()
@ -536,6 +547,28 @@ func (ui *UI) draw(nav *Nav) {
preview.printr(ui.regprev) preview.printr(ui.regprev)
} }
} }
if ui.menubuf != nil {
lines := strings.Split(ui.menubuf.String(), "\n")
lines = lines[:len(lines)-1]
ui.menuwin.h = len(lines) - 1
ui.menuwin.y = ui.wins[0].h - ui.menuwin.h
ui.menuwin.printl(0, 0, termbox.AttrBold, termbox.AttrBold, lines[0])
for i, line := range lines[1:] {
ui.menuwin.printl(0, i+1, termbox.ColorDefault, termbox.ColorDefault, "")
ui.menuwin.print(0, i+1, termbox.ColorDefault, termbox.ColorDefault, line)
}
}
termbox.Flush()
if ui.cmdpref == "" {
// leave the cursor at the beginning of the current file for screen readers
fmt.Printf("[%d;%dH", ui.wins[woff+length-1].y+nav.dirs[doff+length-1].pos+1, ui.wins[woff+length-1].x+1)
}
} }
func findBinds(keys map[string]Expr, prefix string) (binds map[string]Expr, ok bool) { func findBinds(keys map[string]Expr, prefix string) (binds map[string]Expr, ok bool) {
@ -551,33 +584,55 @@ func findBinds(keys map[string]Expr, prefix string) (binds map[string]Expr, ok b
return return
} }
func listBinds(binds map[string]Expr) *bytes.Buffer {
t := new(tabwriter.Writer)
b := new(bytes.Buffer)
var keys []string
for k := range binds {
keys = append(keys, k)
}
sort.Strings(keys)
t.Init(b, 0, gOpts.tabstop, 2, '\t', 0)
fmt.Fprintln(t, "keys\tcommand")
for _, k := range keys {
fmt.Fprintf(t, "%s\t%v\n", k, binds[k])
}
t.Flush()
return b
}
func (ui *UI) pollEvent() termbox.Event { func (ui *UI) pollEvent() termbox.Event {
if len(ui.keysbuf) > 0 { select {
case key := <-ui.keychan:
ev := termbox.Event{Type: termbox.EventKey} ev := termbox.Event{Type: termbox.EventKey}
keys := ui.keysbuf[0]
if len(keys) == 1 { if len(key) == 1 {
ev.Ch, _ = utf8.DecodeRuneInString(keys) ev.Ch, _ = utf8.DecodeRuneInString(key)
} else { } else {
switch keys { switch key {
case "<lt>": case "<lt>":
ev.Ch = '<' ev.Ch = '<'
case "<gt>": case "<gt>":
ev.Ch = '>' ev.Ch = '>'
default: default:
if val, ok := gValKey[keys]; ok { if val, ok := gValKey[key]; ok {
ev.Key = val ev.Key = val
} else { } else {
ev.Key = termbox.KeyEsc ev.Key = termbox.KeyEsc
msg := fmt.Sprintf("unknown key: %s", keys) msg := fmt.Sprintf("unknown key: %s", key)
ui.message = msg ui.message = msg
log.Print(msg) log.Print(msg)
} }
} }
} }
ui.keysbuf = ui.keysbuf[1:]
return ev
case ev := <-ui.evschan:
return ev return ev
} }
return termbox.PollEvent()
} }
type MultiExpr struct { type MultiExpr struct {
@ -585,104 +640,152 @@ type MultiExpr struct {
count int count int
} }
func (ui *UI) prompt(nav *Nav, pref string) string { // This function is used to read expressions on the client side. Digits are
fg, bg := termbox.ColorDefault, termbox.ColorDefault // interpreted as command counts but this is only done for digits preceding any
// non-digit characters (e.g. "42y2k" as 42 times "y2k").
func (ui *UI) readExpr() chan MultiExpr {
ch := make(chan MultiExpr)
win := ui.msgwin renew := &CallExpr{"renew", nil}
count := 1
win.printl(0, 0, fg, bg, pref) var acc []rune
termbox.SetCursor(win.x+len(pref), win.y) var cnt []rune
defer termbox.HideCursor()
termbox.Flush()
var lacc []rune
var racc []rune
var buf []rune
go func() {
for { for {
switch ev := ui.pollEvent(); ev.Type { ev := ui.pollEvent()
if ui.cmdpref != "" {
switch ev.Type {
case termbox.EventKey: case termbox.EventKey:
if ev.Ch != 0 { if ev.Ch != 0 {
lacc = append(lacc, ev.Ch) ch <- MultiExpr{&CallExpr{"cmd-insert", []string{string(ev.Ch)}}, 1}
} else { } else {
// TODO: rest of the keys // TODO: rest of the keys
switch ev.Key { switch ev.Key {
case termbox.KeyEsc: case termbox.KeyEsc:
return "" ch <- MultiExpr{&CallExpr{"cmd-escape", nil}, 1}
case termbox.KeySpace: case termbox.KeySpace:
lacc = append(lacc, ' ') ch <- MultiExpr{&CallExpr{"cmd-insert", []string{" "}}, 1}
case termbox.KeyTab: case termbox.KeyTab:
var matches []string ch <- MultiExpr{&CallExpr{"cmd-comp", nil}, 1}
if pref == ":" {
matches, lacc = compCmd(lacc)
} else {
matches, lacc = compShell(lacc)
}
ui.draw(nav)
if len(matches) > 1 {
ui.listMatches(matches)
}
case termbox.KeyEnter, termbox.KeyCtrlJ: case termbox.KeyEnter, termbox.KeyCtrlJ:
win.printl(0, 0, fg, bg, "") ch <- MultiExpr{&CallExpr{"cmd-enter", nil}, 1}
termbox.Flush()
return string(append(lacc, racc...))
case termbox.KeyBackspace, termbox.KeyBackspace2: case termbox.KeyBackspace, termbox.KeyBackspace2:
if len(lacc) > 0 { ch <- MultiExpr{&CallExpr{"cmd-delete-back", nil}, 1}
lacc = lacc[:len(lacc)-1]
}
case termbox.KeyDelete, termbox.KeyCtrlD: case termbox.KeyDelete, termbox.KeyCtrlD:
if len(racc) > 0 { ch <- MultiExpr{&CallExpr{"cmd-delete", nil}, 1}
racc = racc[1:]
}
case termbox.KeyArrowLeft, termbox.KeyCtrlB: case termbox.KeyArrowLeft, termbox.KeyCtrlB:
if len(lacc) > 0 { ch <- MultiExpr{&CallExpr{"cmd-left", nil}, 1}
racc = append([]rune{lacc[len(lacc)-1]}, racc...)
lacc = lacc[:len(lacc)-1]
}
case termbox.KeyArrowRight, termbox.KeyCtrlF: case termbox.KeyArrowRight, termbox.KeyCtrlF:
if len(racc) > 0 { ch <- MultiExpr{&CallExpr{"cmd-right", nil}, 1}
lacc = append(lacc, racc[0])
racc = racc[1:]
}
case termbox.KeyHome, termbox.KeyCtrlA: case termbox.KeyHome, termbox.KeyCtrlA:
racc = append(lacc, racc...) ch <- MultiExpr{&CallExpr{"cmd-beg", nil}, 1}
lacc = nil
case termbox.KeyEnd, termbox.KeyCtrlE: case termbox.KeyEnd, termbox.KeyCtrlE:
lacc = append(lacc, racc...) ch <- MultiExpr{&CallExpr{"cmd-end", nil}, 1}
racc = nil
case termbox.KeyCtrlK: case termbox.KeyCtrlK:
if len(racc) > 0 { ch <- MultiExpr{&CallExpr{"cmd-delete-end", nil}, 1}
buf = racc
racc = nil
}
case termbox.KeyCtrlU: case termbox.KeyCtrlU:
if len(lacc) > 0 { ch <- MultiExpr{&CallExpr{"cmd-delete-beg", nil}, 1}
buf = lacc
lacc = nil
}
case termbox.KeyCtrlW: case termbox.KeyCtrlW:
ind := strings.LastIndex(strings.TrimRight(string(lacc), " "), " ") + 1 ch <- MultiExpr{&CallExpr{"cmd-delete-word", nil}, 1}
buf = lacc[ind:]
lacc = lacc[:ind]
case termbox.KeyCtrlY: case termbox.KeyCtrlY:
lacc = append(lacc, buf...) ch <- MultiExpr{&CallExpr{"cmd-put", nil}, 1}
case termbox.KeyCtrlT: case termbox.KeyCtrlT:
if len(lacc) > 1 { ch <- MultiExpr{&CallExpr{"cmd-transpose", nil}, 1}
lacc[len(lacc)-1], lacc[len(lacc)-2] = lacc[len(lacc)-2], lacc[len(lacc)-1]
} }
} }
continue
}
} }
win.printl(0, 0, fg, bg, pref) switch ev.Type {
win.print(len(pref), 0, fg, bg, string(lacc)) case termbox.EventKey:
win.print(len(pref)+runeSliceWidth(lacc), 0, fg, bg, string(racc)) if ev.Ch != 0 {
termbox.SetCursor(win.x+len(pref)+runeSliceWidth(lacc), win.y) switch {
termbox.Flush() case ev.Ch == '<':
acc = append(acc, '<', 'l', 't', '>')
case ev.Ch == '>':
acc = append(acc, '<', 'g', 't', '>')
case unicode.IsDigit(ev.Ch) && len(acc) == 0:
cnt = append(cnt, ev.Ch)
default:
acc = append(acc, ev.Ch)
}
} else {
val := gKeyVal[ev.Key]
if string(val) == "<esc>" {
ch <- MultiExpr{renew, 1}
acc = nil
cnt = nil
}
acc = append(acc, val...)
}
binds, ok := findBinds(gOpts.keys, string(acc))
switch len(binds) {
case 0:
ui.message = fmt.Sprintf("unknown mapping: %s", string(acc))
ch <- MultiExpr{renew, 1}
acc = nil
cnt = nil
case 1:
if ok {
if len(cnt) > 0 {
c, err := strconv.Atoi(string(cnt))
if err != nil {
log.Printf("converting command count: %s", err)
}
count = c
} else {
count = 1
}
expr := gOpts.keys[string(acc)]
ch <- MultiExpr{expr, count}
acc = nil
cnt = nil
}
if len(acc) > 0 {
ui.menubuf = listBinds(binds)
ch <- MultiExpr{renew, 1}
} else if ui.menubuf != nil {
ui.menubuf = nil
}
default:
if ok {
// TODO: use a delay
if len(cnt) > 0 {
c, err := strconv.Atoi(string(cnt))
if err != nil {
log.Printf("converting command count: %s", err)
}
count = c
} else {
count = 1
}
expr := gOpts.keys[string(acc)]
ch <- MultiExpr{expr, count}
acc = nil
cnt = nil
}
if len(acc) > 0 {
ui.menubuf = listBinds(binds)
ch <- MultiExpr{renew, 1}
} else {
ui.menubuf = nil
}
}
case termbox.EventResize:
ch <- MultiExpr{renew, 1}
default: default:
// TODO: handle other events // TODO: handle other events
} }
} }
}()
return ch
} }
func (ui *UI) pause() { func (ui *UI) pause() {
@ -701,44 +804,7 @@ func (ui *UI) sync() {
} }
} }
func (ui *UI) showMenu(b *bytes.Buffer) { func listMatches(matches []string) *bytes.Buffer {
lines := strings.Split(b.String(), "\n")
lines = lines[:len(lines)-1]
ui.menuwin.h = len(lines) - 1
ui.menuwin.y = ui.wins[0].h - ui.menuwin.h
ui.menuwin.printl(0, 0, termbox.AttrBold, termbox.AttrBold, lines[0])
for i, line := range lines[1:] {
ui.menuwin.printl(0, i+1, termbox.ColorDefault, termbox.ColorDefault, "")
ui.menuwin.print(0, i+1, termbox.ColorDefault, termbox.ColorDefault, line)
}
termbox.Flush()
}
func (ui *UI) listBinds(binds map[string]Expr) {
t := new(tabwriter.Writer)
b := new(bytes.Buffer)
var keys []string
for k := range binds {
keys = append(keys, k)
}
sort.Strings(keys)
t.Init(b, 0, gOpts.tabstop, 2, '\t', 0)
fmt.Fprintln(t, "keys\tcommand")
for _, k := range keys {
fmt.Fprintf(t, "%s\t%v\n", k, binds[k])
}
t.Flush()
ui.showMenu(b)
}
func (ui *UI) listMatches(matches []string) {
b := new(bytes.Buffer) b := new(bytes.Buffer)
wtot, _ := termbox.Size() wtot, _ := termbox.Size()
@ -759,5 +825,5 @@ func (ui *UI) listMatches(matches []string) {
b.WriteByte('\n') b.WriteByte('\n')
} }
ui.showMenu(b) return b
} }