2016-08-13 12:49:04 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
2016-09-01 19:03:18 +00:00
|
|
|
"io"
|
2016-08-13 12:49:04 +00:00
|
|
|
"log"
|
|
|
|
"os"
|
2016-08-28 12:04:57 +00:00
|
|
|
"os/exec"
|
2016-09-06 20:05:18 +00:00
|
|
|
"path/filepath"
|
2016-08-24 10:08:49 +00:00
|
|
|
"sort"
|
2016-08-28 00:45:05 +00:00
|
|
|
"strconv"
|
2016-08-13 12:49:04 +00:00
|
|
|
"strings"
|
|
|
|
"text/tabwriter"
|
2016-12-15 09:26:06 +00:00
|
|
|
"unicode"
|
2016-08-28 00:45:05 +00:00
|
|
|
"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"
|
|
|
|
)
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
const gEscapeCode = 27
|
2016-08-28 00:45:05 +00:00
|
|
|
|
2016-12-18 15:01:45 +00:00
|
|
|
var gAnsiCodes = map[int]termbox.Attribute{
|
|
|
|
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,
|
|
|
|
}
|
|
|
|
|
2016-09-18 16:21:24 +00:00
|
|
|
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', '>'},
|
2016-09-18 16:21:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2016-08-13 12:49:04 +00:00
|
|
|
w int
|
|
|
|
h int
|
|
|
|
x int
|
|
|
|
y int
|
|
|
|
}
|
|
|
|
|
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) {
|
2016-08-13 12:49:04 +00:00
|
|
|
win.w = w
|
|
|
|
win.h = h
|
|
|
|
win.x = x
|
|
|
|
win.y = y
|
|
|
|
}
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
func (win *win) print(x, y int, fg, bg termbox.Attribute, s string) {
|
2016-08-13 12:49:04 +00:00
|
|
|
off := x
|
2016-08-28 00:45:05 +00:00
|
|
|
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] == '[' {
|
2016-12-17 12:27:27 +00:00
|
|
|
j := strings.IndexByte(s[i:min(len(s), i+32)], 'm')
|
2016-11-29 14:00:40 +00:00
|
|
|
if j == -1 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
toks := strings.Split(s[i+2:i+j], ";")
|
2016-08-28 00:45:05 +00:00
|
|
|
|
2016-11-29 14:00:40 +00:00
|
|
|
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
|
2016-08-28 00:45:05 +00:00
|
|
|
}
|
2016-11-29 14:00:40 +00:00
|
|
|
nums = append(nums, n)
|
|
|
|
}
|
2016-08-28 00:45:05 +00:00
|
|
|
|
2016-11-29 14:00:40 +00:00
|
|
|
for _, n := range nums {
|
|
|
|
if 30 <= n && n <= 37 {
|
|
|
|
fg = termbox.ColorDefault
|
|
|
|
}
|
|
|
|
if 40 <= n && n <= 47 {
|
|
|
|
bg = termbox.ColorDefault
|
2016-08-28 00:45:05 +00:00
|
|
|
}
|
2016-11-29 14:00:40 +00:00
|
|
|
}
|
2016-08-28 00:45:05 +00:00
|
|
|
|
2016-11-29 14:00:40 +00:00
|
|
|
for _, n := range nums {
|
2016-12-18 15:01:45 +00:00
|
|
|
attr, ok := gAnsiCodes[n]
|
|
|
|
if !ok {
|
|
|
|
log.Printf("unknown ansi code: %d", n)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if 1 <= n && n <= 37 {
|
2016-12-18 18:51:27 +00:00
|
|
|
fg |= attr
|
2016-12-18 15:01:45 +00:00
|
|
|
}
|
|
|
|
if 40 <= n && n <= 47 {
|
2016-12-18 18:51:27 +00:00
|
|
|
bg |= attr
|
2016-11-29 14:00:40 +00:00
|
|
|
}
|
2016-08-28 00:45:05 +00:00
|
|
|
}
|
2016-11-29 14:00:40 +00:00
|
|
|
|
|
|
|
i += j
|
|
|
|
continue
|
2016-08-28 00:45:05 +00:00
|
|
|
}
|
|
|
|
|
2016-08-13 12:49:04 +00:00
|
|
|
if x >= win.w {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2016-08-28 00:45:05 +00:00
|
|
|
termbox.SetCell(win.x+x, win.y+y, r, fg, bg)
|
|
|
|
|
|
|
|
i += w - 1
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2016-08-28 00:45:05 +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
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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...))
|
|
|
|
}
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
func (win *win) printl(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), "")
|
|
|
|
}
|
|
|
|
|
2017-01-05 21:23:22 +00:00
|
|
|
func (win *win) printd(dir *dir, marks map[string]int, saves map[string]bool) {
|
2016-08-13 12:49:04 +00:00
|
|
|
if win.w < 3 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fg, bg := termbox.ColorDefault, termbox.ColorDefault
|
|
|
|
|
|
|
|
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 {
|
2016-12-18 15:01:45 +00:00
|
|
|
case f.LinkState == working:
|
|
|
|
fg = termbox.ColorCyan
|
|
|
|
if f.Mode().IsDir() {
|
|
|
|
fg |= termbox.AttrBold
|
2016-10-24 19:18:31 +00:00
|
|
|
}
|
2016-12-18 15:01:45 +00:00
|
|
|
case f.LinkState == broken:
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2016-09-06 20:05:18 +00:00
|
|
|
path := filepath.Join(dir.path, f.Name())
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2017-01-05 21:23:22 +00:00
|
|
|
if _, ok := marks[path]; ok {
|
2016-08-13 12:49:04 +00:00
|
|
|
win.print(0, i, fg, termbox.ColorMagenta, " ")
|
2016-11-07 20:32:19 +00:00
|
|
|
} 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
|
|
|
}
|
|
|
|
|
2016-10-20 20:45:06 +00:00
|
|
|
var s []rune
|
2016-08-13 12:49:04 +00:00
|
|
|
|
|
|
|
s = append(s, ' ')
|
|
|
|
|
2016-10-20 20:45:06 +00:00
|
|
|
for _, r := range f.Name() {
|
|
|
|
s = append(s, r)
|
|
|
|
}
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2016-10-20 20:45:06 +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
|
|
|
}
|
|
|
|
|
2017-02-04 18:28:03 +00:00
|
|
|
var info string
|
|
|
|
|
2017-02-04 18:33:36 +00:00
|
|
|
for _, s := range gOpts.info {
|
2017-02-04 18:28:03 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.Count == -1 {
|
|
|
|
d, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("opening dir to read count: %s", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
names, err := d.Readdirnames(-1)
|
|
|
|
d.Close()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("reading dir count: %s", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
f.Count = len(names)
|
|
|
|
}
|
|
|
|
info = fmt.Sprintf("%s %d", info, f.Count)
|
2017-02-04 18:28:03 +00:00
|
|
|
case "time":
|
|
|
|
info = fmt.Sprintf("%s %12s", info, f.ModTime().Format("Jan _2 15:04"))
|
|
|
|
default:
|
2017-02-04 18:33:36 +00:00
|
|
|
log.Printf("unknown info type: %s", s)
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
2017-02-04 18:28:03 +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
|
|
|
func (win *win) printr(reg []string) {
|
2016-08-13 12:49:04 +00:00
|
|
|
fg, bg := termbox.ColorDefault, termbox.ColorDefault
|
|
|
|
|
2016-09-15 13:16:03 +00:00
|
|
|
for i, l := range reg {
|
|
|
|
win.print(2, i, fg, bg, l)
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
2016-09-15 13:16:03 +00:00
|
|
|
return
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
type ui struct {
|
|
|
|
wins []*win
|
|
|
|
pwdwin *win
|
|
|
|
msgwin *win
|
|
|
|
menuwin *win
|
2016-08-13 12:49:04 +00:00
|
|
|
message string
|
2016-09-15 13:16:03 +00:00
|
|
|
regprev []string
|
2016-12-17 21:47:37 +00:00
|
|
|
dirprev *dir
|
2016-12-15 09:26:06 +00:00
|
|
|
keychan chan string
|
|
|
|
evschan chan termbox.Event
|
2016-12-18 15:01:45 +00:00
|
|
|
menubuf *bytes.Buffer
|
2016-12-15 09:26:06 +00:00
|
|
|
cmdpref string
|
|
|
|
cmdlacc []rune
|
|
|
|
cmdracc []rune
|
|
|
|
cmdbuf []rune
|
2016-12-18 15:01:45 +00:00
|
|
|
keyacc []rune
|
|
|
|
keycnt []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()
|
|
|
|
|
2016-12-15 09:26:06 +00:00
|
|
|
key := make(chan string, 1000)
|
|
|
|
evs := make(chan termbox.Event)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
evs <- termbox.PollEvent()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
return &ui{
|
2016-12-18 19:38:28 +00:00
|
|
|
wins: getWins(),
|
2016-08-13 12:49:04 +00:00
|
|
|
pwdwin: newWin(wtot, 1, 0, 0),
|
|
|
|
msgwin: newWin(wtot, 1, 0, htot-1),
|
|
|
|
menuwin: newWin(wtot, 1, 0, htot-2),
|
2016-12-15 09:26:06 +00:00
|
|
|
keychan: key,
|
|
|
|
evschan: evs,
|
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]
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.msgwin.renew(wtot, 1, 0, htot-1)
|
|
|
|
}
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
func (ui *ui) loadFileInfo(nav *nav) {
|
2016-10-24 19:18:31 +00:00
|
|
|
curr, err := nav.currFile()
|
|
|
|
if err != nil {
|
2016-08-13 12:49:04 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-12-19 19:08:53 +00:00
|
|
|
ui.message = fmt.Sprintf("%v %4s %v", curr.Mode(), humanize(curr.Size()), curr.ModTime().Format(gOpts.timefmt))
|
2016-12-02 22:05:49 +00:00
|
|
|
}
|
|
|
|
|
2016-12-17 21:47:37 +00:00
|
|
|
func (ui *ui) loadFile(nav *nav) {
|
2016-12-02 22:05:49 +00:00
|
|
|
curr, err := nav.currFile()
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2016-09-15 13:16:03 +00:00
|
|
|
|
|
|
|
if !gOpts.preview {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-10-24 19:18:31 +00:00
|
|
|
if curr.IsDir() {
|
2017-11-18 22:25:41 +00:00
|
|
|
dir := nav.load(curr.Path)
|
2016-09-15 13:16:03 +00:00
|
|
|
ui.dirprev = dir
|
2016-10-24 19:18:31 +00:00
|
|
|
} else if curr.Mode().IsRegular() {
|
2016-09-16 10:27:38 +00:00
|
|
|
var reader io.Reader
|
2016-09-15 13:16:03 +00:00
|
|
|
|
|
|
|
if len(gOpts.previewer) != 0 {
|
2016-10-24 19:18:31 +00:00
|
|
|
cmd := exec.Command(gOpts.previewer, curr.Path, strconv.Itoa(nav.height))
|
2016-09-15 13:16:03 +00:00
|
|
|
|
2016-09-16 10:27:38 +00:00
|
|
|
out, err := cmd.StdoutPipe()
|
2016-09-15 13:16:03 +00:00
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("previewing file: %s", err)
|
|
|
|
ui.message = msg
|
|
|
|
log.Print(msg)
|
|
|
|
}
|
|
|
|
|
2016-09-15 17:09:57 +00:00
|
|
|
if err := cmd.Start(); err != nil {
|
|
|
|
msg := fmt.Sprintf("previewing file: %s", err)
|
|
|
|
ui.message = msg
|
|
|
|
log.Print(msg)
|
|
|
|
}
|
2016-09-16 10:27:38 +00:00
|
|
|
|
2016-09-18 09:56:13 +00:00
|
|
|
defer cmd.Wait()
|
2016-09-16 12:22:39 +00:00
|
|
|
defer out.Close()
|
2016-09-16 10:27:38 +00:00
|
|
|
reader = out
|
2016-09-15 13:16:03 +00:00
|
|
|
} else {
|
2016-10-24 19:18:31 +00:00
|
|
|
f, err := os.Open(curr.Path)
|
2016-09-15 13:16:03 +00:00
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("opening file: %s", err)
|
|
|
|
ui.message = msg
|
|
|
|
log.Print(msg)
|
|
|
|
}
|
2016-09-16 10:27:38 +00:00
|
|
|
|
|
|
|
defer f.Close()
|
|
|
|
reader = f
|
2016-09-15 13:16:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ui.regprev = nil
|
|
|
|
|
2016-09-16 10:27:38 +00:00
|
|
|
buf := bufio.NewScanner(reader)
|
2016-09-15 13:16:03 +00:00
|
|
|
|
|
|
|
for i := 0; i < nav.height && buf.Scan(); i++ {
|
|
|
|
for _, r := range buf.Text() {
|
2016-10-01 21:29:14 +00:00
|
|
|
if r == 0 {
|
2016-12-18 15:01:45 +00:00
|
|
|
ui.regprev = []string{"\033[1mbinary\033[0m"}
|
2016-09-15 13:16:03 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ui.regprev = append(ui.regprev, buf.Text())
|
|
|
|
}
|
|
|
|
|
|
|
|
if buf.Err() != nil {
|
|
|
|
msg := fmt.Sprintf("loading file: %s", buf.Err())
|
|
|
|
ui.message = msg
|
|
|
|
log.Print(msg)
|
|
|
|
}
|
|
|
|
}
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
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)
|
2016-10-08 11:18:26 +00:00
|
|
|
|
2016-08-13 12:49:04 +00:00
|
|
|
dir := nav.currDir()
|
|
|
|
|
2017-08-06 08:05:46 +00:00
|
|
|
path := strings.Replace(dir.path, gUser.HomeDir, "~", -1)
|
2016-10-01 21:39:03 +00:00
|
|
|
path = filepath.Clean(path)
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2017-08-06 08:05:46 +00:00
|
|
|
ui.pwdwin.printf(0, 0, termbox.AttrBold|termbox.ColorGreen, bg, "%s@%s", gUser.Username, gHostname)
|
|
|
|
ui.pwdwin.printf(len(gUser.Username)+len(gHostname)+1, 0, fg, bg, ":")
|
|
|
|
ui.pwdwin.printf(len(gUser.Username)+len(gHostname)+2, 0, termbox.AttrBold|termbox.ColorBlue, bg, "%s", path)
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2016-12-15 09:26:06 +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
|
|
|
|
}
|
|
|
|
|
2016-12-15 09:26:06 +00:00
|
|
|
doff := len(nav.dirs) - length
|
2016-08-13 12:49:04 +00:00
|
|
|
for i := 0; i < length; i++ {
|
2016-11-07 20:32:19 +00:00
|
|
|
ui.wins[woff+i].printd(nav.dirs[doff+i], nav.marks, nav.saves)
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
|
2016-12-15 09:26:06 +00:00
|
|
|
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()
|
|
|
|
}
|
2016-08-14 12:15:54 +00:00
|
|
|
|
2016-08-13 12:49:04 +00:00
|
|
|
if gOpts.preview {
|
2016-10-24 19:18:31 +00:00
|
|
|
f, err := nav.currFile()
|
2016-12-15 13:43:29 +00:00
|
|
|
if err == nil {
|
|
|
|
preview := ui.wins[len(ui.wins)-1]
|
2016-08-14 12:15:54 +00:00
|
|
|
|
2016-12-15 13:43:29 +00:00
|
|
|
if f.IsDir() {
|
|
|
|
preview.printd(ui.dirprev, nav.marks, nav.saves)
|
|
|
|
} else if f.Mode().IsRegular() {
|
|
|
|
preview.printr(ui.regprev)
|
|
|
|
}
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
|
|
|
}
|
2016-12-15 09:26:06 +00:00
|
|
|
|
|
|
|
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:] {
|
2016-12-18 15:01:45 +00:00
|
|
|
ui.menuwin.printl(0, i+1, fg, bg, "")
|
|
|
|
ui.menuwin.print(0, i+1, fg, bg, line)
|
2016-12-15 09:26:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
termbox.Flush()
|
|
|
|
|
|
|
|
if ui.cmdpref == "" {
|
|
|
|
// 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-12-15 09:26:06 +00:00
|
|
|
}
|
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 {
|
2016-12-15 09:26:06 +00:00
|
|
|
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 {
|
2016-12-15 09:26:06 +00:00
|
|
|
select {
|
|
|
|
case key := <-ui.keychan:
|
2016-09-18 16:21:24 +00:00
|
|
|
ev := termbox.Event{Type: termbox.EventKey}
|
2016-12-15 09:26:06 +00:00
|
|
|
|
|
|
|
if len(key) == 1 {
|
|
|
|
ev.Ch, _ = utf8.DecodeRuneInString(key)
|
2016-09-18 16:21:24 +00:00
|
|
|
} else {
|
2016-12-15 09:26:06 +00:00
|
|
|
switch key {
|
2016-09-18 16:21:24 +00:00
|
|
|
case "<lt>":
|
|
|
|
ev.Ch = '<'
|
|
|
|
case "<gt>":
|
|
|
|
ev.Ch = '>'
|
|
|
|
default:
|
2016-12-15 09:26:06 +00:00
|
|
|
if val, ok := gValKey[key]; ok {
|
2016-09-18 16:21:24 +00:00
|
|
|
ev.Key = val
|
|
|
|
} else {
|
|
|
|
ev.Key = termbox.KeyEsc
|
2016-12-15 09:26:06 +00:00
|
|
|
msg := fmt.Sprintf("unknown key: %s", key)
|
2016-09-18 16:21:24 +00:00
|
|
|
ui.message = msg
|
|
|
|
log.Print(msg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-15 09:26:06 +00:00
|
|
|
|
|
|
|
return ev
|
|
|
|
case ev := <-ui.evschan:
|
2016-09-18 16:21:24 +00:00
|
|
|
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 {
|
2017-03-10 15:53:21 +00:00
|
|
|
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) {
|
2016-12-17 21:47:37 +00:00
|
|
|
renew := &callExpr{"renew", nil}
|
2016-12-15 09:26:06 +00:00
|
|
|
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 == '<':
|
|
|
|
ui.keyacc = append(ui.keyacc, '<', 'l', 't', '>')
|
|
|
|
case ev.Ch == '>':
|
|
|
|
ui.keyacc = append(ui.keyacc, '<', 'g', 't', '>')
|
|
|
|
case unicode.IsDigit(ev.Ch) && len(ui.keyacc) == 0:
|
|
|
|
ui.keycnt = append(ui.keycnt, ev.Ch)
|
|
|
|
default:
|
|
|
|
ui.keyacc = append(ui.keyacc, ev.Ch)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
val := gKeyVal[ev.Key]
|
|
|
|
if string(val) == "<esc>" {
|
|
|
|
ch <- multiExpr{renew, 1}
|
|
|
|
ui.keyacc = nil
|
|
|
|
ui.keycnt = nil
|
2016-12-15 09:26:06 +00:00
|
|
|
}
|
2016-12-18 15:01:45 +00:00
|
|
|
ui.keyacc = append(ui.keyacc, val...)
|
|
|
|
}
|
2016-12-15 09:26:06 +00:00
|
|
|
|
2016-12-18 15:01:45 +00:00
|
|
|
binds, ok := findBinds(gOpts.keys, string(ui.keyacc))
|
|
|
|
|
|
|
|
switch len(binds) {
|
|
|
|
case 0:
|
|
|
|
ui.message = fmt.Sprintf("unknown mapping: %s", string(ui.keyacc))
|
|
|
|
ch <- multiExpr{renew, 1}
|
|
|
|
ui.keyacc = nil
|
|
|
|
ui.keycnt = nil
|
|
|
|
case 1:
|
|
|
|
if ok {
|
|
|
|
if len(ui.keycnt) > 0 {
|
|
|
|
c, err := strconv.Atoi(string(ui.keycnt))
|
|
|
|
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
|
2016-12-15 09:26:06 +00:00
|
|
|
} else {
|
2016-12-18 15:01:45 +00:00
|
|
|
count = 1
|
2016-12-15 09:26:06 +00:00
|
|
|
}
|
2016-12-18 15:01:45 +00:00
|
|
|
expr := gOpts.keys[string(ui.keyacc)]
|
|
|
|
ch <- multiExpr{expr, count}
|
|
|
|
ui.keyacc = nil
|
|
|
|
ui.keycnt = nil
|
|
|
|
}
|
|
|
|
if len(ui.keyacc) > 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(ui.keycnt) > 0 {
|
|
|
|
c, err := strconv.Atoi(string(ui.keycnt))
|
|
|
|
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
|
|
|
}
|
2016-12-18 15:01:45 +00:00
|
|
|
expr := gOpts.keys[string(ui.keyacc)]
|
|
|
|
ch <- multiExpr{expr, count}
|
|
|
|
ui.keyacc = nil
|
|
|
|
ui.keycnt = nil
|
|
|
|
}
|
|
|
|
if len(ui.keyacc) > 0 {
|
|
|
|
ui.menubuf = listBinds(binds)
|
2016-12-17 21:47:37 +00:00
|
|
|
ch <- multiExpr{renew, 1}
|
2016-12-18 15:01:45 +00:00
|
|
|
} else {
|
|
|
|
ui.menubuf = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case termbox.EventResize:
|
|
|
|
ch <- multiExpr{renew, 1}
|
|
|
|
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() {
|
|
|
|
for {
|
|
|
|
ev := ui.pollEvent()
|
|
|
|
|
|
|
|
if ui.cmdpref != "" && ev.Type == termbox.EventKey {
|
|
|
|
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
|
|
|
}
|
2016-12-15 09:26:06 +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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-15 09:26:06 +00:00
|
|
|
func listMatches(matches []string) *bytes.Buffer {
|
2016-08-21 15:41:03 +00:00
|
|
|
b := new(bytes.Buffer)
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2016-08-21 15:41:03 +00:00
|
|
|
wtot, _ := termbox.Size()
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2016-08-21 15:41:03 +00:00
|
|
|
wcol := 0
|
|
|
|
for _, m := range matches {
|
|
|
|
wcol = max(wcol, len(m))
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|
2016-08-21 15:41:03 +00:00
|
|
|
wcol += gOpts.tabstop - wcol%gOpts.tabstop
|
2016-08-13 12:49:04 +00:00
|
|
|
|
2016-08-21 15:41:03 +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')
|
|
|
|
}
|
|
|
|
|
2016-12-15 09:26:06 +00:00
|
|
|
return b
|
2016-08-13 12:49:04 +00:00
|
|
|
}
|