aerc/widgets/exline.go

138 lines
2.7 KiB
Go
Raw Normal View History

2018-02-27 04:54:39 +01:00
package widgets
import (
2018-02-27 22:46:14 +01:00
"github.com/mattn/go-runewidth"
tb "github.com/nsf/termbox-go"
2018-02-27 04:54:39 +01:00
"git.sr.ht/~sircmpwn/aerc2/lib/ui"
)
// TODO: history
// TODO: tab completion
// TODO: scrolling
type ExLine struct {
2018-02-27 22:46:14 +01:00
command []rune
2018-02-28 03:02:56 +01:00
commit func(cmd string)
cancel func()
index int
scroll int
2018-02-27 04:54:39 +01:00
onInvalidate func(d ui.Drawable)
}
2018-02-28 03:02:56 +01:00
func NewExLine(commit func (cmd string), cancel func()) *ExLine {
return &ExLine{
cancel: cancel,
commit: commit,
command: []rune{},
}
}
2018-02-27 04:54:39 +01:00
func (ex *ExLine) OnInvalidate(onInvalidate func(d ui.Drawable)) {
ex.onInvalidate = onInvalidate
}
func (ex *ExLine) Invalidate() {
if ex.onInvalidate != nil {
ex.onInvalidate(ex)
}
}
2018-02-27 04:54:39 +01:00
func (ex *ExLine) Draw(ctx *ui.Context) {
cell := tb.Cell{
Fg: tb.ColorDefault,
Bg: tb.ColorDefault,
Ch: ' ',
}
ctx.Fill(0, 0, ctx.Width(), ctx.Height(), cell)
2018-02-27 22:46:14 +01:00
ctx.Printf(0, 0, cell, ":%s", string(ex.command))
cells := runewidth.StringWidth(string(ex.command[:ex.index]))
tb.SetCursor(ctx.X()+cells+1, ctx.Y())
}
func (ex *ExLine) insert(ch rune) {
2018-02-27 22:46:14 +01:00
left := ex.command[:ex.index]
right := ex.command[ex.index:]
ex.command = append(left, append([]rune{ch}, right...)...)
ex.index++
ex.Invalidate()
}
func (ex *ExLine) deleteWord() {
// TODO: Break on any of / " '
2018-02-27 22:46:14 +01:00
if len(ex.command) == 0 {
return
}
i := ex.index - 1
2018-02-27 22:46:14 +01:00
if ex.command[i] == ' ' {
i--
}
for ; i >= 0; i-- {
2018-02-27 22:46:14 +01:00
if ex.command[i] == ' ' {
break
}
}
2018-02-27 22:46:14 +01:00
ex.command = append(ex.command[:i+1], ex.command[ex.index:]...)
ex.index = i + 1
ex.Invalidate()
}
func (ex *ExLine) deleteChar() {
2018-02-27 22:46:14 +01:00
if len(ex.command) > 0 && ex.index != len(ex.command) {
ex.command = append(ex.command[:ex.index], ex.command[ex.index+1:]...)
ex.Invalidate()
}
}
func (ex *ExLine) backspace() {
2018-02-27 22:46:14 +01:00
if len(ex.command) > 0 && ex.index != 0 {
ex.command = append(ex.command[:ex.index-1], ex.command[ex.index:]...)
ex.index--
ex.Invalidate()
}
}
func (ex *ExLine) Event(event tb.Event) bool {
switch event.Type {
case tb.EventKey:
switch event.Key {
case tb.KeySpace:
ex.insert(' ')
case tb.KeyBackspace, tb.KeyBackspace2:
ex.backspace()
case tb.KeyCtrlD, tb.KeyDelete:
ex.deleteChar()
case tb.KeyCtrlB, tb.KeyArrowLeft:
if ex.index > 0 {
ex.index--
ex.Invalidate()
}
case tb.KeyCtrlF, tb.KeyArrowRight:
2018-02-27 22:46:14 +01:00
if ex.index < len(ex.command) {
ex.index++
ex.Invalidate()
}
case tb.KeyCtrlA, tb.KeyHome:
ex.index = 0
ex.Invalidate()
case tb.KeyCtrlE, tb.KeyEnd:
2018-02-27 22:46:14 +01:00
ex.index = len(ex.command)
ex.Invalidate()
case tb.KeyCtrlW:
ex.deleteWord()
2018-02-28 03:02:56 +01:00
case tb.KeyEnter:
tb.HideCursor()
ex.commit(string(ex.command))
case tb.KeyEsc, tb.KeyCtrlC:
tb.HideCursor()
ex.cancel()
default:
if event.Ch != 0 {
ex.insert(event.Ch)
}
}
}
return true
}