lf/ui.go

890 lines
20 KiB
Go
Raw Normal View History

2016-08-13 12:49:04 +00:00
package main
import (
"bytes"
"fmt"
"io"
2016-08-13 12:49:04 +00:00
"log"
"os"
"path/filepath"
"sort"
"strconv"
2016-08-13 12:49:04 +00:00
"strings"
"text/tabwriter"
"unicode"
"unicode/utf8"
2016-08-13 12:49:04 +00:00
2016-12-22 19:58:55 +00:00
"github.com/mattn/go-runewidth"
2016-08-13 12:49:04 +00:00
"github.com/nsf/termbox-go"
)
2017-11-22 14:17:23 +00:00
const (
gEscapeCode = 27
gAnsiColorResetMask = termbox.AttrBold | termbox.AttrUnderline | termbox.AttrReverse
)
2016-12-18 15:01:45 +00:00
var gAnsiCodes = map[int]termbox.Attribute{
2017-11-20 15:46:00 +00:00
0: termbox.ColorDefault,
2016-12-18 15:01:45 +00:00
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,
}
var gKeyVal = map[termbox.Key][]rune{
2016-12-18 18:51:27 +00:00
termbox.KeyF1: {'<', 'f', '-', '1', '>'},
termbox.KeyF2: {'<', 'f', '-', '2', '>'},
termbox.KeyF3: {'<', 'f', '-', '3', '>'},
termbox.KeyF4: {'<', 'f', '-', '4', '>'},
termbox.KeyF5: {'<', 'f', '-', '5', '>'},
termbox.KeyF6: {'<', 'f', '-', '6', '>'},
termbox.KeyF7: {'<', 'f', '-', '7', '>'},
termbox.KeyF8: {'<', 'f', '-', '8', '>'},
termbox.KeyF9: {'<', 'f', '-', '9', '>'},
termbox.KeyF10: {'<', 'f', '-', '1', '0', '>'},
termbox.KeyF11: {'<', 'f', '-', '1', '1', '>'},
termbox.KeyF12: {'<', 'f', '-', '1', '2', '>'},
termbox.KeyInsert: {'<', 'i', 'n', 's', 'e', 'r', 't', '>'},
termbox.KeyDelete: {'<', 'd', 'e', 'l', 'e', 't', 'e', '>'},
termbox.KeyHome: {'<', 'h', 'o', 'm', 'e', '>'},
termbox.KeyEnd: {'<', 'e', 'n', 'd', '>'},
termbox.KeyPgup: {'<', 'p', 'g', 'u', 'p', '>'},
termbox.KeyPgdn: {'<', 'p', 'g', 'd', 'n', '>'},
termbox.KeyArrowUp: {'<', 'u', 'p', '>'},
termbox.KeyArrowDown: {'<', 'd', 'o', 'w', 'n', '>'},
termbox.KeyArrowLeft: {'<', 'l', 'e', 'f', 't', '>'},
termbox.KeyArrowRight: {'<', 'r', 'i', 'g', 'h', 't', '>'},
termbox.KeyCtrlSpace: {'<', 'c', '-', 's', 'p', 'a', 'c', 'e', '>'},
termbox.KeyCtrlA: {'<', 'c', '-', 'a', '>'},
termbox.KeyCtrlB: {'<', 'c', '-', 'b', '>'},
termbox.KeyCtrlC: {'<', 'c', '-', 'c', '>'},
termbox.KeyCtrlD: {'<', 'c', '-', 'd', '>'},
termbox.KeyCtrlE: {'<', 'c', '-', 'e', '>'},
termbox.KeyCtrlF: {'<', 'c', '-', 'f', '>'},
termbox.KeyCtrlG: {'<', 'c', '-', 'g', '>'},
termbox.KeyBackspace: {'<', 'b', 's', '>'},
termbox.KeyTab: {'<', 't', 'a', 'b', '>'},
termbox.KeyCtrlJ: {'<', 'c', '-', 'j', '>'},
termbox.KeyCtrlK: {'<', 'c', '-', 'k', '>'},
termbox.KeyCtrlL: {'<', 'c', '-', 'l', '>'},
termbox.KeyEnter: {'<', 'e', 'n', 't', 'e', 'r', '>'},
termbox.KeyCtrlN: {'<', 'c', '-', 'n', '>'},
termbox.KeyCtrlO: {'<', 'c', '-', 'o', '>'},
termbox.KeyCtrlP: {'<', 'c', '-', 'p', '>'},
termbox.KeyCtrlQ: {'<', 'c', '-', 'q', '>'},
termbox.KeyCtrlR: {'<', 'c', '-', 'r', '>'},
termbox.KeyCtrlS: {'<', 'c', '-', 's', '>'},
termbox.KeyCtrlT: {'<', 'c', '-', 't', '>'},
termbox.KeyCtrlU: {'<', 'c', '-', 'u', '>'},
termbox.KeyCtrlV: {'<', 'c', '-', 'v', '>'},
termbox.KeyCtrlW: {'<', 'c', '-', 'w', '>'},
termbox.KeyCtrlX: {'<', 'c', '-', 'x', '>'},
termbox.KeyCtrlY: {'<', 'c', '-', 'y', '>'},
termbox.KeyCtrlZ: {'<', 'c', '-', 'z', '>'},
termbox.KeyEsc: {'<', 'e', 's', 'c', '>'},
termbox.KeyCtrlBackslash: {'<', 'c', '-', '\\', '>'},
termbox.KeyCtrlRsqBracket: {'<', 'c', '-', ']', '>'},
termbox.KeyCtrl6: {'<', 'c', '-', '6', '>'},
termbox.KeyCtrlSlash: {'<', 'c', '-', '/', '>'},
termbox.KeySpace: {'<', 's', 'p', 'a', 'c', 'e', '>'},
termbox.KeyBackspace2: {'<', 'b', 's', '2', '>'},
}
var gValKey map[string]termbox.Key
func init() {
gValKey = make(map[string]termbox.Key)
for k, v := range gKeyVal {
gValKey[string(v)] = k
}
}
2016-12-17 21:47:37 +00:00
type win struct {
2017-11-19 18:55:13 +00:00
w, h, x, y int
2016-08-13 12:49:04 +00:00
}
2016-12-17 21:47:37 +00:00
func newWin(w, h, x, y int) *win {
return &win{w, h, x, y}
2016-08-13 12:49:04 +00:00
}
2016-12-17 21:47:37 +00:00
func (win *win) renew(w, h, x, y int) {
2017-11-19 18:55:13 +00:00
win.w, win.h, win.x, win.y = w, h, x, y
2016-08-13 12:49:04 +00:00
}
2017-11-22 14:17:23 +00:00
func applyAnsiCodes(s string, fg, bg termbox.Attribute) (termbox.Attribute, termbox.Attribute) {
toks := strings.Split(s, ";")
var nums []int
for _, t := range toks {
if t == "" {
fg = termbox.ColorDefault
bg = termbox.ColorDefault
break
}
n, err := strconv.Atoi(t)
if err != nil {
log.Printf("converting escape code: %s", err)
continue
}
nums = append(nums, n)
}
// Parse 256 color terminal ansi codes
// termbox-go has a color offset of one, because of attributes
if len(nums) == 3 {
if nums[0] == 48 && nums[1] == 5 {
bg = termbox.Attribute(nums[2])
bg++
}
if nums[0] == 38 && nums[1] == 5 {
fg = termbox.Attribute(nums[2])
fg++
}
return fg, bg
}
2017-11-22 14:17:23 +00:00
for _, n := range nums {
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
}
func printLength(s string) int {
ind := 0
off := 0
for i := 0; i < len(s); i++ {
r, w := utf8.DecodeRuneInString(s[i:])
if r == gEscapeCode && i+1 < len(s) && s[i+1] == '[' {
j := strings.IndexByte(s[i:min(len(s), i+32)], 'm')
if j == -1 {
continue
}
i += j
continue
}
i += w - 1
if r == '\t' {
ind += gOpts.tabstop - (ind-off)%gOpts.tabstop
} else {
ind += runewidth.RuneWidth(r)
}
}
return ind
}
2017-11-22 14:17:23 +00:00
func (win *win) print(x, y int, fg, bg termbox.Attribute, s string) (termbox.Attribute, termbox.Attribute) {
2016-08-13 12:49:04 +00:00
off := x
for i := 0; i < len(s); i++ {
r, w := utf8.DecodeRuneInString(s[i:])
2016-12-17 21:47:37 +00:00
if r == gEscapeCode && i+1 < len(s) && s[i+1] == '[' {
j := strings.IndexByte(s[i:min(len(s), i+32)], 'm')
2016-11-29 14:00:40 +00:00
if j == -1 {
continue
}
2017-11-22 14:17:23 +00:00
fg, bg = applyAnsiCodes(s[i+2:i+j], fg, bg)
2016-11-29 14:00:40 +00:00
i += j
continue
}
2018-03-22 15:07:17 +00:00
if x < win.w {
termbox.SetCell(win.x+x, win.y+y, r, fg, bg)
2016-08-13 12:49:04 +00:00
}
i += w - 1
2016-08-13 12:49:04 +00:00
if r == '\t' {
2016-08-13 12:49:04 +00:00
x += gOpts.tabstop - (x-off)%gOpts.tabstop
} else {
2016-12-22 19:58:55 +00:00
x += runewidth.RuneWidth(r)
2016-08-13 12:49:04 +00:00
}
}
2017-11-22 14:17:23 +00:00
return fg, bg
2016-08-13 12:49:04 +00:00
}
2016-12-17 21:47:37 +00:00
func (win *win) printf(x, y int, fg, bg termbox.Attribute, format string, a ...interface{}) {
2016-08-13 12:49:04 +00:00
win.print(x, y, fg, bg, fmt.Sprintf(format, a...))
}
func (win *win) printLine(x, y int, fg, bg termbox.Attribute, s string) {
2016-08-13 12:49:04 +00:00
win.printf(x, y, fg, bg, "%s%*s", s, win.w-len(s), "")
}
func (win *win) printRight(y int, fg, bg termbox.Attribute, s string) {
win.print(win.w-len(s), y, fg, bg, s)
}
func (win *win) printReg(reg *reg) {
if reg == nil {
return
}
2017-11-19 18:55:13 +00:00
fg, bg := termbox.ColorDefault, termbox.ColorDefault
for i, l := range reg.lines {
2017-11-22 14:17:23 +00:00
fg, bg = win.print(2, i, fg, bg, l)
2017-11-19 18:55:13 +00:00
}
return
}
func (win *win) printDir(dir *dir, marks map[string]int, saves map[string]bool) {
2016-08-13 12:49:04 +00:00
if win.w < 3 {
return
}
2017-11-19 18:55:13 +00:00
if dir == nil {
return
}
2016-08-13 12:49:04 +00:00
fg, bg := termbox.ColorDefault, termbox.ColorDefault
if dir.loading {
fg = termbox.AttrBold
win.print(2, 0, fg, bg, "loading...")
return
}
2016-08-13 12:49:04 +00:00
if len(dir.fi) == 0 {
fg = termbox.AttrBold
2016-12-18 15:01:45 +00:00
win.print(2, 0, fg, bg, "empty")
2016-08-13 12:49:04 +00:00
return
}
maxind := len(dir.fi) - 1
beg := max(dir.ind-dir.pos, 0)
end := min(beg+win.h, maxind+1)
for i, f := range dir.fi[beg:end] {
switch {
2017-11-19 18:55:13 +00:00
case f.linkState == working:
2016-12-18 15:01:45 +00:00
fg = termbox.ColorCyan
if f.Mode().IsDir() {
fg |= termbox.AttrBold
}
2017-11-19 18:55:13 +00:00
case f.linkState == broken:
2016-12-18 15:01:45 +00:00
fg = termbox.ColorMagenta
2016-08-13 12:49:04 +00:00
case f.Mode().IsRegular():
if f.Mode()&0111 != 0 {
fg = termbox.AttrBold | termbox.ColorGreen
} else {
fg = termbox.ColorDefault
}
case f.Mode().IsDir():
fg = termbox.AttrBold | termbox.ColorBlue
case f.Mode()&os.ModeNamedPipe != 0:
fg = termbox.ColorRed
case f.Mode()&os.ModeSocket != 0:
fg = termbox.ColorYellow
case f.Mode()&os.ModeDevice != 0:
fg = termbox.ColorWhite
}
path := filepath.Join(dir.path, f.Name())
2016-08-13 12:49:04 +00:00
if _, ok := marks[path]; ok {
2016-08-13 12:49:04 +00:00
win.print(0, i, fg, termbox.ColorMagenta, " ")
} else if copy, ok := saves[path]; ok {
if copy {
win.print(0, i, fg, termbox.ColorYellow, " ")
} else {
win.print(0, i, fg, termbox.ColorRed, " ")
}
2016-08-13 12:49:04 +00:00
}
if i == dir.pos {
2016-12-18 18:51:27 +00:00
fg |= termbox.AttrReverse
2016-08-13 12:49:04 +00:00
}
var s []rune
2016-08-13 12:49:04 +00:00
s = append(s, ' ')
for _, r := range f.Name() {
s = append(s, r)
}
2016-08-13 12:49:04 +00:00
w := runeSliceWidth(s)
if w > win.w-2 {
s = runeSliceWidthRange(s, 0, win.w-2)
2016-08-13 12:49:04 +00:00
} else {
2016-12-22 19:58:55 +00:00
for i := 0; i < win.w-2-w; i++ {
s = append(s, ' ')
}
2016-08-13 12:49:04 +00:00
}
var info string
for _, s := range gOpts.info {
switch s {
case "size":
2017-06-03 11:12:43 +00:00
if !(gOpts.dircounts && f.IsDir()) {
info = fmt.Sprintf("%s %4s", info, humanize(f.Size()))
continue
}
2017-11-19 18:55:13 +00:00
if f.count == -1 {
2017-06-03 11:12:43 +00:00
d, err := os.Open(path)
if err != nil {
f.count = -2
2017-06-03 11:12:43 +00:00
}
names, err := d.Readdirnames(1000)
2017-06-03 11:12:43 +00:00
d.Close()
if names == nil && err != io.EOF {
f.count = -2
} else {
f.count = len(names)
2017-06-03 11:12:43 +00:00
}
}
2017-06-03 11:12:43 +00:00
switch {
case f.count < 0:
info = fmt.Sprintf("%s ?", info)
case f.count < 1000:
info = fmt.Sprintf("%s %4d", info, f.count)
default:
info = fmt.Sprintf("%s 999+", info)
2017-06-03 11:12:43 +00:00
}
case "time":
info = fmt.Sprintf("%s %12s", info, f.ModTime().Format("Jan _2 15:04"))
default:
log.Printf("unknown info type: %s", s)
2016-08-13 12:49:04 +00:00
}
}
if len(info) > 0 && win.w > 2*len(info) {
s = runeSliceWidthRange(s, 0, win.w-2-len(info))
for _, r := range info {
s = append(s, r)
2016-08-13 12:49:04 +00:00
}
}
// TODO: add a trailing '~' to the name if cut
win.print(1, i, fg, bg, string(s))
}
}
2016-12-17 21:47:37 +00:00
type ui struct {
2017-11-19 18:55:13 +00:00
wins []*win
promptWin *win
2017-11-19 18:55:13 +00:00
msgWin *win
menuWin *win
msg string
regPrev *reg
2017-11-19 18:55:13 +00:00
dirPrev *dir
keyChan chan string
evChan chan termbox.Event
menuBuf *bytes.Buffer
cmdPrefix string
cmdAccLeft []rune
cmdAccRight []rune
cmdBuf []rune
keyAcc []rune
keyCount []rune
2016-08-13 12:49:04 +00:00
}
func getWidths(wtot int) []int {
rsum := 0
for _, rat := range gOpts.ratios {
rsum += rat
}
wlen := len(gOpts.ratios)
widths := make([]int, wlen)
wsum := 0
for i := 0; i < wlen-1; i++ {
widths[i] = gOpts.ratios[i] * (wtot / rsum)
wsum += widths[i]
}
widths[wlen-1] = wtot - wsum
return widths
}
2016-12-18 19:38:28 +00:00
func getWins() []*win {
2016-08-13 12:49:04 +00:00
wtot, htot := termbox.Size()
2016-12-17 21:47:37 +00:00
var wins []*win
2016-08-13 12:49:04 +00:00
widths := getWidths(wtot)
wacc := 0
wlen := len(widths)
for i := 0; i < wlen; i++ {
wins = append(wins, newWin(widths[i], htot-2, wacc, 1))
wacc += widths[i]
}
2016-12-18 19:38:28 +00:00
return wins
}
func newUI() *ui {
wtot, htot := termbox.Size()
2017-11-19 18:55:13 +00:00
evChan := make(chan termbox.Event)
go func() {
for {
2017-11-19 18:55:13 +00:00
evChan <- termbox.PollEvent()
}
}()
2016-12-17 21:47:37 +00:00
return &ui{
wins: getWins(),
promptWin: newWin(wtot, 1, 0, 0),
msgWin: newWin(wtot, 1, 0, htot-1),
menuWin: newWin(wtot, 1, 0, htot-2),
keyChan: make(chan string, 1000),
evChan: evChan,
2016-08-13 12:49:04 +00:00
}
}
2016-12-17 21:47:37 +00:00
func (ui *ui) renew() {
2016-08-13 12:49:04 +00:00
wtot, htot := termbox.Size()
widths := getWidths(wtot)
wacc := 0
wlen := len(widths)
for i := 0; i < wlen; i++ {
ui.wins[i].renew(widths[i], htot-2, wacc, 1)
wacc += widths[i]
}
2017-11-19 18:55:13 +00:00
ui.msgWin.renew(wtot, 1, 0, htot-1)
2016-08-13 12:49:04 +00:00
}
2017-11-19 18:55:13 +00:00
func (ui *ui) print(msg string) {
ui.msg = msg
log.Print(msg)
}
2016-08-13 12:49:04 +00:00
2017-11-19 18:55:13 +00:00
func (ui *ui) printf(format string, a ...interface{}) {
ui.print(fmt.Sprintf(format, a...))
}
type reg struct {
path string
lines []string
}
func (ui *ui) loadFile(nav *nav) {
curr, err := nav.currFile()
if err != nil {
return
}
if !gOpts.preview {
return
}
if curr.IsDir() {
ui.dirPrev = nav.loadDir(curr.path)
} else if curr.Mode().IsRegular() {
ui.regPrev = nav.loadReg(ui, curr.path)
}
2016-08-13 12:49:04 +00:00
}
2017-11-19 18:55:13 +00:00
func (ui *ui) loadFileInfo(nav *nav) {
curr, err := nav.currFile()
if err != nil {
return
}
ui.msg = fmt.Sprintf("%v %4s %v", curr.Mode(), humanize(curr.Size()), curr.ModTime().Format(gOpts.timefmt))
}
func (ui *ui) drawPromptLine(nav *nav) {
fg, bg := termbox.ColorDefault, termbox.ColorDefault
dir := nav.currDir()
pwd := strings.Replace(dir.path, gUser.HomeDir, "~", -1)
pwd = filepath.Clean(pwd)
var fname string
curr, err := nav.currFile()
if err == nil {
fname = filepath.Base(curr.path)
}
var prompt string
prompt = strings.Replace(gOpts.promptfmt, "%u", gUser.Username, -1)
prompt = strings.Replace(prompt, "%h", gHostname, -1)
prompt = strings.Replace(prompt, "%f", fname, -1)
if printLength(strings.Replace(prompt, "%w", pwd, -1)) > ui.promptWin.w {
sep := string(filepath.Separator)
names := strings.Split(pwd, sep)
2018-03-22 14:54:24 +00:00
for i := range names {
r, _ := utf8.DecodeRuneInString(names[i])
names[i] = string(r)
if printLength(strings.Replace(prompt, "%w", strings.Join(names, sep), -1)) <= ui.promptWin.w {
break
}
}
pwd = strings.Join(names, sep)
}
prompt = strings.Replace(prompt, "%w", pwd, -1)
ui.promptWin.print(0, 0, fg, bg, prompt)
}
func (ui *ui) drawStatLine(nav *nav) {
fg, bg := termbox.ColorDefault, termbox.ColorDefault
currDir := nav.currDir()
ui.msgWin.print(0, 0, fg, bg, ui.msg)
tot := len(currDir.fi)
ind := min(currDir.ind+1, tot)
ruler := fmt.Sprintf("%d/%d", ind, tot)
ui.msgWin.printRight(0, fg, bg, ruler)
}
2016-12-17 21:47:37 +00:00
func (ui *ui) draw(nav *nav) {
2016-08-13 12:49:04 +00:00
fg, bg := termbox.ColorDefault, termbox.ColorDefault
termbox.Clear(fg, bg)
ui.drawPromptLine(nav)
2016-08-13 12:49:04 +00:00
length := min(len(ui.wins), len(nav.dirs))
woff := len(ui.wins) - length
2016-08-13 12:49:04 +00:00
if gOpts.preview {
length = min(len(ui.wins)-1, len(nav.dirs))
woff = len(ui.wins) - 1 - length
}
doff := len(nav.dirs) - length
2016-08-13 12:49:04 +00:00
for i := 0; i < length; i++ {
2017-11-19 18:55:13 +00:00
ui.wins[woff+i].printDir(nav.dirs[doff+i], nav.marks, nav.saves)
2016-08-13 12:49:04 +00:00
}
2017-11-19 18:55:13 +00:00
if ui.cmdPrefix != "" {
ui.msgWin.printLine(0, 0, fg, bg, ui.cmdPrefix)
2017-11-19 18:55:13 +00:00
ui.msgWin.print(len(ui.cmdPrefix), 0, fg, bg, string(ui.cmdAccLeft))
ui.msgWin.print(len(ui.cmdPrefix)+runeSliceWidth(ui.cmdAccLeft), 0, fg, bg, string(ui.cmdAccRight))
termbox.SetCursor(ui.msgWin.x+len(ui.cmdPrefix)+runeSliceWidth(ui.cmdAccLeft), ui.msgWin.y)
} else {
ui.drawStatLine(nav)
termbox.HideCursor()
}
2016-08-13 12:49:04 +00:00
if gOpts.preview {
f, err := nav.currFile()
if err == nil {
preview := ui.wins[len(ui.wins)-1]
if f.IsDir() {
2017-11-19 18:55:13 +00:00
preview.printDir(ui.dirPrev, nav.marks, nav.saves)
} else if f.Mode().IsRegular() {
2017-11-19 18:55:13 +00:00
preview.printReg(ui.regPrev)
}
2016-08-13 12:49:04 +00:00
}
}
2017-11-19 18:55:13 +00:00
if ui.menuBuf != nil {
lines := strings.Split(ui.menuBuf.String(), "\n")
lines = lines[:len(lines)-1]
2017-11-19 18:55:13 +00:00
ui.menuWin.h = len(lines) - 1
ui.menuWin.y = ui.wins[0].h - ui.menuWin.h
ui.menuWin.printLine(0, 0, termbox.AttrBold, termbox.AttrBold, lines[0])
for i, line := range lines[1:] {
ui.menuWin.printLine(0, i+1, fg, bg, "")
2017-11-19 18:55:13 +00:00
ui.menuWin.print(0, i+1, fg, bg, line)
}
}
termbox.Flush()
2017-11-19 18:55:13 +00:00
if ui.cmdPrefix == "" {
// leave the cursor at the beginning of the current file for screen readers
2017-01-07 15:02:14 +00:00
moveCursor(ui.wins[woff+length-1].y+nav.dirs[doff+length-1].pos+1, ui.wins[woff+length-1].x+1)
}
2016-08-13 12:49:04 +00:00
}
2016-12-17 21:47:37 +00:00
func findBinds(keys map[string]expr, prefix string) (binds map[string]expr, ok bool) {
binds = make(map[string]expr)
2016-08-13 12:49:04 +00:00
for key, expr := range keys {
if strings.HasPrefix(key, prefix) {
binds[key] = expr
if key == prefix {
ok = true
}
}
}
return
}
2016-12-17 21:47:37 +00:00
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
}
2016-12-17 21:47:37 +00:00
func (ui *ui) pollEvent() termbox.Event {
select {
2017-11-19 18:55:13 +00:00
case key := <-ui.keyChan:
ev := termbox.Event{Type: termbox.EventKey}
if len(key) == 1 {
ev.Ch, _ = utf8.DecodeRuneInString(key)
} else {
switch key {
case "<lt>":
ev.Ch = '<'
case "<gt>":
ev.Ch = '>'
default:
if val, ok := gValKey[key]; ok {
ev.Key = val
} else {
ev.Key = termbox.KeyEsc
2017-11-19 18:55:13 +00:00
ui.printf("unknown key: %s", key)
}
}
}
return ev
2017-11-19 18:55:13 +00:00
case ev := <-ui.evChan:
return ev
}
}
2016-12-17 21:47:37 +00:00
type multiExpr struct {
expr expr
2016-10-29 23:20:35 +00:00
count int
}
2016-12-18 15:01:45 +00:00
func readCmdEvent(ch chan<- multiExpr, ev termbox.Event) {
if ev.Ch != 0 {
ch <- multiExpr{&callExpr{"cmd-insert", []string{string(ev.Ch)}}, 1}
} else {
val := gKeyVal[ev.Key]
if expr, ok := gOpts.cmdkeys[string(val)]; ok {
ch <- multiExpr{expr, 1}
2016-12-18 15:01:45 +00:00
}
}
}
2016-08-13 12:49:04 +00:00
2016-12-18 15:01:45 +00:00
func (ui *ui) readEvent(ch chan<- multiExpr, ev termbox.Event) {
redraw := &callExpr{"redraw", nil}
count := 1
2016-08-13 12:49:04 +00:00
2016-12-18 15:01:45 +00:00
switch ev.Type {
case termbox.EventKey:
if ev.Ch != 0 {
switch {
case ev.Ch == '<':
2017-11-19 18:55:13 +00:00
ui.keyAcc = append(ui.keyAcc, '<', 'l', 't', '>')
2016-12-18 15:01:45 +00:00
case ev.Ch == '>':
2017-11-19 18:55:13 +00:00
ui.keyAcc = append(ui.keyAcc, '<', 'g', 't', '>')
case unicode.IsDigit(ev.Ch) && len(ui.keyAcc) == 0:
ui.keyCount = append(ui.keyCount, ev.Ch)
2016-12-18 15:01:45 +00:00
default:
2017-11-19 18:55:13 +00:00
ui.keyAcc = append(ui.keyAcc, ev.Ch)
2016-12-18 15:01:45 +00:00
}
} else {
val := gKeyVal[ev.Key]
if string(val) == "<esc>" {
ch <- multiExpr{redraw, 1}
2017-11-19 18:55:13 +00:00
ui.keyAcc = nil
ui.keyCount = nil
}
2017-11-19 18:55:13 +00:00
ui.keyAcc = append(ui.keyAcc, val...)
2016-12-18 15:01:45 +00:00
}
2017-11-19 18:55:13 +00:00
binds, ok := findBinds(gOpts.keys, string(ui.keyAcc))
2016-12-18 15:01:45 +00:00
switch len(binds) {
case 0:
2017-11-19 18:55:13 +00:00
ui.printf("unknown mapping: %s", string(ui.keyAcc))
ch <- multiExpr{redraw, 1}
2017-11-19 18:55:13 +00:00
ui.keyAcc = nil
ui.keyCount = nil
2018-01-11 16:44:02 +00:00
ui.menuBuf = nil
2016-12-18 15:01:45 +00:00
case 1:
if ok {
2017-11-19 18:55:13 +00:00
if len(ui.keyCount) > 0 {
c, err := strconv.Atoi(string(ui.keyCount))
2016-12-18 15:01:45 +00:00
if err != nil {
log.Printf("converting command count: %s", err)
2016-09-07 19:34:29 +00:00
}
2016-12-18 15:01:45 +00:00
count = c
} else {
2016-12-18 15:01:45 +00:00
count = 1
}
2017-11-19 18:55:13 +00:00
expr := gOpts.keys[string(ui.keyAcc)]
2016-12-18 15:01:45 +00:00
ch <- multiExpr{expr, count}
2017-11-19 18:55:13 +00:00
ui.keyAcc = nil
ui.keyCount = nil
2016-12-18 15:01:45 +00:00
}
2017-11-19 18:55:13 +00:00
if len(ui.keyAcc) > 0 {
ui.menuBuf = listBinds(binds)
ch <- multiExpr{redraw, 1}
2017-11-19 18:55:13 +00:00
} else if ui.menuBuf != nil {
ui.menuBuf = nil
2016-12-18 15:01:45 +00:00
}
default:
if ok {
// TODO: use a delay
2017-11-19 18:55:13 +00:00
if len(ui.keyCount) > 0 {
c, err := strconv.Atoi(string(ui.keyCount))
2016-12-18 15:01:45 +00:00
if err != nil {
log.Printf("converting command count: %s", err)
2016-09-07 19:34:29 +00:00
}
2016-12-18 15:01:45 +00:00
count = c
} else {
count = 1
2016-08-13 12:49:04 +00:00
}
2017-11-19 18:55:13 +00:00
expr := gOpts.keys[string(ui.keyAcc)]
2016-12-18 15:01:45 +00:00
ch <- multiExpr{expr, count}
2017-11-19 18:55:13 +00:00
ui.keyAcc = nil
ui.keyCount = nil
2016-12-18 15:01:45 +00:00
}
2017-11-19 18:55:13 +00:00
if len(ui.keyAcc) > 0 {
ui.menuBuf = listBinds(binds)
ch <- multiExpr{redraw, 1}
2016-12-18 15:01:45 +00:00
} else {
2017-11-19 18:55:13 +00:00
ui.menuBuf = nil
2016-12-18 15:01:45 +00:00
}
}
case termbox.EventResize:
ch <- multiExpr{redraw, 1}
2016-12-18 15:01:45 +00:00
default:
// TODO: handle other events
}
}
// This function is used to read expressions on the client side. 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 (ui *ui) readExpr() <-chan multiExpr {
ch := make(chan multiExpr)
go func() {
2017-11-19 18:55:13 +00:00
ch <- multiExpr{&callExpr{"redraw", nil}, 1}
2016-12-18 15:01:45 +00:00
for {
ev := ui.pollEvent()
2017-11-19 18:55:13 +00:00
if ui.cmdPrefix != "" && ev.Type == termbox.EventKey {
2016-12-18 15:01:45 +00:00
readCmdEvent(ch, ev)
continue
2016-08-13 12:49:04 +00:00
}
2016-12-18 15:01:45 +00:00
ui.readEvent(ch, ev)
2016-08-13 12:49:04 +00:00
}
}()
return ch
2016-08-13 12:49:04 +00:00
}
2016-12-17 21:47:37 +00:00
func (ui *ui) pause() {
2016-08-13 12:49:04 +00:00
termbox.Close()
}
2016-12-17 21:47:37 +00:00
func (ui *ui) resume() {
2016-08-13 12:49:04 +00:00
if err := termbox.Init(); err != nil {
2016-08-17 20:22:11 +00:00
log.Fatalf("initializing termbox: %s", err)
2016-08-13 12:49:04 +00:00
}
}
2016-12-17 21:47:37 +00:00
func (ui *ui) sync() {
2016-08-17 20:28:42 +00:00
if err := termbox.Sync(); err != nil {
2016-08-17 20:22:11 +00:00
log.Printf("syncing termbox: %s", err)
2016-08-13 12:49:04 +00:00
}
}
func listMatches(matches []string) *bytes.Buffer {
b := new(bytes.Buffer)
2016-08-13 12:49:04 +00:00
wtot, _ := termbox.Size()
2016-08-13 12:49:04 +00:00
wcol := 0
for _, m := range matches {
wcol = max(wcol, len(m))
2016-08-13 12:49:04 +00:00
}
wcol += gOpts.tabstop - wcol%gOpts.tabstop
2016-08-13 12:49:04 +00:00
ncol := wtot / wcol
b.WriteString("possible matches\n")
for i := 0; i < len(matches); i++ {
for j := 0; j < ncol && i < len(matches); i, j = i+1, j+1 {
b.WriteString(fmt.Sprintf("%s%*s", matches[i], wcol-len(matches[i]), ""))
}
b.WriteByte('\n')
}
return b
2016-08-13 12:49:04 +00:00
}