readline: Automatically do word completion based on history

This commit is contained in:
Kovid Goyal 2023-03-07 16:44:02 +05:30
parent 4cef83ffd0
commit 0e73c01093
No known key found for this signature in database
GPG Key ID: 06BC317B515ACE7C
5 changed files with 85 additions and 5 deletions

View File

@ -560,18 +560,18 @@ func (self *Command) Exec(args ...string) {
} }
func (self *Command) GetCompletions(argv []string, init_completions func(*Completions)) *Completions { func (self *Command) GetCompletions(argv []string, init_completions func(*Completions)) *Completions {
ans := Completions{Groups: make([]*MatchGroup, 0, 4)} ans := NewCompletions()
if init_completions != nil { if init_completions != nil {
init_completions(&ans) init_completions(ans)
} }
if len(argv) > 0 { if len(argv) > 0 {
exe := argv[0] exe := argv[0]
cmd := self.FindSubCommand(exe) cmd := self.FindSubCommand(exe)
if cmd != nil { if cmd != nil {
if cmd.ParseArgsForCompletion != nil { if cmd.ParseArgsForCompletion != nil {
cmd.ParseArgsForCompletion(cmd, argv[1:], &ans) cmd.ParseArgsForCompletion(cmd, argv[1:], ans)
} else { } else {
completion_parse_args(cmd, argv[1:], &ans) completion_parse_args(cmd, argv[1:], ans)
} }
} }
} }
@ -582,5 +582,5 @@ func (self *Command) GetCompletions(argv []string, init_completions func(*Comple
} }
} }
ans.Groups = non_empty_groups ans.Groups = non_empty_groups
return &ans return ans
} }

View File

@ -115,12 +115,46 @@ type Completions struct {
split_on_equals bool // true if the cmdline is split on = (BASH does this because readline does this) split_on_equals bool // true if the cmdline is split on = (BASH does this because readline does this)
} }
func NewCompletions() *Completions {
return &Completions{Groups: make([]*MatchGroup, 0, 4)}
}
func (self *Completions) AddPrefixToAllMatches(prefix string) { func (self *Completions) AddPrefixToAllMatches(prefix string) {
for _, mg := range self.Groups { for _, mg := range self.Groups {
mg.AddPrefixToAllMatches(prefix) mg.AddPrefixToAllMatches(prefix)
} }
} }
func (self *Completions) MergeMatchGroup(mg *MatchGroup) {
if len(mg.Matches) == 0 {
return
}
var dest *MatchGroup
for _, q := range self.Groups {
if q.Title == mg.Title {
dest = q
break
}
}
if dest == nil {
dest = self.AddMatchGroup(mg.Title)
dest.NoTrailingSpace = mg.NoTrailingSpace
dest.IsFiles = mg.IsFiles
}
seen := utils.NewSet[string](64)
for _, q := range self.Groups {
for _, m := range q.Matches {
seen.Add(m.Word)
}
}
for _, m := range mg.Matches {
if !seen.Has(m.Word) {
seen.Add(m.Word)
dest.Matches = append(dest.Matches, m)
}
}
}
func (self *Completions) AddMatchGroup(title string) *MatchGroup { func (self *Completions) AddMatchGroup(title string) *MatchGroup {
for _, q := range self.Groups { for _, q := range self.Groups {
if q.Title == title { if q.Title == title {

View File

@ -156,6 +156,9 @@ func New(loop *loop.Loop, r RlInit) *Readline {
completions: completions{completer: r.Completer}, completions: completions{completer: r.Completer},
kill_ring: kill_ring{items: list.New().Init()}, kill_ring: kill_ring{items: list.New().Init()},
} }
if ans.completions.completer == nil && r.HistoryPath != "" {
ans.completions.completer = ans.HistoryCompleter
}
ans.prompt = ans.make_prompt(r.Prompt, false) ans.prompt = ans.make_prompt(r.Prompt, false)
t := "" t := ""
if r.ContinuationPrompt != "" || !r.EmptyContinuationPrompt { if r.ContinuationPrompt != "" || !r.EmptyContinuationPrompt {
@ -168,6 +171,10 @@ func New(loop *loop.Loop, r RlInit) *Readline {
return ans return ans
} }
func (self *Readline) HistoryCompleter(before_cursor, after_cursor string) *cli.Completions {
return self.history_completer(before_cursor, after_cursor)
}
func (self *Readline) SetPrompt(prompt string) { func (self *Readline) SetPrompt(prompt string) {
self.prompt = self.make_prompt(prompt, false) self.prompt = self.make_prompt(prompt, false)
} }

View File

@ -7,6 +7,7 @@ import (
"strings" "strings"
"kitty/tools/cli" "kitty/tools/cli"
"kitty/tools/tty"
"kitty/tools/utils" "kitty/tools/utils"
"kitty/tools/wcswidth" "kitty/tools/wcswidth"
) )
@ -63,6 +64,8 @@ type completions struct {
current completion current completion
} }
var _ = tty.DebugPrintln
func (self *Readline) complete(forwards bool, repeat_count uint) bool { func (self *Readline) complete(forwards bool, repeat_count uint) bool {
c := &self.completions c := &self.completions
if c.completer == nil { if c.completer == nil {

View File

@ -10,6 +10,8 @@ import (
"strings" "strings"
"time" "time"
"kitty/tools/cli"
"kitty/tools/tty"
"kitty/tools/utils" "kitty/tools/utils"
"kitty/tools/utils/shlex" "kitty/tools/utils/shlex"
"kitty/tools/wcswidth" "kitty/tools/wcswidth"
@ -396,3 +398,37 @@ func (self *Readline) history_search_prompt() string {
} }
return fmt.Sprintf("history %s: ", ans) return fmt.Sprintf("history %s: ", ans)
} }
var _ = tty.DebugPrintln
func (self *Readline) history_completer(before_cursor, after_cursor string) (ans *cli.Completions) {
ans = cli.NewCompletions()
if before_cursor != "" {
var words_before_cursor []string
words_before_cursor, ans.CurrentWordIdx = shlex.SplitForCompletion(before_cursor)
idx := len(words_before_cursor)
if idx > 0 {
idx--
}
seen := utils.NewSet[string](16)
mg := ans.AddMatchGroup("History")
for _, x := range self.history.items {
if strings.HasPrefix(x.Cmd, before_cursor) {
words, _ := shlex.SplitForCompletion(x.Cmd)
if idx < len(words) {
word := words[idx]
desc := ""
if !seen.Has(word) {
if word != x.Cmd {
desc = x.Cmd
}
mg.AddMatch(word, desc)
seen.Add(word)
}
}
}
}
}
return
}