This repository was archived by the owner on Aug 10, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathcompletions.go
116 lines (109 loc) · 2.74 KB
/
completions.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
package senpai
import (
"strings"
"git.sr.ht/~taiite/senpai/ui"
)
func (app *App) completionsChannelMembers(cs []ui.Completion, cursorIdx int, text []rune) []ui.Completion {
var start int
for start = cursorIdx - 1; 0 <= start; start-- {
if text[start] == ' ' {
break
}
}
start++
word := text[start:cursorIdx]
if len(word) == 0 {
return cs
}
netID, buffer := app.win.CurrentBuffer()
s := app.sessions[netID] // is not nil
wordCf := s.Casemap(string(word))
for _, name := range s.Names(buffer) {
if strings.HasPrefix(s.Casemap(name.Name.Name), wordCf) {
nickComp := []rune(name.Name.Name)
if start == 0 {
nickComp = append(nickComp, ':')
}
nickComp = append(nickComp, ' ')
c := make([]rune, len(text)+len(nickComp)-len(word))
copy(c[:start], text[:start])
if cursorIdx < len(text) {
copy(c[start+len(nickComp):], text[cursorIdx:])
}
copy(c[start:], nickComp)
cs = append(cs, ui.Completion{
Text: c,
CursorIdx: start + len(nickComp),
})
}
}
return cs
}
func (app *App) completionsChannelTopic(cs []ui.Completion, cursorIdx int, text []rune) []ui.Completion {
if !hasPrefix(text, []rune("/topic ")) {
return cs
}
netID, buffer := app.win.CurrentBuffer()
s := app.sessions[netID] // is not nil
topic, _, _ := s.Topic(buffer)
if cursorIdx == len(text) {
compText := append(text, []rune(topic)...)
cs = append(cs, ui.Completion{
Text: compText,
CursorIdx: len(compText),
})
}
return cs
}
func (app *App) completionsMsg(cs []ui.Completion, cursorIdx int, text []rune) []ui.Completion {
if !hasPrefix(text, []rune("/msg ")) {
return cs
}
s := app.CurrentSession() // is not nil
// Check if the first word (target) is already written and complete (in
// which case we don't have completions to provide).
var word string
hasMetALetter := false
for i := 5; i < cursorIdx; i += 1 {
if hasMetALetter && text[i] == ' ' {
return cs
}
if !hasMetALetter && text[i] != ' ' {
word = s.Casemap(string(text[i:cursorIdx]))
hasMetALetter = true
}
}
if word == "" {
return cs
}
for _, user := range s.Users() {
if strings.HasPrefix(s.Casemap(user), word) {
nickComp := append([]rune(user), ' ')
c := make([]rune, len(text)+5+len(nickComp)-cursorIdx)
copy(c[:5], []rune("/msg "))
copy(c[5:], nickComp)
if cursorIdx < len(text) {
copy(c[5+len(nickComp):], text[cursorIdx:])
}
cs = append(cs, ui.Completion{
Text: c,
CursorIdx: 5 + len(nickComp),
})
}
}
return cs
}
func hasPrefix(s, prefix []rune) bool {
return len(prefix) <= len(s) && equal(prefix, s[:len(prefix)])
}
func equal(a, b []rune) bool {
if len(a) != len(b) {
return false
}
for i := 0; i < len(a); i++ {
if a[i] != b[i] {
return false
}
}
return true
}