2017-05-11 10:49:59 -05:00
|
|
|
package complete
|
|
|
|
|
2017-05-12 16:17:48 -05:00
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2017-11-04 04:20:50 -05:00
|
|
|
"strings"
|
|
|
|
"unicode"
|
2017-05-12 16:17:48 -05:00
|
|
|
)
|
|
|
|
|
2017-05-11 12:28:31 -05:00
|
|
|
// Args describes command line arguments
|
|
|
|
type Args struct {
|
2017-05-12 07:19:47 -05:00
|
|
|
// All lists of all arguments in command line (not including the command itself)
|
2017-05-12 14:43:33 -05:00
|
|
|
All []string
|
2017-05-12 07:19:47 -05:00
|
|
|
// Completed lists of all completed arguments in command line,
|
|
|
|
// If the last one is still being typed - no space after it,
|
|
|
|
// it won't appear in this list of arguments.
|
2017-05-12 14:43:33 -05:00
|
|
|
Completed []string
|
2017-05-12 07:19:47 -05:00
|
|
|
// Last argument in command line, the one being typed, if the last
|
|
|
|
// character in the command line is a space, this argument will be empty,
|
|
|
|
// otherwise this would be the last word.
|
2017-05-12 14:43:33 -05:00
|
|
|
Last string
|
2017-05-12 07:19:47 -05:00
|
|
|
// LastCompleted is the last argument that was fully typed.
|
|
|
|
// If the last character in the command line is space, this would be the
|
|
|
|
// last word, otherwise, it would be the word before that.
|
2017-05-11 12:28:31 -05:00
|
|
|
LastCompleted string
|
2017-05-11 10:49:59 -05:00
|
|
|
}
|
|
|
|
|
2017-05-12 16:17:48 -05:00
|
|
|
// Directory gives the directory of the current written
|
|
|
|
// last argument if it represents a file name being written.
|
|
|
|
// in case that it is not, we fall back to the current directory.
|
2019-07-05 07:10:58 -05:00
|
|
|
//
|
|
|
|
// Deprecated.
|
2017-05-12 16:17:48 -05:00
|
|
|
func (a Args) Directory() string {
|
|
|
|
if info, err := os.Stat(a.Last); err == nil && info.IsDir() {
|
2017-05-15 15:52:04 -05:00
|
|
|
return fixPathForm(a.Last, a.Last)
|
2017-05-12 16:17:48 -05:00
|
|
|
}
|
|
|
|
dir := filepath.Dir(a.Last)
|
2017-05-13 14:44:36 -05:00
|
|
|
if info, err := os.Stat(dir); err != nil || !info.IsDir() {
|
2017-05-12 16:17:48 -05:00
|
|
|
return "./"
|
|
|
|
}
|
2017-05-15 15:52:04 -05:00
|
|
|
return fixPathForm(a.Last, dir)
|
2017-05-12 16:17:48 -05:00
|
|
|
}
|
|
|
|
|
2017-11-04 04:20:50 -05:00
|
|
|
func newArgs(line string) Args {
|
|
|
|
var (
|
|
|
|
all []string
|
|
|
|
completed []string
|
|
|
|
)
|
|
|
|
parts := splitFields(line)
|
|
|
|
if len(parts) > 0 {
|
|
|
|
all = parts[1:]
|
|
|
|
completed = removeLast(parts[1:])
|
|
|
|
}
|
2017-05-11 12:28:31 -05:00
|
|
|
return Args{
|
2017-11-04 04:20:50 -05:00
|
|
|
All: all,
|
2017-05-11 12:28:31 -05:00
|
|
|
Completed: completed,
|
2017-11-04 04:20:50 -05:00
|
|
|
Last: last(parts),
|
2017-05-11 12:28:31 -05:00
|
|
|
LastCompleted: last(completed),
|
2017-05-11 10:49:59 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-19 12:10:37 -05:00
|
|
|
// splitFields returns a list of fields from the given command line.
|
|
|
|
// If the last character is space, it appends an empty field in the end
|
|
|
|
// indicating that the field before it was completed.
|
|
|
|
// If the last field is of the form "a=b", it splits it to two fields: "a", "b",
|
|
|
|
// So it can be completed.
|
2017-11-04 04:20:50 -05:00
|
|
|
func splitFields(line string) []string {
|
|
|
|
parts := strings.Fields(line)
|
2018-10-19 12:10:37 -05:00
|
|
|
|
|
|
|
// Add empty field if the last field was completed.
|
2017-11-04 04:20:50 -05:00
|
|
|
if len(line) > 0 && unicode.IsSpace(rune(line[len(line)-1])) {
|
|
|
|
parts = append(parts, "")
|
|
|
|
}
|
2018-10-19 12:10:37 -05:00
|
|
|
|
|
|
|
// Treat the last field if it is of the form "a=b"
|
2017-11-04 04:20:50 -05:00
|
|
|
parts = splitLastEqual(parts)
|
|
|
|
return parts
|
|
|
|
}
|
|
|
|
|
|
|
|
func splitLastEqual(line []string) []string {
|
|
|
|
if len(line) == 0 {
|
|
|
|
return line
|
|
|
|
}
|
|
|
|
parts := strings.Split(line[len(line)-1], "=")
|
|
|
|
return append(line[:len(line)-1], parts...)
|
|
|
|
}
|
|
|
|
|
2019-10-23 14:46:40 -05:00
|
|
|
// from returns a copy of Args of all arguments after the i'th argument.
|
2017-05-11 12:28:31 -05:00
|
|
|
func (a Args) from(i int) Args {
|
2019-10-23 14:46:40 -05:00
|
|
|
if i >= len(a.All) {
|
|
|
|
i = len(a.All) - 1
|
2017-05-13 14:44:36 -05:00
|
|
|
}
|
2019-10-23 14:46:40 -05:00
|
|
|
a.All = a.All[i+1:]
|
2017-05-13 14:44:36 -05:00
|
|
|
|
2019-10-23 14:46:40 -05:00
|
|
|
if i >= len(a.Completed) {
|
|
|
|
i = len(a.Completed) - 1
|
2017-05-13 14:44:36 -05:00
|
|
|
}
|
2019-10-23 14:46:40 -05:00
|
|
|
a.Completed = a.Completed[i+1:]
|
2017-05-11 10:49:59 -05:00
|
|
|
return a
|
|
|
|
}
|
|
|
|
|
|
|
|
func removeLast(a []string) []string {
|
|
|
|
if len(a) > 0 {
|
|
|
|
return a[:len(a)-1]
|
|
|
|
}
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
|
2017-11-04 04:20:50 -05:00
|
|
|
func last(args []string) string {
|
|
|
|
if len(args) == 0 {
|
|
|
|
return ""
|
2017-05-11 10:49:59 -05:00
|
|
|
}
|
2017-11-04 04:20:50 -05:00
|
|
|
return args[len(args)-1]
|
2017-05-11 10:49:59 -05:00
|
|
|
}
|