lf/eval.go

507 lines
12 KiB
Go
Raw Normal View History

2016-08-13 12:49:04 +00:00
package main
import (
"fmt"
"log"
"os"
2016-08-13 12:49:04 +00:00
"strconv"
"strings"
"unicode/utf8"
2016-08-13 12:49:04 +00:00
)
2016-12-17 21:47:37 +00:00
func (e *setExpr) eval(app *app, args []string) {
2016-08-13 12:49:04 +00:00
switch e.opt {
case "dirfirst":
gOpts.dirfirst = true
app.nav.renew(app.nav.height)
case "nodirfirst":
gOpts.dirfirst = false
app.nav.renew(app.nav.height)
case "dirfirst!":
gOpts.dirfirst = !gOpts.dirfirst
app.nav.renew(app.nav.height)
2016-08-13 12:49:04 +00:00
case "hidden":
gOpts.hidden = true
app.nav.renew(app.nav.height)
case "nohidden":
gOpts.hidden = false
app.nav.renew(app.nav.height)
case "hidden!":
gOpts.hidden = !gOpts.hidden
app.nav.renew(app.nav.height)
case "preview":
gOpts.preview = true
case "nopreview":
gOpts.preview = false
case "preview!":
gOpts.preview = !gOpts.preview
case "scrolloff":
n, err := strconv.Atoi(e.val)
if err != nil {
2016-08-17 20:00:16 +00:00
msg := fmt.Sprintf("scrolloff: %s", err)
app.ui.message = msg
log.Print(msg)
2016-08-13 12:49:04 +00:00
return
}
if n < 0 {
2016-08-17 20:00:16 +00:00
msg := "scrolloff: value should be a non-negative number"
2016-08-13 12:49:04 +00:00
app.ui.message = msg
log.Print(msg)
return
}
2016-09-02 20:06:25 +00:00
max := app.ui.wins[0].h / 2
2016-08-13 12:49:04 +00:00
if n > max {
n = max
}
gOpts.scrolloff = n
case "tabstop":
n, err := strconv.Atoi(e.val)
if err != nil {
2016-08-17 20:00:16 +00:00
msg := fmt.Sprintf("tabstop: %s", err)
app.ui.message = msg
log.Print(msg)
2016-08-13 12:49:04 +00:00
return
}
if n <= 0 {
2016-08-17 20:00:16 +00:00
msg := "tabstop: value should be a positive number"
2016-08-13 12:49:04 +00:00
app.ui.message = msg
log.Print(msg)
return
}
gOpts.tabstop = n
case "filesep":
gOpts.filesep = e.val
2016-08-13 12:49:04 +00:00
case "ifs":
gOpts.ifs = e.val
case "previewer":
gOpts.previewer = strings.Replace(e.val, "~", envHome, -1)
case "shell":
gOpts.shell = e.val
2016-08-13 12:49:04 +00:00
case "showinfo":
if e.val != "none" && e.val != "size" && e.val != "time" {
msg := "showinfo should either be 'none', 'size' or 'time'"
app.ui.message = msg
log.Print(msg)
return
}
gOpts.showinfo = e.val
case "sortby":
if e.val != "natural" && e.val != "name" && e.val != "size" && e.val != "time" {
msg := "sortby should either be 'natural', 'name', 'size' or 'time'"
2016-08-13 12:49:04 +00:00
app.ui.message = msg
log.Print(msg)
return
}
gOpts.sortby = e.val
app.nav.renew(app.nav.height)
case "timefmt":
gOpts.timefmt = e.val
2016-08-13 12:49:04 +00:00
case "ratios":
toks := strings.Split(e.val, ":")
var rats []int
for _, s := range toks {
i, err := strconv.Atoi(s)
if err != nil {
2016-08-17 20:00:16 +00:00
msg := fmt.Sprintf("ratios: %s", err)
app.ui.message = msg
log.Print(msg)
2016-08-13 12:49:04 +00:00
return
}
rats = append(rats, i)
}
gOpts.ratios = rats
2016-12-18 19:38:28 +00:00
app.ui.wins = getWins()
app.ui.loadFile(app.nav)
2016-08-13 12:49:04 +00:00
default:
msg := fmt.Sprintf("unknown option: %s", e.opt)
app.ui.message = msg
log.Print(msg)
}
}
2016-12-17 21:47:37 +00:00
func (e *mapExpr) eval(app *app, args []string) {
if e.expr == nil {
delete(gOpts.keys, e.keys)
return
}
2016-08-13 12:49:04 +00:00
gOpts.keys[e.keys] = e.expr
}
2016-12-17 21:47:37 +00:00
func (e *cmdExpr) eval(app *app, args []string) {
if e.expr == nil {
delete(gOpts.cmds, e.name)
return
}
2016-08-13 12:49:04 +00:00
gOpts.cmds[e.name] = e.expr
}
func splitKeys(s string) (keys []string) {
for i := 0; i < len(s); {
c, w := utf8.DecodeRuneInString(s[i:])
if c != '<' {
keys = append(keys, s[i:i+w])
i += w
} else {
j := i + w
for c != '>' && j < len(s) {
c, w = utf8.DecodeRuneInString(s[j:])
j += w
}
keys = append(keys, s[i:j])
i = j
}
}
return
}
2016-12-17 21:47:37 +00:00
func (e *callExpr) eval(app *app, args []string) {
2016-08-13 12:49:04 +00:00
// TODO: check for extra toks in each case
switch e.name {
case "up":
app.nav.up(1)
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
case "half-up":
app.nav.up(app.nav.height / 2)
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
case "page-up":
app.nav.up(app.nav.height)
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
2016-08-28 14:02:58 +00:00
case "down":
app.nav.down(1)
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
case "half-down":
app.nav.down(app.nav.height / 2)
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
case "page-down":
app.nav.down(app.nav.height)
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
2016-08-13 12:49:04 +00:00
case "updir":
2016-08-17 20:28:42 +00:00
if err := app.nav.updir(); err != nil {
2016-08-13 12:49:04 +00:00
app.ui.message = err.Error()
log.Print(err)
return
}
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
2016-08-13 12:49:04 +00:00
case "open":
2016-12-18 15:01:45 +00:00
curr, err := app.nav.currFile()
if err != nil {
msg := fmt.Sprintf("opening: %s", err)
app.ui.message = msg
log.Print(msg)
return
}
2016-12-18 15:01:45 +00:00
if curr.IsDir() {
app.nav.open()
if err != nil {
msg := fmt.Sprintf("opening directory: %s", err)
app.ui.message = msg
log.Print(msg)
return
}
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
2016-08-14 12:37:22 +00:00
return
}
if gSelectionPath != "" {
out, err := os.Create(gSelectionPath)
if err != nil {
2016-08-17 20:00:16 +00:00
log.Printf("opening selection file: %s", err)
return
2016-08-14 12:37:22 +00:00
}
defer out.Close()
var path string
2016-08-14 12:37:22 +00:00
if len(app.nav.marks) != 0 {
marks := app.nav.currMarks()
path = strings.Join(marks, "\n")
} else if curr, err := app.nav.currFile(); err == nil {
path = curr.Path
} else {
return
2016-08-14 12:37:22 +00:00
}
_, err = out.WriteString(path)
if err != nil {
2016-08-17 20:00:16 +00:00
log.Printf("writing selection file: %s", err)
2016-08-14 12:37:22 +00:00
}
2016-11-22 17:42:37 +00:00
app.quit <- true
2016-08-14 12:37:22 +00:00
return
}
if cmd, ok := gOpts.cmds["open-file"]; ok {
cmd.eval(app, e.args)
2016-08-13 12:49:04 +00:00
}
2016-08-28 14:02:58 +00:00
case "quit":
2016-11-22 17:42:37 +00:00
app.quit <- true
2016-08-13 12:49:04 +00:00
case "bot":
app.nav.bot()
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
2016-08-13 12:49:04 +00:00
case "top":
app.nav.top()
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
2016-08-13 12:49:04 +00:00
case "read":
app.ui.cmdpref = ":"
2016-08-13 12:49:04 +00:00
case "read-shell":
app.ui.cmdpref = "$"
2016-08-13 12:49:04 +00:00
case "read-shell-wait":
app.ui.cmdpref = "!"
2016-08-13 12:49:04 +00:00
case "read-shell-async":
app.ui.cmdpref = "&"
2016-08-13 12:49:04 +00:00
case "search":
app.ui.cmdpref = "/"
2016-08-13 12:49:04 +00:00
case "search-back":
app.ui.cmdpref = "?"
case "search-next":
app.nav.searchNext()
case "search-prev":
app.nav.searchPrev()
2016-08-13 12:49:04 +00:00
case "toggle":
app.nav.toggle()
case "invert":
app.nav.invert()
2016-08-13 12:49:04 +00:00
case "yank":
2016-08-17 20:28:42 +00:00
if err := app.nav.save(true); err != nil {
2016-08-13 12:49:04 +00:00
msg := fmt.Sprintf("yank: %s", err)
app.ui.message = msg
log.Printf(msg)
return
}
app.nav.marks = make(map[string]bool)
if err := sendRemote("send sync"); err != nil {
msg := fmt.Sprintf("yank: %s", err)
app.ui.message = msg
log.Printf(msg)
}
2016-08-13 12:49:04 +00:00
case "delete":
2016-08-17 20:28:42 +00:00
if err := app.nav.save(false); err != nil {
2016-08-13 12:49:04 +00:00
msg := fmt.Sprintf("delete: %s", err)
app.ui.message = msg
log.Printf(msg)
return
}
app.nav.marks = make(map[string]bool)
if err := sendRemote("send sync"); err != nil {
msg := fmt.Sprintf("delete: %s", err)
app.ui.message = msg
log.Printf(msg)
}
2016-11-06 15:09:18 +00:00
case "put":
if err := app.nav.put(); err != nil {
msg := fmt.Sprintf("put: %s", err)
2016-08-13 12:49:04 +00:00
app.ui.message = msg
log.Printf(msg)
return
}
app.nav.renew(app.nav.height)
app.nav.save(false)
app.nav.saves = make(map[string]bool)
2016-08-13 12:49:04 +00:00
saveFiles(nil, false)
2016-12-19 19:00:50 +00:00
if err := sendRemote("send sync"); err != nil {
msg := fmt.Sprintf("put: %s", err)
app.ui.message = msg
log.Printf(msg)
}
case "clear":
if err := saveFiles(nil, false); err != nil {
msg := fmt.Sprintf("clear: %s", err)
app.ui.message = msg
log.Printf(msg)
return
}
if err := sendRemote("send sync"); err != nil {
msg := fmt.Sprintf("clear: %s", err)
app.ui.message = msg
log.Printf(msg)
}
2016-08-24 09:36:52 +00:00
case "renew":
2016-08-24 09:34:54 +00:00
app.ui.sync()
2016-08-13 12:49:04 +00:00
app.ui.renew()
app.nav.renew(app.ui.wins[0].h)
case "sync":
if err := app.nav.sync(); err != nil {
msg := fmt.Sprintf("sync: %s", err)
app.ui.message = msg
log.Printf(msg)
}
2016-08-28 14:02:58 +00:00
case "echo":
app.ui.message = strings.Join(e.args, " ")
case "cd":
path := "~"
if len(e.args) > 0 {
path = e.args[0]
}
if err := app.nav.cd(path); err != nil {
app.ui.message = err.Error()
log.Print(err)
return
}
app.ui.loadFile(app.nav)
app.ui.loadFileInfo(app.nav)
case "push":
if len(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.nav.search = s
app.nav.searchNext()
case "?":
log.Printf("search-back: %s", s)
app.nav.search = s
app.nav.searchPrev()
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]
}
2016-08-13 12:49:04 +00:00
default:
cmd, ok := gOpts.cmds[e.name]
if !ok {
msg := fmt.Sprintf("command not found: %s", e.name)
app.ui.message = msg
log.Print(msg)
return
}
cmd.eval(app, e.args)
}
}
2016-12-17 21:47:37 +00:00
func (e *execExpr) eval(app *app, args []string) {
2016-08-13 12:49:04 +00:00
switch e.pref {
case "$":
log.Printf("shell: %s -- %s", e, args)
app.runShell(e.expr, args, false, false)
case "!":
log.Printf("shell-wait: %s -- %s", e, args)
app.runShell(e.expr, args, true, false)
case "&":
log.Printf("shell-async: %s -- %s", e, args)
app.runShell(e.expr, args, false, true)
case "/":
log.Printf("search: %s -- %s", e, args)
// TODO: implement
case "?":
log.Printf("search-back: %s -- %s", e, args)
// TODO: implement
default:
log.Printf("evaluating unknown execution prefix: %q", e.pref)
2016-08-13 12:49:04 +00:00
}
}
2016-12-17 21:47:37 +00:00
func (e *listExpr) eval(app *app, args []string) {
2016-08-13 12:49:04 +00:00
for _, expr := range e.exprs {
expr.eval(app, nil)
}
}