mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-10 15:14:09 +00:00
163 lines
2.6 KiB
Go
163 lines
2.6 KiB
Go
|
package readline
|
||
|
|
||
|
const (
|
||
|
vim_NORMAL = iota
|
||
|
vim_INSERT
|
||
|
vim_VISUAL
|
||
|
)
|
||
|
|
||
|
type opVim struct {
|
||
|
op *operation
|
||
|
vimMode int
|
||
|
}
|
||
|
|
||
|
func newVimMode(op *operation) *opVim {
|
||
|
ov := &opVim{
|
||
|
op: op,
|
||
|
vimMode: vim_INSERT,
|
||
|
}
|
||
|
return ov
|
||
|
}
|
||
|
|
||
|
func (o *opVim) IsEnableVimMode() bool {
|
||
|
return o.op.GetConfig().VimMode
|
||
|
}
|
||
|
|
||
|
func (o *opVim) handleVimNormalMovement(r rune, readNext func() rune) (t rune, handled bool) {
|
||
|
rb := o.op.buf
|
||
|
handled = true
|
||
|
switch r {
|
||
|
case 'h':
|
||
|
t = CharBackward
|
||
|
case 'j':
|
||
|
t = CharNext
|
||
|
case 'k':
|
||
|
t = CharPrev
|
||
|
case 'l':
|
||
|
t = CharForward
|
||
|
case '0', '^':
|
||
|
rb.MoveToLineStart()
|
||
|
case '$':
|
||
|
rb.MoveToLineEnd()
|
||
|
case 'x':
|
||
|
rb.Delete()
|
||
|
if rb.IsCursorInEnd() {
|
||
|
rb.MoveBackward()
|
||
|
}
|
||
|
case 'r':
|
||
|
rb.Replace(readNext())
|
||
|
case 'd':
|
||
|
next := readNext()
|
||
|
switch next {
|
||
|
case 'd':
|
||
|
rb.Erase()
|
||
|
case 'w':
|
||
|
rb.DeleteWord()
|
||
|
case 'h':
|
||
|
rb.Backspace()
|
||
|
case 'l':
|
||
|
rb.Delete()
|
||
|
}
|
||
|
case 'p':
|
||
|
rb.Yank()
|
||
|
case 'b', 'B':
|
||
|
rb.MoveToPrevWord()
|
||
|
case 'w', 'W':
|
||
|
rb.MoveToNextWord()
|
||
|
case 'e', 'E':
|
||
|
rb.MoveToEndWord()
|
||
|
case 'f', 'F', 't', 'T':
|
||
|
next := readNext()
|
||
|
prevChar := r == 't' || r == 'T'
|
||
|
reverse := r == 'F' || r == 'T'
|
||
|
switch next {
|
||
|
case CharEsc:
|
||
|
default:
|
||
|
rb.MoveTo(next, prevChar, reverse)
|
||
|
}
|
||
|
default:
|
||
|
return r, false
|
||
|
}
|
||
|
return t, true
|
||
|
}
|
||
|
|
||
|
func (o *opVim) handleVimNormalEnterInsert(r rune, readNext func() rune) (t rune, handled bool) {
|
||
|
rb := o.op.buf
|
||
|
handled = true
|
||
|
switch r {
|
||
|
case 'i':
|
||
|
case 'I':
|
||
|
rb.MoveToLineStart()
|
||
|
case 'a':
|
||
|
rb.MoveForward()
|
||
|
case 'A':
|
||
|
rb.MoveToLineEnd()
|
||
|
case 's':
|
||
|
rb.Delete()
|
||
|
case 'S':
|
||
|
rb.Erase()
|
||
|
case 'c':
|
||
|
next := readNext()
|
||
|
switch next {
|
||
|
case 'c':
|
||
|
rb.Erase()
|
||
|
case 'w':
|
||
|
rb.DeleteWord()
|
||
|
case 'h':
|
||
|
rb.Backspace()
|
||
|
case 'l':
|
||
|
rb.Delete()
|
||
|
}
|
||
|
default:
|
||
|
return r, false
|
||
|
}
|
||
|
|
||
|
o.EnterVimInsertMode()
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (o *opVim) HandleVimNormal(r rune, readNext func() rune) (t rune) {
|
||
|
switch r {
|
||
|
case CharEnter, CharInterrupt:
|
||
|
o.vimMode = vim_INSERT // ???
|
||
|
return r
|
||
|
}
|
||
|
|
||
|
if r, handled := o.handleVimNormalMovement(r, readNext); handled {
|
||
|
return r
|
||
|
}
|
||
|
|
||
|
if r, handled := o.handleVimNormalEnterInsert(r, readNext); handled {
|
||
|
return r
|
||
|
}
|
||
|
|
||
|
// invalid operation
|
||
|
o.op.t.Bell()
|
||
|
return 0
|
||
|
}
|
||
|
|
||
|
func (o *opVim) EnterVimInsertMode() {
|
||
|
o.vimMode = vim_INSERT
|
||
|
}
|
||
|
|
||
|
func (o *opVim) ExitVimInsertMode() {
|
||
|
o.vimMode = vim_NORMAL
|
||
|
}
|
||
|
|
||
|
func (o *opVim) HandleVim(r rune, readNext func() rune) rune {
|
||
|
if o.vimMode == vim_NORMAL {
|
||
|
return o.HandleVimNormal(r, readNext)
|
||
|
}
|
||
|
if r == CharEsc {
|
||
|
o.ExitVimInsertMode()
|
||
|
return 0
|
||
|
}
|
||
|
|
||
|
switch o.vimMode {
|
||
|
case vim_INSERT:
|
||
|
return r
|
||
|
case vim_VISUAL:
|
||
|
}
|
||
|
return r
|
||
|
}
|