208 lines
3.7 KiB
Go
208 lines
3.7 KiB
Go
package main
|
|
|
|
import (
|
|
"io/ioutil"
|
|
"log"
|
|
"os"
|
|
"path"
|
|
"strings"
|
|
)
|
|
|
|
var (
|
|
gCmdWords = []string{"set", "map", "cmd"}
|
|
gOptWords = []string{
|
|
"preview",
|
|
"nopreview",
|
|
"preview!",
|
|
"hidden",
|
|
"nohidden",
|
|
"hidden!",
|
|
"tabstop",
|
|
"scrolloff",
|
|
"sortby",
|
|
"showinfo",
|
|
"ratios",
|
|
}
|
|
)
|
|
|
|
func matchLongest(s1, s2 string) string {
|
|
i := 0
|
|
for ; i < len(s1) && i < len(s2); i++ {
|
|
if s1[i] != s2[i] {
|
|
break
|
|
}
|
|
}
|
|
return s1[:i]
|
|
}
|
|
|
|
func matchWord(s string, words []string) (matches []string, longest string) {
|
|
for _, w := range words {
|
|
if strings.HasPrefix(w, s) {
|
|
matches = append(matches, w)
|
|
if longest != "" {
|
|
longest = matchLongest(longest, w)
|
|
} else {
|
|
longest = w + " "
|
|
}
|
|
}
|
|
}
|
|
|
|
if longest == "" {
|
|
longest = s
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func matchExec(s string) (matches []string, longest string) {
|
|
paths := strings.Split(envPath, ":")
|
|
|
|
for _, p := range paths {
|
|
if _, err := os.Stat(p); os.IsNotExist(err) {
|
|
continue
|
|
}
|
|
|
|
fi, err := ioutil.ReadDir(p)
|
|
if err != nil {
|
|
log.Printf("reading path: %s", err)
|
|
}
|
|
|
|
for _, f := range fi {
|
|
if strings.HasPrefix(f.Name(), s) {
|
|
f, err = os.Stat(path.Join(p, f.Name()))
|
|
if err != nil {
|
|
log.Printf("getting file information: %s", err)
|
|
}
|
|
|
|
if !f.Mode().IsRegular() || f.Mode()&0111 == 0 {
|
|
continue
|
|
}
|
|
|
|
matches = append(matches, f.Name())
|
|
if longest != "" {
|
|
longest = matchLongest(longest, f.Name())
|
|
} else {
|
|
longest = f.Name() + " "
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if longest == "" {
|
|
longest = s
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func matchFile(s string) (matches []string, longest string) {
|
|
dir := strings.Replace(s, "~", envHome, -1)
|
|
|
|
if !path.IsAbs(dir) {
|
|
wd, err := os.Getwd()
|
|
if err != nil {
|
|
log.Printf("getting current directory: %s", err)
|
|
}
|
|
dir = wd + "/" + dir
|
|
}
|
|
|
|
dir = path.Dir(dir)
|
|
|
|
fi, err := ioutil.ReadDir(dir)
|
|
if err != nil {
|
|
log.Printf("reading directory: %s", err)
|
|
}
|
|
|
|
for _, f := range fi {
|
|
f, err := os.Stat(path.Join(dir, f.Name()))
|
|
if err != nil {
|
|
log.Printf("getting file information: %s", err)
|
|
return
|
|
}
|
|
|
|
_, last := path.Split(s)
|
|
if strings.HasPrefix(f.Name(), last) {
|
|
name := f.Name()
|
|
if isRoot(s) || path.Base(s) != s {
|
|
name = path.Join(path.Dir(s), f.Name())
|
|
}
|
|
matches = append(matches, f.Name())
|
|
if longest != "" {
|
|
longest = matchLongest(longest, name)
|
|
} else {
|
|
if f.Mode().IsRegular() {
|
|
longest = name + " "
|
|
} else {
|
|
longest = name + "/"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if longest == "" {
|
|
longest = s
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func compCmd(acc []rune) (matches []string, longestAcc []rune) {
|
|
if len(acc) == 0 || acc[len(acc)-1] == ' ' {
|
|
return matches, acc
|
|
}
|
|
|
|
s := string(acc)
|
|
f := strings.Fields(s)
|
|
|
|
var longest string
|
|
|
|
switch len(f) {
|
|
case 0:
|
|
longestAcc = acc
|
|
case 1:
|
|
words := gCmdWords
|
|
for c, _ := range gOpts.cmds {
|
|
words = append(words, c)
|
|
}
|
|
matches, longest = matchWord(s, words)
|
|
longestAcc = []rune(longest)
|
|
default:
|
|
switch f[0] {
|
|
case "set":
|
|
matches, longest = matchWord(f[1], gOptWords)
|
|
longestAcc = append(acc[:len(acc)-len(f[len(f)-1])], []rune(longest)...)
|
|
case "map", "cmd":
|
|
longestAcc = acc
|
|
default:
|
|
matches, longest = matchFile(f[len(f)-1])
|
|
longestAcc = append(acc[:len(acc)-len(f[len(f)-1])], []rune(longest)...)
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func compShell(acc []rune) (matches []string, longestAcc []rune) {
|
|
if len(acc) == 0 || acc[len(acc)-1] == ' ' {
|
|
return matches, acc
|
|
}
|
|
|
|
s := string(acc)
|
|
f := strings.Fields(s)
|
|
|
|
var longest string
|
|
|
|
switch len(f) {
|
|
case 0:
|
|
longestAcc = acc
|
|
case 1:
|
|
matches, longest = matchExec(s)
|
|
longestAcc = []rune(longest)
|
|
default:
|
|
matches, longest = matchFile(f[len(f)-1])
|
|
longestAcc = append(acc[:len(acc)-len(f[len(f)-1])], []rune(longest)...)
|
|
}
|
|
|
|
return
|
|
}
|