mirror of
https://github.com/diamondburned/cchat-gtk.git
synced 2025-01-26 20:06:49 +00:00
245 lines
4.9 KiB
Go
245 lines
4.9 KiB
Go
package completion
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/diamondburned/cchat"
|
|
"github.com/diamondburned/cchat-gtk/internal/gts/httputil"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/primitives"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/primitives/scrollinput"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/rich"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/rich/parser/markup"
|
|
"github.com/diamondburned/cchat/text"
|
|
"github.com/diamondburned/cchat/utils/split"
|
|
"github.com/diamondburned/imgutil"
|
|
"github.com/gotk3/gotk3/gtk"
|
|
)
|
|
|
|
const (
|
|
ImageSmall = 25
|
|
ImageLarge = 40
|
|
ImagePadding = 6
|
|
)
|
|
|
|
// post-processor icon
|
|
var ppIcon = []imgutil.Processor{imgutil.Round(true)}
|
|
|
|
type Completer struct {
|
|
Input *gtk.TextView
|
|
Buffer *gtk.TextBuffer
|
|
List *gtk.ListBox
|
|
Popover *gtk.Popover
|
|
popdown bool
|
|
|
|
Splitter split.SplitFunc
|
|
|
|
words []string
|
|
index int64
|
|
cursor int64
|
|
|
|
entries []cchat.CompletionEntry
|
|
completer cchat.Completer
|
|
}
|
|
|
|
func WrapCompleter(input *gtk.TextView) {
|
|
NewCompleter(input)
|
|
}
|
|
|
|
func NewCompleter(input *gtk.TextView) *Completer {
|
|
l, _ := gtk.ListBoxNew()
|
|
l.Show()
|
|
|
|
s := scrollinput.NewVScroll(150)
|
|
s.Add(l)
|
|
s.Show()
|
|
|
|
p := NewPopover(input)
|
|
p.Add(s)
|
|
|
|
input.Connect("key-press-event", KeyDownHandler(l, input.GrabFocus))
|
|
ibuf, _ := input.GetBuffer()
|
|
|
|
c := &Completer{
|
|
Input: input,
|
|
Buffer: ibuf,
|
|
List: l,
|
|
Popover: p,
|
|
Splitter: split.SpaceIndexed,
|
|
}
|
|
|
|
// This one is for buffer modification.
|
|
ibuf.Connect("end-user-action", func(interface{}) { c.onChange() })
|
|
// This one is for when the cursor moves.
|
|
input.Connect("move-cursor", func(interface{}) { c.onChange() })
|
|
|
|
l.Connect("row-activated", func(l *gtk.ListBox, r *gtk.ListBoxRow) {
|
|
SwapWord(ibuf, c.entries[r.GetIndex()].Raw, c.cursor)
|
|
c.onChange() // signal change
|
|
c.Popdown()
|
|
input.GrabFocus()
|
|
})
|
|
|
|
return c
|
|
}
|
|
|
|
// SetCompleter sets the current completer. If completer is nil, then the
|
|
// completer is disabled.
|
|
func (c *Completer) SetCompleter(completer cchat.Completer) {
|
|
c.Popdown()
|
|
c.completer = completer
|
|
}
|
|
|
|
func (c *Completer) Reset() {
|
|
c.SetCompleter(nil)
|
|
}
|
|
|
|
func (c *Completer) Popup() {
|
|
if c.popdown {
|
|
c.Popover.Popup()
|
|
c.popdown = false
|
|
}
|
|
}
|
|
|
|
func (c *Completer) Popdown() {
|
|
if !c.popdown {
|
|
c.Popover.Popdown()
|
|
c.popdown = true
|
|
c.Clear()
|
|
}
|
|
}
|
|
|
|
func (c *Completer) Clear() {
|
|
var children = c.List.GetChildren()
|
|
if children.Length() == 0 {
|
|
return
|
|
}
|
|
|
|
children.Foreach(func(i interface{}) {
|
|
i.(primitives.WidgetDestroyer).Destroy()
|
|
})
|
|
}
|
|
|
|
// Words returns the buffer content split into words.
|
|
func (c *Completer) Content() []string {
|
|
// This method not to be confused with c.words, which contains the state of
|
|
// completer words.
|
|
|
|
text, _ := c.Buffer.GetText(c.Buffer.GetStartIter(), c.Buffer.GetEndIter(), true)
|
|
if text == "" {
|
|
return nil
|
|
}
|
|
words, _ := c.Splitter(text, 0)
|
|
return words
|
|
}
|
|
|
|
func (c *Completer) onChange() {
|
|
t, v, blank := State(c.Buffer)
|
|
c.cursor = v
|
|
|
|
// If the cursor is on a blank character, then we should not
|
|
// autocomplete anything, so we set the states to nil.
|
|
if blank {
|
|
c.words = nil
|
|
c.index = -1
|
|
c.Popdown()
|
|
return
|
|
}
|
|
|
|
c.words, c.index = c.Splitter(t, v)
|
|
c.complete()
|
|
}
|
|
|
|
func (c *Completer) complete() {
|
|
c.Clear()
|
|
|
|
var widgets []gtk.IWidget
|
|
if len(c.words) > 0 {
|
|
widgets = c.update()
|
|
}
|
|
|
|
if len(widgets) > 0 {
|
|
c.Popover.SetPointingTo(CursorRect(c.Input))
|
|
c.Popup()
|
|
} else {
|
|
c.Popdown()
|
|
return
|
|
}
|
|
|
|
for i, widget := range widgets {
|
|
r, _ := gtk.ListBoxRowNew()
|
|
r.Add(widget)
|
|
r.Show()
|
|
|
|
c.List.Add(r)
|
|
|
|
if i == 0 {
|
|
c.List.SelectRow(r)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *Completer) update() []gtk.IWidget {
|
|
// If we don't have a completer, then don't run.
|
|
if c.completer == nil {
|
|
return nil
|
|
}
|
|
|
|
c.entries = c.completer.Complete(c.words, c.index)
|
|
|
|
var widgets = make([]gtk.IWidget, len(c.entries))
|
|
|
|
for i, entry := range c.entries {
|
|
// Container that holds the label.
|
|
lbox, _ := gtk.BoxNew(gtk.ORIENTATION_VERTICAL, 0)
|
|
lbox.SetVAlign(gtk.ALIGN_CENTER)
|
|
lbox.Show()
|
|
|
|
// Label for the primary text.
|
|
l := rich.NewLabel(entry.Text)
|
|
l.Show()
|
|
lbox.PackStart(l, false, false, 0)
|
|
|
|
// Get the iamge size so we can change and use if needed. The default
|
|
var size = ImageSmall
|
|
if !entry.Secondary.IsEmpty() {
|
|
size = ImageLarge
|
|
|
|
s := rich.NewLabel(text.Rich{})
|
|
s.SetMarkup(fmt.Sprintf(
|
|
`<span alpha="50%%" size="small">%s</span>`,
|
|
markup.Render(entry.Secondary),
|
|
))
|
|
s.Show()
|
|
|
|
lbox.PackStart(s, false, false, 0)
|
|
}
|
|
|
|
b, _ := gtk.BoxNew(gtk.ORIENTATION_HORIZONTAL, 0)
|
|
b.PackEnd(lbox, true, true, ImagePadding)
|
|
b.Show()
|
|
|
|
// Do we have an icon?
|
|
if entry.IconURL != "" {
|
|
img, _ := gtk.ImageNew()
|
|
img.SetMarginStart(ImagePadding)
|
|
img.SetSizeRequest(size, size)
|
|
img.Show()
|
|
|
|
// Prepend the image into the box.
|
|
b.PackEnd(img, false, false, 0)
|
|
|
|
var pps []imgutil.Processor
|
|
if !entry.Image {
|
|
pps = ppIcon
|
|
}
|
|
|
|
httputil.AsyncImage(context.Background(), img, entry.IconURL, pps...)
|
|
}
|
|
|
|
widgets[i] = b
|
|
}
|
|
|
|
return widgets
|
|
}
|