Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
kardolus
GitHub Repository: kardolus/chatgpt-cli
Path: blob/main/vendor/github.com/chzyer/readline/complete_segment.go
2875 views
1
package readline
2
3
type SegmentCompleter interface {
4
// a
5
// |- a1
6
// |--- a11
7
// |- a2
8
// b
9
// input:
10
// DoTree([], 0) [a, b]
11
// DoTree([a], 1) [a]
12
// DoTree([a, ], 0) [a1, a2]
13
// DoTree([a, a], 1) [a1, a2]
14
// DoTree([a, a1], 2) [a1]
15
// DoTree([a, a1, ], 0) [a11]
16
// DoTree([a, a1, a], 1) [a11]
17
DoSegment([][]rune, int) [][]rune
18
}
19
20
type dumpSegmentCompleter struct {
21
f func([][]rune, int) [][]rune
22
}
23
24
func (d *dumpSegmentCompleter) DoSegment(segment [][]rune, n int) [][]rune {
25
return d.f(segment, n)
26
}
27
28
func SegmentFunc(f func([][]rune, int) [][]rune) AutoCompleter {
29
return &SegmentComplete{&dumpSegmentCompleter{f}}
30
}
31
32
func SegmentAutoComplete(completer SegmentCompleter) *SegmentComplete {
33
return &SegmentComplete{
34
SegmentCompleter: completer,
35
}
36
}
37
38
type SegmentComplete struct {
39
SegmentCompleter
40
}
41
42
func RetSegment(segments [][]rune, cands [][]rune, idx int) ([][]rune, int) {
43
ret := make([][]rune, 0, len(cands))
44
lastSegment := segments[len(segments)-1]
45
for _, cand := range cands {
46
if !runes.HasPrefix(cand, lastSegment) {
47
continue
48
}
49
ret = append(ret, cand[len(lastSegment):])
50
}
51
return ret, idx
52
}
53
54
func SplitSegment(line []rune, pos int) ([][]rune, int) {
55
segs := [][]rune{}
56
lastIdx := -1
57
line = line[:pos]
58
pos = 0
59
for idx, l := range line {
60
if l == ' ' {
61
pos = 0
62
segs = append(segs, line[lastIdx+1:idx])
63
lastIdx = idx
64
} else {
65
pos++
66
}
67
}
68
segs = append(segs, line[lastIdx+1:])
69
return segs, pos
70
}
71
72
func (c *SegmentComplete) Do(line []rune, pos int) (newLine [][]rune, offset int) {
73
74
segment, idx := SplitSegment(line, pos)
75
76
cands := c.DoSegment(segment, idx)
77
newLine, offset = RetSegment(segment, cands, idx)
78
for idx := range newLine {
79
newLine[idx] = append(newLine[idx], ' ')
80
}
81
return newLine, offset
82
}
83
84