From 9c92a64bf4121c153484eed3eec9146af76559bf Mon Sep 17 00:00:00 2001 From: David Haukeness Date: Wed, 9 Oct 2019 15:21:25 -0600 Subject: [PATCH 1/4] working PoC tab completion, currently writes to Feed --- main.go | 85 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 85 insertions(+) diff --git a/main.go b/main.go index 981172a..ead1cf6 100644 --- a/main.go +++ b/main.go @@ -202,6 +202,85 @@ func populateList() { } } +func filterStringSlice(ss []string, fv string) []string { + var rs []string + for _, s := range ss { + if strings.HasPrefix(s, fv) { + rs = append(rs, s) + } + } + return rs +} + +func longestCommonPrefix(ss []string) string { + // cover the case where the slice has no or one members + switch len(ss) { + case 0: + return "" + case 1: + return ss[0] + } + // all strings are compared by bytes here forward (TBD unicode normalization?) + // establish min, max lenth members of the slice by iterating over the members + min, max := ss[0], ss[0] + for _, s := range ss[1:] { + switch { + case s < min: + min = s + case s > max: + max = s + } + } + // then iterate over the characters from min to max, as soon as chars don't match return + for i := 0; i < len(min) && i < len(max); i++ { + if min[i] != max[i] { + return min[:i] + } + } + // to cover the case where all members are equal, just return one + return min +} + +func generateTabCompletionSlice(inputWord string) []string { + // gets all possible tab completion cantidates + if testVar, err := k.ChatList(); err != nil { + log.Printf("%+v", err) + } else { + // create a slice to hold the values + var firstSlice []string + // iterate over all the conversation results + for _, s := range testVar.Result.Conversations { + if s.Channel.MembersType == keybase.TEAM { + // its a team so add the topic name as a possible tab completion + firstSlice = append(firstSlice, s.Channel.TopicName) + } else { + // its a user, so clean the name and append the users name as a possible tab completion + firstSlice = append(firstSlice, cleanChannelName(s.Channel.Name)) + } + } + // now return the resultSlice which contains all that are prefixed with inputWord + resultSlice := filterStringSlice(firstSlice, inputWord) + return resultSlice + } + return nil +} + +func handleTab() error { + inputString, err := getInputString("Input") + if err != nil { + return err + } else { + // if you successfully get an input string, grab the last word from the string + ss := strings.Split(inputString, " ") + s := ss[len(ss)-1] + // now call get the list of all possible cantidates that have that as a prefix + resultSlice := generateTabCompletionSlice(s) + result := longestCommonPrefix(resultSlice) + printToView("Feed", fmt.Sprintf("TabCompletion: %s", result)) + } + return nil +} + func clearView(viewName string) { g.Update(func(g *gocui.Gui) error { inputView, err := g.View(viewName) @@ -327,6 +406,12 @@ func initKeybindings() error { }); err != nil { return err } + if err := g.SetKeybinding("Input", gocui.KeyTab, gocui.ModNone, + func(g *gocui.Gui, v *gocui.View) error { + return handleTab() + }); err != nil { + return err + } if err := g.SetKeybinding("Edit", gocui.KeyEnter, gocui.ModNone, func(g *gocui.Gui, v *gocui.View) error { popupView("Chat") From 316fbfa9aaf80b2041379eb20cb059d790ece21e Mon Sep 17 00:00:00 2001 From: David Haukeness Date: Thu, 10 Oct 2019 09:48:03 -0600 Subject: [PATCH 2/4] updated to use cached conversation list working autocomplete on input line still only a partial list of team/channel pairs, need to see what i'm missing --- main.go | 75 +++++++++++++++++++++++++++++++++++++++++---------------- 1 file changed, 54 insertions(+), 21 deletions(-) diff --git a/main.go b/main.go index ead1cf6..bb58795 100644 --- a/main.go +++ b/main.go @@ -241,28 +241,46 @@ func longestCommonPrefix(ss []string) string { return min } +func stringRemainder(aStr, bStr string) string { + var long, short string + //figure out which string is longer + switch { + case len(aStr) < len (bStr): + short = aStr + long = bStr + default: + short = bStr + long = aStr + } + // iterate over the strings using an external iterator so we don't lose the value + i := 0 + for i < len(short) && i < len(long) { + if short[i] != long[i] { + // the strings aren't equal so don't return anything + return "" + } + i++ + } + // return whatever's left of the longer string + return long[i:] +} + func generateTabCompletionSlice(inputWord string) []string { - // gets all possible tab completion cantidates - if testVar, err := k.ChatList(); err != nil { - log.Printf("%+v", err) - } else { - // create a slice to hold the values - var firstSlice []string - // iterate over all the conversation results - for _, s := range testVar.Result.Conversations { - if s.Channel.MembersType == keybase.TEAM { - // its a team so add the topic name as a possible tab completion - firstSlice = append(firstSlice, s.Channel.TopicName) - } else { - // its a user, so clean the name and append the users name as a possible tab completion - firstSlice = append(firstSlice, cleanChannelName(s.Channel.Name)) - } + // create a slice to hold the values + var firstSlice []string + // iterate over all the conversation results + for _, s := range channels { + if s.MembersType == keybase.TEAM { + // its a team so add the topic name as a possible tab completion + firstSlice = append(firstSlice, s.TopicName) + } else { + // its a user, so clean the name and append the users name as a possible tab completion + firstSlice = append(firstSlice, cleanChannelName(s.Name)) } - // now return the resultSlice which contains all that are prefixed with inputWord - resultSlice := filterStringSlice(firstSlice, inputWord) - return resultSlice } - return nil + // now return the resultSlice which contains all that are prefixed with inputWord + resultSlice := filterStringSlice(firstSlice, inputWord) + return resultSlice } func handleTab() error { @@ -275,8 +293,9 @@ func handleTab() error { s := ss[len(ss)-1] // now call get the list of all possible cantidates that have that as a prefix resultSlice := generateTabCompletionSlice(s) - result := longestCommonPrefix(resultSlice) - printToView("Feed", fmt.Sprintf("TabCompletion: %s", result)) + lcp := longestCommonPrefix(resultSlice) + remainder := stringRemainder(s, lcp) + writeToView("Input", remainder) } return nil } @@ -296,6 +315,20 @@ func clearView(viewName string) { } +func writeToView(viewName string, message string) { + g.Update(func(g *gocui.Gui) error { + updatingView, err := g.View(viewName) + if err != nil { + return err + } else { + for _, c := range message { + updatingView.EditWrite(c) + } + } + return nil + }) +} + func printToView(viewName string, message string) { g.Update(func(g *gocui.Gui) error { updatingView, err := g.View(viewName) From ccf2136a61a5a7356eac765d048289c19c880047 Mon Sep 17 00:00:00 2001 From: David Haukeness Date: Thu, 10 Oct 2019 09:57:36 -0600 Subject: [PATCH 3/4] fixed condition where blank lcp would create never ending strings --- main.go | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/main.go b/main.go index bb58795..293dc3d 100644 --- a/main.go +++ b/main.go @@ -294,8 +294,10 @@ func handleTab() error { // now call get the list of all possible cantidates that have that as a prefix resultSlice := generateTabCompletionSlice(s) lcp := longestCommonPrefix(resultSlice) - remainder := stringRemainder(s, lcp) - writeToView("Input", remainder) + if lcp != "" { + remainder := stringRemainder(s, lcp) + writeToView("Input", remainder) + } } return nil } From 3579f1648d2f3247dedec3146df6751c96ce8238 Mon Sep 17 00:00:00 2001 From: David Haukeness Date: Thu, 10 Oct 2019 10:00:54 -0600 Subject: [PATCH 4/4] updated to also add channel names in addition to topics --- main.go | 1 + 1 file changed, 1 insertion(+) diff --git a/main.go b/main.go index 293dc3d..337b576 100644 --- a/main.go +++ b/main.go @@ -273,6 +273,7 @@ func generateTabCompletionSlice(inputWord string) []string { if s.MembersType == keybase.TEAM { // its a team so add the topic name as a possible tab completion firstSlice = append(firstSlice, s.TopicName) + firstSlice = append(firstSlice, s.Name) } else { // its a user, so clean the name and append the users name as a possible tab completion firstSlice = append(firstSlice, cleanChannelName(s.Name))