mirror of https://github.com/chzyer/readline.git
Reimplement in a compatible way.
Signed-off-by: Dan Cripe <dan@quattronetworks.com>
This commit is contained in:
parent
f89cf57370
commit
e0fe5c6252
59
complete.go
59
complete.go
|
@ -9,56 +9,13 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type AutoCompleter interface {
|
type AutoCompleter interface {
|
||||||
GetName() []rune
|
// Readline will pass the whole line and current offset to it
|
||||||
GetChildren() []AutoCompleter
|
// Completer need to pass all the candidates, and how long they shared the same characters in line
|
||||||
}
|
// Example:
|
||||||
|
// Do("g", 1) => ["o", "it", "it-shell", "rep"], 1
|
||||||
func Do(ac AutoCompleter, line []rune, pos int) (newLine [][]rune, offset int) {
|
// Do("gi", 2) => ["t", "t-shell"], 1
|
||||||
line = line[:pos]
|
// Do("git", 3) => ["", "-shell"], 0
|
||||||
goNext := false
|
Do(line []rune, pos int) (newLine [][]rune, length int)
|
||||||
var lineCompleter AutoCompleter
|
|
||||||
|
|
||||||
for _, child := range ac.GetChildren() {
|
|
||||||
childName := child.GetName()
|
|
||||||
if len(line) >= len(childName) {
|
|
||||||
if runes.HasPrefix(line, childName) {
|
|
||||||
if len(line) == len(childName) {
|
|
||||||
newLine = append(newLine, []rune{' '})
|
|
||||||
} else {
|
|
||||||
newLine = append(newLine, childName)
|
|
||||||
}
|
|
||||||
offset = len(childName)
|
|
||||||
lineCompleter = child
|
|
||||||
goNext = true
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if runes.HasPrefix(childName, line) {
|
|
||||||
newLine = append(newLine, childName[len(line):])
|
|
||||||
offset = len(line)
|
|
||||||
lineCompleter = child
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(newLine) != 1 {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
tmpLine := make([]rune, 0, len(line))
|
|
||||||
for i := offset; i < len(line); i++ {
|
|
||||||
if line[i] == ' ' {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
tmpLine = append(tmpLine, line[i:]...)
|
|
||||||
return Do(lineCompleter, tmpLine, len(tmpLine))
|
|
||||||
}
|
|
||||||
|
|
||||||
if goNext {
|
|
||||||
return Do(lineCompleter, nil, 0)
|
|
||||||
}
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type opCompleter struct {
|
type opCompleter struct {
|
||||||
|
@ -118,7 +75,7 @@ func (o *opCompleter) OnComplete() {
|
||||||
|
|
||||||
o.ExitCompleteSelectMode()
|
o.ExitCompleteSelectMode()
|
||||||
o.candidateSource = rs
|
o.candidateSource = rs
|
||||||
newLines, offset := Do(o.ac, rs, buf.idx)
|
newLines, offset := o.ac.Do(rs, buf.idx)
|
||||||
if len(newLines) == 0 {
|
if len(newLines) == 0 {
|
||||||
o.ExitCompleteMode(false)
|
o.ExitCompleteMode(false)
|
||||||
return
|
return
|
||||||
|
|
|
@ -1,27 +1,92 @@
|
||||||
package readline
|
package readline
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/chzyer/readline/runes"
|
||||||
|
)
|
||||||
|
|
||||||
type PrefixCompleter struct {
|
type PrefixCompleter struct {
|
||||||
Name []rune
|
Name []rune
|
||||||
Children []AutoCompleter
|
Children []*PrefixCompleter
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p PrefixCompleter) GetName() []rune {
|
func (p *PrefixCompleter) Tree(prefix string) string {
|
||||||
return p.Name
|
buf := bytes.NewBuffer(nil)
|
||||||
|
p.Print(prefix, 0, buf)
|
||||||
|
return buf.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p PrefixCompleter) GetChildren() []AutoCompleter {
|
func (p *PrefixCompleter) Print(prefix string, level int, buf *bytes.Buffer) {
|
||||||
return p.Children
|
if strings.TrimSpace(string(p.Name)) != "" {
|
||||||
|
buf.WriteString(prefix)
|
||||||
|
if level > 0 {
|
||||||
|
buf.WriteString("├")
|
||||||
|
buf.WriteString(strings.Repeat("─", (level*4)-2))
|
||||||
|
buf.WriteString(" ")
|
||||||
|
}
|
||||||
|
buf.WriteString(string(p.Name) + "\n")
|
||||||
|
level++
|
||||||
|
}
|
||||||
|
for _, ch := range p.Children {
|
||||||
|
ch.Print(prefix, level, buf)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewPrefixCompleter(pc ...AutoCompleter) AutoCompleter {
|
func NewPrefixCompleter(pc ...*PrefixCompleter) *PrefixCompleter {
|
||||||
return PcItem("", pc...)
|
return PcItem("", pc...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func PcItem(name string, pc ...AutoCompleter) AutoCompleter {
|
func PcItem(name string, pc ...*PrefixCompleter) *PrefixCompleter {
|
||||||
name += " "
|
name += " "
|
||||||
result := AutoCompleter(PrefixCompleter{
|
return &PrefixCompleter{
|
||||||
Name: []rune(name),
|
Name: []rune(name),
|
||||||
Children: pc,
|
Children: pc,
|
||||||
})
|
}
|
||||||
return result
|
}
|
||||||
|
|
||||||
|
func (p *PrefixCompleter) Do(line []rune, pos int) (newLine [][]rune, offset int) {
|
||||||
|
line = line[:pos]
|
||||||
|
goNext := false
|
||||||
|
var lineCompleter *PrefixCompleter
|
||||||
|
for _, child := range p.Children {
|
||||||
|
if len(line) >= len(child.Name) {
|
||||||
|
if runes.HasPrefix(line, child.Name) {
|
||||||
|
if len(line) == len(child.Name) {
|
||||||
|
newLine = append(newLine, []rune{' '})
|
||||||
|
} else {
|
||||||
|
newLine = append(newLine, child.Name)
|
||||||
|
}
|
||||||
|
offset = len(child.Name)
|
||||||
|
lineCompleter = child
|
||||||
|
goNext = true
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if runes.HasPrefix(child.Name, line) {
|
||||||
|
newLine = append(newLine, child.Name[len(line):])
|
||||||
|
offset = len(line)
|
||||||
|
lineCompleter = child
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(newLine) != 1 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpLine := make([]rune, 0, len(line))
|
||||||
|
for i := offset; i < len(line); i++ {
|
||||||
|
if line[i] == ' ' {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpLine = append(tmpLine, line[i:]...)
|
||||||
|
return lineCompleter.Do(tmpLine, len(tmpLine))
|
||||||
|
}
|
||||||
|
|
||||||
|
if goNext {
|
||||||
|
return lineCompleter.Do(nil, 0)
|
||||||
|
}
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,110 @@
|
||||||
|
package readline
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/chzyer/readline/runes"
|
||||||
|
)
|
||||||
|
|
||||||
|
type PrefixCompleterInterface interface {
|
||||||
|
Print(prefix string, level int, buf *bytes.Buffer)
|
||||||
|
Do(line []rune, pos int) (newLine [][]rune, length int)
|
||||||
|
GetName() []rune
|
||||||
|
GetChildren() []PrefixCompleterInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
type PrefixCompleterExt struct {
|
||||||
|
Name []rune
|
||||||
|
Children []PrefixCompleterInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func Print(p PrefixCompleterInterface, prefix string, level int, buf *bytes.Buffer) {
|
||||||
|
if strings.TrimSpace(string(p.GetName())) != "" {
|
||||||
|
buf.WriteString(prefix)
|
||||||
|
if level > 0 {
|
||||||
|
buf.WriteString("├")
|
||||||
|
buf.WriteString(strings.Repeat("─", (level*4)-2))
|
||||||
|
buf.WriteString(" ")
|
||||||
|
}
|
||||||
|
buf.WriteString(string(p.GetName()) + "\n")
|
||||||
|
level++
|
||||||
|
}
|
||||||
|
for _, ch := range p.GetChildren() {
|
||||||
|
ch.Print(prefix, level, buf)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *PrefixCompleterExt) Print(prefix string, level int, buf *bytes.Buffer) {
|
||||||
|
Print(p, prefix, level, buf)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *PrefixCompleterExt) GetName() []rune {
|
||||||
|
return p.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *PrefixCompleterExt) GetChildren() []PrefixCompleterInterface {
|
||||||
|
return p.Children
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewPrefixCompleterExt(pc ...PrefixCompleterInterface) *PrefixCompleterExt {
|
||||||
|
return PceItem("", pc...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func PceItem(name string, pc ...PrefixCompleterInterface) *PrefixCompleterExt {
|
||||||
|
name += " "
|
||||||
|
return &PrefixCompleterExt{
|
||||||
|
Name: []rune(name),
|
||||||
|
Children: pc,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *PrefixCompleterExt) Do(line []rune, pos int) (newLine [][]rune, offset int) {
|
||||||
|
return Do(p, line, pos)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Do(p PrefixCompleterInterface, line []rune, pos int) (newLine [][]rune, offset int) {
|
||||||
|
line = line[:pos]
|
||||||
|
goNext := false
|
||||||
|
var lineCompleter PrefixCompleterInterface
|
||||||
|
for _, child := range p.GetChildren() {
|
||||||
|
childName := child.GetName()
|
||||||
|
if len(line) >= len(childName) {
|
||||||
|
if runes.HasPrefix(line, childName) {
|
||||||
|
if len(line) == len(childName) {
|
||||||
|
newLine = append(newLine, []rune{' '})
|
||||||
|
} else {
|
||||||
|
newLine = append(newLine, childName)
|
||||||
|
}
|
||||||
|
offset = len(childName)
|
||||||
|
lineCompleter = child
|
||||||
|
goNext = true
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if runes.HasPrefix(childName, line) {
|
||||||
|
newLine = append(newLine, childName[len(line):])
|
||||||
|
offset = len(line)
|
||||||
|
lineCompleter = child
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(newLine) != 1 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpLine := make([]rune, 0, len(line))
|
||||||
|
for i := offset; i < len(line); i++ {
|
||||||
|
if line[i] == ' ' {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpLine = append(tmpLine, line[i:]...)
|
||||||
|
return lineCompleter.Do(tmpLine, len(tmpLine))
|
||||||
|
}
|
||||||
|
|
||||||
|
if goNext {
|
||||||
|
return lineCompleter.Do(nil, 0)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
Loading…
Reference in New Issue