package main import ( "io" "log" "os" "path/filepath" "strconv" "strings" "time" "unicode" "unicode/utf8" ) func (e *setExpr) eval(app *app, args []string) { switch e.opt { case "anchorfind": gOpts.anchorfind = true case "noanchorfind": gOpts.anchorfind = false case "anchorfind!": gOpts.anchorfind = !gOpts.anchorfind case "color256": gOpts.color256 = true setColorMode() app.ui.pause() app.ui.resume() case "nocolor256": gOpts.color256 = false setColorMode() app.ui.pause() app.ui.resume() case "color256!": gOpts.color256 = !gOpts.color256 setColorMode() app.ui.pause() app.ui.resume() case "dircounts": gOpts.dircounts = true case "nodircounts": gOpts.dircounts = false case "dircounts!": gOpts.dircounts = !gOpts.dircounts case "dirfirst": gOpts.sortType.option |= dirfirstSort app.nav.sort() app.ui.sort() case "nodirfirst": gOpts.sortType.option &= ^dirfirstSort app.nav.sort() app.ui.sort() case "dirfirst!": gOpts.sortType.option ^= dirfirstSort app.nav.sort() app.ui.sort() case "drawbox": gOpts.drawbox = true app.ui.renew() app.nav.height = app.ui.wins[0].h case "nodrawbox": gOpts.drawbox = false app.ui.renew() app.nav.height = app.ui.wins[0].h case "drawbox!": gOpts.drawbox = !gOpts.drawbox app.ui.renew() app.nav.height = app.ui.wins[0].h case "globsearch": gOpts.globsearch = true case "noglobsearch": gOpts.globsearch = false case "globsearch!": gOpts.globsearch = !gOpts.globsearch case "hidden": gOpts.sortType.option |= hiddenSort app.nav.sort() app.nav.position() app.ui.sort() app.ui.loadFile(app.nav) case "nohidden": gOpts.sortType.option &= ^hiddenSort app.nav.sort() app.nav.position() app.ui.sort() app.ui.loadFile(app.nav) case "hidden!": gOpts.sortType.option ^= hiddenSort app.nav.sort() app.nav.position() app.ui.sort() app.ui.loadFile(app.nav) case "ignorecase": gOpts.ignorecase = true case "noignorecase": gOpts.ignorecase = false case "ignorecase!": gOpts.ignorecase = !gOpts.ignorecase case "ignoredia": gOpts.ignoredia = true case "noignoredia": gOpts.ignoredia = false case "ignoredia!": gOpts.ignoredia = !gOpts.ignoredia case "incsearch": gOpts.incsearch = true case "noincsearch": gOpts.incsearch = false case "incsearch!": gOpts.incsearch = !gOpts.incsearch case "preview": gOpts.preview = true case "nopreview": gOpts.preview = false case "preview!": gOpts.preview = !gOpts.preview case "reverse": gOpts.sortType.option |= reverseSort app.nav.sort() app.ui.sort() case "noreverse": gOpts.sortType.option &= ^reverseSort app.nav.sort() app.ui.sort() case "reverse!": gOpts.sortType.option ^= reverseSort app.nav.sort() app.ui.sort() case "smartcase": gOpts.smartcase = true case "nosmartcase": gOpts.smartcase = false case "smartcase!": gOpts.smartcase = !gOpts.smartcase case "smartdia": gOpts.smartdia = true case "nosmartdia": gOpts.smartdia = false case "smartdia!": gOpts.smartdia = !gOpts.smartdia case "wrapscan": gOpts.wrapscan = true case "nowrapscan": gOpts.wrapscan = false case "wrapscan!": gOpts.wrapscan = !gOpts.wrapscan case "number": gOpts.number = true case "nonumber": gOpts.number = false case "number!": gOpts.number = !gOpts.number case "relativenumber": gOpts.relativenumber = true case "norelativenumber": gOpts.relativenumber = false case "relativenumber!": gOpts.relativenumber = !gOpts.relativenumber case "findlen": n, err := strconv.Atoi(e.val) if err != nil { app.ui.printf("findlen: %s", err) return } if n < 0 { app.ui.print("findlen: value should be a non-negative number") return } gOpts.findlen = n case "period": n, err := strconv.Atoi(e.val) if err != nil { app.ui.printf("period: %s", err) return } if n < 0 { app.ui.print("period: value should be a non-negative number") return } gOpts.period = n if n == 0 { app.ticker.Stop() } else { app.ticker.Stop() app.ticker = time.NewTicker(time.Duration(gOpts.period) * time.Second) } case "scrolloff": n, err := strconv.Atoi(e.val) if err != nil { app.ui.printf("scrolloff: %s", err) return } if n < 0 { app.ui.print("scrolloff: value should be a non-negative number") return } gOpts.scrolloff = n case "tabstop": n, err := strconv.Atoi(e.val) if err != nil { app.ui.printf("tabstop: %s", err) return } if n <= 0 { app.ui.print("tabstop: value should be a positive number") return } gOpts.tabstop = n case "filesep": gOpts.filesep = e.val case "ifs": gOpts.ifs = e.val case "previewer": gOpts.previewer = strings.Replace(e.val, "~", gUser.HomeDir, -1) case "promptfmt": gOpts.promptfmt = e.val case "shell": gOpts.shell = e.val case "sortby": if e.val != "natural" && e.val != "name" && e.val != "size" && e.val != "time" { app.ui.print("sortby: value should either be 'natural', 'name', 'size' or 'time'") return } switch e.val { case "natural": gOpts.sortType.method = naturalSort case "name": gOpts.sortType.method = nameSort case "size": gOpts.sortType.method = sizeSort case "time": gOpts.sortType.method = timeSort } app.nav.sort() app.ui.sort() case "timefmt": gOpts.timefmt = e.val case "ratios": toks := strings.Split(e.val, ":") var rats []int for _, s := range toks { n, err := strconv.Atoi(s) if err != nil { app.ui.printf("ratios: %s", err) return } if n <= 0 { app.ui.print("ratios: value should be a positive number") return } rats = append(rats, n) } if gOpts.preview && len(rats) < 2 { app.ui.print("ratios: should consist of at least two numbers when 'preview' is enabled") return } gOpts.ratios = rats app.ui.wins = getWins() app.ui.loadFile(app.nav) case "info": toks := strings.Split(e.val, ":") for _, s := range toks { if s != "" && s != "size" && s != "time" { app.ui.print("info: should consist of 'size' or 'time' separated with colon") return } } gOpts.info = toks case "shellopts": gOpts.shellopts = strings.Split(e.val, ":") default: app.ui.printf("unknown option: %s", e.opt) } } func (e *mapExpr) eval(app *app, args []string) { if e.expr == nil { delete(gOpts.keys, e.keys) return } gOpts.keys[e.keys] = e.expr } func (e *cmapExpr) eval(app *app, args []string) { if e.cmd == "" { delete(gOpts.cmdkeys, e.key) return } gOpts.cmdkeys[e.key] = &callExpr{e.cmd, nil, 1} } func (e *cmdExpr) eval(app *app, args []string) { if e.expr == nil { delete(gOpts.cmds, e.name) return } gOpts.cmds[e.name] = e.expr } func splitKeys(s string) (keys []string) { for i := 0; i < len(s); { r, w := utf8.DecodeRuneInString(s[i:]) if r != '<' { keys = append(keys, s[i:i+w]) i += w } else { j := i + w for r != '>' && j < len(s) { r, w = utf8.DecodeRuneInString(s[j:]) j += w } keys = append(keys, s[i:j]) i = j } } return } func update(app *app) { switch { case gOpts.incsearch && app.ui.cmdPrefix == "/": app.nav.search = string(app.ui.cmdAccLeft) + string(app.ui.cmdAccRight) last := app.nav.currDir() last.ind = app.nav.searchInd last.pos = app.nav.searchPos if app.nav.searchBack { if err := app.nav.searchPrev(); err != nil { app.ui.printf("search-back: %s: %s", err, app.nav.search) return } } else { if err := app.nav.searchNext(); err != nil { app.ui.printf("search: %s: %s", err, app.nav.search) return } } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case gOpts.incsearch && app.ui.cmdPrefix == "?": app.nav.search = string(app.ui.cmdAccLeft) + string(app.ui.cmdAccRight) last := app.nav.currDir() last.ind = app.nav.searchInd last.pos = app.nav.searchPos if app.nav.searchBack { if err := app.nav.searchNext(); err != nil { app.ui.printf("search-back: %s: %s", err, app.nav.search) return } } else { if err := app.nav.searchPrev(); err != nil { app.ui.printf("search: %s: %s", err, app.nav.search) return } } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) } } func insert(app *app, arg string) { switch { case gOpts.incsearch && (app.ui.cmdPrefix == "/" || app.ui.cmdPrefix == "?"): app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(arg)...) update(app) case app.ui.cmdPrefix == "find: ": app.nav.find = string(app.ui.cmdAccLeft) + arg + string(app.ui.cmdAccRight) if gOpts.findlen == 0 { switch app.nav.findSingle() { case 0: app.ui.printf("find: pattern not found: %s", app.nav.find) case 1: app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) default: app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(arg)...) return } } else { if len(app.nav.find) < gOpts.findlen { app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(arg)...) return } if !app.nav.findNext() { app.ui.printf("find: pattern not found: %s", app.nav.find) } else { app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) } } app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" case app.ui.cmdPrefix == "find-back: ": app.nav.find = string(app.ui.cmdAccLeft) + arg + string(app.ui.cmdAccRight) if gOpts.findlen == 0 { switch app.nav.findSingle() { case 0: app.ui.printf("find-back: pattern not found: %s", app.nav.find) case 1: app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) default: app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(arg)...) return } } else { if len(app.nav.find) < gOpts.findlen { app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(arg)...) return } if !app.nav.findPrev() { app.ui.printf("find-back: pattern not found: %s", app.nav.find) } else { app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) } } app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" case app.ui.cmdPrefix == "delete?[y/N]: ": app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" if arg == "y" { if err := app.nav.del(); err != nil { app.ui.printf("delete: %s", err) return } app.nav.unselect() if err := sendRemote("send load"); err != nil { app.ui.printf("delete: %s", err) } } case app.ui.cmdPrefix == "mark-save: ": app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" wd, err := os.Getwd() if err != nil { log.Printf("getting current directory: %s", err) return } app.nav.marks[arg] = wd case app.ui.cmdPrefix == "mark-load: ": app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" wd, err := os.Getwd() if err != nil { log.Printf("getting current directory: %s", err) } path, ok := app.nav.marks[arg] if !ok { app.ui.print("mark-load: no such mark") return } if err := app.nav.cd(path); err != nil { app.ui.printf("%s", err) return } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) if wd != path { app.nav.marks["'"] = wd } default: app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(arg)...) } } func (e *callExpr) eval(app *app, args []string) { switch e.name { case "up": app.nav.up(e.count) app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "half-up": app.nav.up(e.count * app.nav.height / 2) app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "page-up": app.nav.up(e.count * app.nav.height) app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "down": app.nav.down(e.count) app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "half-down": app.nav.down(e.count * app.nav.height / 2) app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "page-down": app.nav.down(e.count * app.nav.height) app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "updir": for i := 0; i < e.count; i++ { if err := app.nav.updir(); err != nil { app.ui.printf("%s", err) return } } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "open": curr, err := app.nav.currFile() if err != nil { app.ui.printf("opening: %s", err) return } if curr.IsDir() { err := app.nav.open() if err != nil { app.ui.printf("opening directory: %s", err) return } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) return } if gSelectionPath != "" { out, err := os.Create(gSelectionPath) if err != nil { log.Printf("opening selection file: %s", err) return } defer out.Close() var path string if list, err := app.nav.currFileOrSelections(); err == nil { path = strings.Join(list, "\n") } else { return } _, err = out.WriteString(path) if err != nil { log.Printf("writing selection file: %s", err) } app.quitChan <- true return } if cmd, ok := gOpts.cmds["open"]; ok { cmd.eval(app, e.args) } case "quit": app.quitChan <- true case "top": app.nav.top() app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "bottom": app.nav.bottom() app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "toggle": for i := 0; i < e.count; i++ { app.nav.toggle() } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "invert": app.nav.invert() case "unselect": app.nav.unselect() case "copy": if err := app.nav.save(true); err != nil { app.ui.printf("copy: %s", err) return } app.nav.unselect() if err := sendRemote("send sync"); err != nil { app.ui.printf("copy: %s", err) } case "cut": if err := app.nav.save(false); err != nil { app.ui.printf("cut: %s", err) return } app.nav.unselect() if err := sendRemote("send sync"); err != nil { app.ui.printf("cut: %s", err) } case "paste": if cmd, ok := gOpts.cmds["paste"]; ok { cmd.eval(app, e.args) } else if err := app.nav.paste(); err != nil { app.ui.printf("paste: %s", err) return } if err := sendRemote("send load"); err != nil { app.ui.printf("paste: %s", err) } if err := sendRemote("send sync"); err != nil { app.ui.printf("paste: %s", err) } case "delete": if cmd, ok := gOpts.cmds["delete"]; ok { cmd.eval(app, e.args) app.nav.unselect() if err := sendRemote("send load"); err != nil { app.ui.printf("delete: %s", err) } } else { if _, err := app.nav.currFileOrSelections(); err != nil { app.ui.printf("delete: %s", err) return } app.ui.cmdPrefix = "delete?[y/N]: " } case "clear": if err := saveFiles(nil, false); err != nil { app.ui.printf("clear: %s", err) return } if err := sendRemote("send sync"); err != nil { app.ui.printf("clear: %s", err) } case "draw": case "redraw": app.ui.sync() app.ui.renew() app.nav.height = app.ui.wins[0].h case "load": app.nav.renew() case "reload": if err := app.nav.reload(); err != nil { app.ui.printf("reload: %s", err) } case "read": app.ui.cmdPrefix = ":" case "shell": app.ui.cmdPrefix = "$" case "shell-pipe": app.ui.cmdPrefix = "%" case "shell-wait": app.ui.cmdPrefix = "!" case "shell-async": app.ui.cmdPrefix = "&" case "find": app.ui.cmdPrefix = "find: " app.nav.findBack = false case "find-back": app.ui.cmdPrefix = "find-back: " app.nav.findBack = true case "find-next": for i := 0; i < e.count; i++ { if app.nav.findBack { app.nav.findPrev() } else { app.nav.findNext() } } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "find-prev": for i := 0; i < e.count; i++ { if app.nav.findBack { app.nav.findNext() } else { app.nav.findPrev() } } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "search": app.ui.cmdPrefix = "/" last := app.nav.currDir() app.nav.searchInd = last.ind app.nav.searchPos = last.pos app.nav.searchBack = false case "search-back": app.ui.cmdPrefix = "?" last := app.nav.currDir() app.nav.searchInd = last.ind app.nav.searchPos = last.pos app.nav.searchBack = true case "search-next": for i := 0; i < e.count; i++ { if app.nav.searchBack { if err := app.nav.searchPrev(); err != nil { app.ui.printf("search-back: %s: %s", err, app.nav.search) return } } else { if err := app.nav.searchNext(); err != nil { app.ui.printf("search: %s: %s", err, app.nav.search) return } } } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "search-prev": for i := 0; i < e.count; i++ { if app.nav.searchBack { if err := app.nav.searchNext(); err != nil { app.ui.printf("search-back: %s: %s", err, app.nav.search) return } } else { if err := app.nav.searchPrev(); err != nil { app.ui.printf("search: %s: %s", err, app.nav.search) return } } } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) case "mark-save": app.ui.cmdPrefix = "mark-save: " case "mark-load": app.ui.menuBuf = listMarks(app.nav.marks) app.ui.cmdPrefix = "mark-load: " case "sync": if err := app.nav.sync(); err != nil { app.ui.printf("sync: %s", err) } case "echo": app.ui.msg = strings.Join(e.args, " ") case "cd": path := "~" if len(e.args) > 0 { path = e.args[0] } wd, err := os.Getwd() if err != nil { log.Printf("getting current directory: %s", err) } if err := app.nav.cd(path); err != nil { app.ui.printf("%s", err) return } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) if !filepath.IsAbs(path) { path = filepath.Join(wd, path) } else { path = filepath.Clean(path) } if wd != path { app.nav.marks["'"] = wd } case "select": if len(e.args) != 1 { app.ui.print("select: requires an argument") return } wd, err := os.Getwd() if err != nil { log.Printf("getting current directory: %s", err) } if err := app.nav.sel(e.args[0]); err != nil { app.ui.printf("%s", err) return } app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) path := filepath.Dir(e.args[0]) if !filepath.IsAbs(path) { path = filepath.Join(wd, path) } else { path = filepath.Clean(path) } if wd != path { app.nav.marks["'"] = wd } case "source": if len(e.args) != 1 { app.ui.print("source: requires an argument") return } app.readFile(strings.Replace(e.args[0], "~", gUser.HomeDir, -1)) case "push": if len(e.args) != 1 { app.ui.print("push: requires an argument") return } 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 { return } insert(app, e.args[0]) case "cmd-escape": if app.ui.cmdPrefix == ">" { return } if gOpts.incsearch && (app.ui.cmdPrefix == "/" || app.ui.cmdPrefix == "?") { last := app.nav.currDir() last.ind = app.nav.searchInd last.pos = app.nav.searchPos app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) } app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" case "cmd-complete": var matches []string if app.ui.cmdPrefix == ":" { matches, app.ui.cmdAccLeft = completeCmd(app.ui.cmdAccLeft) } else { matches, app.ui.cmdAccLeft = completeShell(app.ui.cmdAccLeft) } 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.cmdAccLeft, app.ui.cmdAccRight...)) if len(s) == 0 { return } app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil switch app.ui.cmdPrefix { case ":": log.Printf("command: %s", s) app.ui.cmdPrefix = "" app.cmdHistory = append(app.cmdHistory, cmdItem{":", s}) p := newParser(strings.NewReader(s)) for p.parse() { p.expr.eval(app, nil) } if p.err != nil { app.ui.printf("%s", p.err) } case "$": log.Printf("shell: %s", s) app.ui.cmdPrefix = "" app.cmdHistory = append(app.cmdHistory, cmdItem{"$", s}) app.runShell(s, nil, "$") case "%": log.Printf("shell-pipe: %s", s) app.cmdHistory = append(app.cmdHistory, cmdItem{"%", s}) app.runShell(s, nil, "%") case ">": io.WriteString(app.cmdIn, s+"\n") app.cmdOutBuf = nil case "!": log.Printf("shell-wait: %s", s) app.ui.cmdPrefix = "" app.cmdHistory = append(app.cmdHistory, cmdItem{"!", s}) app.runShell(s, nil, "!") case "&": log.Printf("shell-async: %s", s) app.ui.cmdPrefix = "" app.cmdHistory = append(app.cmdHistory, cmdItem{"&", s}) app.runShell(s, nil, "&") case "/": if gOpts.incsearch { last := app.nav.currDir() last.ind = app.nav.searchInd last.pos = app.nav.searchPos } log.Printf("search: %s", s) app.ui.cmdPrefix = "" app.nav.search = s if err := app.nav.searchNext(); err != nil { app.ui.printf("search: %s: %s", err, app.nav.search) } else { app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) } case "?": if gOpts.incsearch { last := app.nav.currDir() last.ind = app.nav.searchInd last.pos = app.nav.searchPos } log.Printf("search-back: %s", s) app.ui.cmdPrefix = "" app.nav.search = s if err := app.nav.searchPrev(); err != nil { app.ui.printf("search-back: %s: %s", err, app.nav.search) } else { app.ui.loadFile(app.nav) app.ui.loadFileInfo(app.nav) } default: log.Printf("entering unknown execution prefix: %q", app.ui.cmdPrefix) } case "cmd-history-next": if app.ui.cmdPrefix == "" || app.ui.cmdPrefix == ">" { return } if app.cmdHistoryInd > 0 { app.cmdHistoryInd-- } if app.cmdHistoryInd == 0 { app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" return } cmd := app.cmdHistory[len(app.cmdHistory)-app.cmdHistoryInd] app.ui.cmdPrefix = cmd.prefix app.ui.cmdAccLeft = []rune(cmd.value) app.ui.cmdAccRight = nil app.ui.menuBuf = nil case "cmd-history-prev": if app.ui.cmdPrefix == ">" { return } if app.ui.cmdPrefix == "" { app.cmdHistoryInd = 0 } if app.cmdHistoryInd == len(app.cmdHistory) { return } app.cmdHistoryInd++ cmd := app.cmdHistory[len(app.cmdHistory)-app.cmdHistoryInd] app.ui.cmdPrefix = cmd.prefix app.ui.cmdAccLeft = []rune(cmd.value) app.ui.cmdAccRight = nil app.ui.menuBuf = nil case "cmd-delete": if len(app.ui.cmdAccRight) == 0 { return } app.ui.cmdAccRight = app.ui.cmdAccRight[1:] update(app) case "cmd-delete-back": if len(app.ui.cmdAccLeft) == 0 { return } app.ui.cmdAccLeft = app.ui.cmdAccLeft[:len(app.ui.cmdAccLeft)-1] update(app) case "cmd-left": if len(app.ui.cmdAccLeft) == 0 { return } app.ui.cmdAccRight = append([]rune{app.ui.cmdAccLeft[len(app.ui.cmdAccLeft)-1]}, app.ui.cmdAccRight...) app.ui.cmdAccLeft = app.ui.cmdAccLeft[:len(app.ui.cmdAccLeft)-1] case "cmd-right": if len(app.ui.cmdAccRight) == 0 { return } app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, app.ui.cmdAccRight[0]) app.ui.cmdAccRight = app.ui.cmdAccRight[1:] case "cmd-home": app.ui.cmdAccRight = append(app.ui.cmdAccLeft, app.ui.cmdAccRight...) app.ui.cmdAccLeft = nil case "cmd-end": app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, app.ui.cmdAccRight...) app.ui.cmdAccRight = nil case "cmd-delete-home": if len(app.ui.cmdAccLeft) == 0 { return } app.ui.cmdYankBuf = app.ui.cmdAccLeft app.ui.cmdAccLeft = nil update(app) case "cmd-delete-end": if len(app.ui.cmdAccRight) == 0 { return } app.ui.cmdYankBuf = app.ui.cmdAccRight app.ui.cmdAccRight = nil update(app) case "cmd-delete-unix-word": ind := strings.LastIndex(strings.TrimRight(string(app.ui.cmdAccLeft), " "), " ") + 1 app.ui.cmdYankBuf = app.ui.cmdAccLeft[ind:] app.ui.cmdAccLeft = app.ui.cmdAccLeft[:ind] update(app) case "cmd-yank": app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, app.ui.cmdYankBuf...) update(app) case "cmd-transpose": if len(app.ui.cmdAccLeft) < 2 { return } app.ui.cmdAccLeft[len(app.ui.cmdAccLeft)-1], app.ui.cmdAccLeft[len(app.ui.cmdAccLeft)-2] = app.ui.cmdAccLeft[len(app.ui.cmdAccLeft)-2], app.ui.cmdAccLeft[len(app.ui.cmdAccLeft)-1] update(app) case "cmd-interrupt": if app.cmd != nil { app.cmd.Process.Kill() } app.ui.menuBuf = nil app.ui.cmdAccLeft = nil app.ui.cmdAccRight = nil app.ui.cmdPrefix = "" case "cmd-word": if len(app.ui.cmdAccRight) == 0 { return } loc := reWordEnd.FindStringIndex(string(app.ui.cmdAccRight)) if loc == nil { return } ind := loc[0] + 1 app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, app.ui.cmdAccRight[:ind]...) app.ui.cmdAccRight = app.ui.cmdAccRight[ind:] case "cmd-word-back": if len(app.ui.cmdAccLeft) == 0 { return } locs := reWordBeg.FindAllStringIndex(string(app.ui.cmdAccLeft), -1) if locs == nil { return } ind := locs[len(locs)-1][1] - 1 old := app.ui.cmdAccRight app.ui.cmdAccRight = append([]rune{}, app.ui.cmdAccLeft[ind:]...) app.ui.cmdAccRight = append(app.ui.cmdAccRight, old...) app.ui.cmdAccLeft = app.ui.cmdAccLeft[:ind] case "cmd-capitalize-word": if len(app.ui.cmdAccRight) == 0 { return } ind := 0 for ; ind < len(app.ui.cmdAccRight) && unicode.IsSpace(app.ui.cmdAccRight[ind]); ind++ { } if ind >= len(app.ui.cmdAccRight) { return } app.ui.cmdAccRight[ind] = unicode.ToUpper(app.ui.cmdAccRight[ind]) loc := reWordEnd.FindStringIndex(string(app.ui.cmdAccRight)) if loc == nil { return } ind = loc[0] + 1 app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, app.ui.cmdAccRight[:ind]...) app.ui.cmdAccRight = app.ui.cmdAccRight[ind:] update(app) case "cmd-delete-word": if len(app.ui.cmdAccRight) == 0 { return } loc := reWordEnd.FindStringIndex(string(app.ui.cmdAccRight)) if loc == nil { return } ind := loc[0] + 1 app.ui.cmdAccRight = app.ui.cmdAccRight[ind:] update(app) case "cmd-uppercase-word": if len(app.ui.cmdAccRight) == 0 { return } loc := reWordEnd.FindStringIndex(string(app.ui.cmdAccRight)) if loc == nil { return } ind := loc[0] + 1 app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(strings.ToUpper(string(app.ui.cmdAccRight[:ind])))...) app.ui.cmdAccRight = app.ui.cmdAccRight[ind:] update(app) case "cmd-lowercase-word": if len(app.ui.cmdAccRight) == 0 { return } loc := reWordEnd.FindStringIndex(string(app.ui.cmdAccRight)) if loc == nil { return } ind := loc[0] + 1 app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, []rune(strings.ToLower(string(app.ui.cmdAccRight[:ind])))...) app.ui.cmdAccRight = app.ui.cmdAccRight[ind:] update(app) case "cmd-transpose-word": if len(app.ui.cmdAccLeft) == 0 { return } locs := reWord.FindAllStringIndex(string(app.ui.cmdAccLeft), -1) if len(locs) < 2 { return } if len(app.ui.cmdAccRight) > 0 { loc := reWordEnd.FindStringIndex(string(app.ui.cmdAccRight)) if loc != nil { ind := loc[0] + 1 app.ui.cmdAccLeft = append(app.ui.cmdAccLeft, app.ui.cmdAccRight[:ind]...) app.ui.cmdAccRight = app.ui.cmdAccRight[ind:] } } locs = reWord.FindAllStringIndex(string(app.ui.cmdAccLeft), -1) beg1, end1 := locs[len(locs)-2][0], locs[len(locs)-2][1] beg2, end2 := locs[len(locs)-1][0], locs[len(locs)-1][1] var acc []rune acc = append(acc, app.ui.cmdAccLeft[:beg1]...) acc = append(acc, app.ui.cmdAccLeft[beg2:end2]...) acc = append(acc, app.ui.cmdAccLeft[end1:beg2]...) acc = append(acc, app.ui.cmdAccLeft[beg1:end1]...) acc = append(acc, app.ui.cmdAccLeft[end2:]...) app.ui.cmdAccLeft = acc update(app) default: cmd, ok := gOpts.cmds[e.name] if !ok { app.ui.printf("command not found: %s", e.name) return } cmd.eval(app, e.args) } } func (e *execExpr) eval(app *app, args []string) { switch e.prefix { case "$": log.Printf("shell: %s -- %s", e, args) app.runShell(e.value, args, e.prefix) case "%": log.Printf("shell-pipe: %s -- %s", e, args) app.runShell(e.value, args, e.prefix) case "!": log.Printf("shell-wait: %s -- %s", e, args) app.runShell(e.value, args, e.prefix) case "&": log.Printf("shell-async: %s -- %s", e, args) app.runShell(e.value, args, e.prefix) default: log.Printf("evaluating unknown execution prefix: %q", e.prefix) } } func (e *listExpr) eval(app *app, args []string) { for _, expr := range e.exprs { expr.eval(app, nil) } }