mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-12-31 15:06:26 +00:00
190 lines
3.6 KiB
Go
190 lines
3.6 KiB
Go
|
package readline
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"container/list"
|
||
|
"fmt"
|
||
|
"sync"
|
||
|
)
|
||
|
|
||
|
type searchState uint
|
||
|
|
||
|
const (
|
||
|
searchStateFound searchState = iota
|
||
|
searchStateFailing
|
||
|
)
|
||
|
|
||
|
type searchDirection uint
|
||
|
|
||
|
const (
|
||
|
searchDirectionForward searchDirection = iota
|
||
|
searchDirectionBackward
|
||
|
)
|
||
|
|
||
|
type opSearch struct {
|
||
|
mutex sync.Mutex
|
||
|
inMode bool
|
||
|
state searchState
|
||
|
dir searchDirection
|
||
|
source *list.Element
|
||
|
w *terminal
|
||
|
buf *runeBuffer
|
||
|
data []rune
|
||
|
history *opHistory
|
||
|
markStart int
|
||
|
markEnd int
|
||
|
}
|
||
|
|
||
|
func newOpSearch(w *terminal, buf *runeBuffer, history *opHistory) *opSearch {
|
||
|
return &opSearch{
|
||
|
w: w,
|
||
|
buf: buf,
|
||
|
history: history,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) IsSearchMode() bool {
|
||
|
o.mutex.Lock()
|
||
|
defer o.mutex.Unlock()
|
||
|
return o.inMode
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) SearchBackspace() {
|
||
|
o.mutex.Lock()
|
||
|
defer o.mutex.Unlock()
|
||
|
if len(o.data) > 0 {
|
||
|
o.data = o.data[:len(o.data)-1]
|
||
|
o.search(true)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) findHistoryBy(isNewSearch bool) (int, *list.Element) {
|
||
|
if o.dir == searchDirectionBackward {
|
||
|
return o.history.FindBck(isNewSearch, o.data, o.buf.idx)
|
||
|
}
|
||
|
return o.history.FindFwd(isNewSearch, o.data, o.buf.idx)
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) search(isChange bool) bool {
|
||
|
if len(o.data) == 0 {
|
||
|
o.state = searchStateFound
|
||
|
o.searchRefresh(-1)
|
||
|
return true
|
||
|
}
|
||
|
idx, elem := o.findHistoryBy(isChange)
|
||
|
if elem == nil {
|
||
|
o.searchRefresh(-2)
|
||
|
return false
|
||
|
}
|
||
|
o.history.current = elem
|
||
|
|
||
|
item := o.history.showItem(o.history.current.Value)
|
||
|
start, end := 0, 0
|
||
|
if o.dir == searchDirectionBackward {
|
||
|
start, end = idx, idx+len(o.data)
|
||
|
} else {
|
||
|
start, end = idx, idx+len(o.data)
|
||
|
idx += len(o.data)
|
||
|
}
|
||
|
o.buf.SetWithIdx(idx, item)
|
||
|
o.markStart, o.markEnd = start, end
|
||
|
o.searchRefresh(idx)
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) SearchChar(r rune) {
|
||
|
o.mutex.Lock()
|
||
|
defer o.mutex.Unlock()
|
||
|
|
||
|
o.data = append(o.data, r)
|
||
|
o.search(true)
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) SearchMode(dir searchDirection) bool {
|
||
|
o.mutex.Lock()
|
||
|
defer o.mutex.Unlock()
|
||
|
|
||
|
tWidth, _ := o.w.GetWidthHeight()
|
||
|
if tWidth == 0 {
|
||
|
return false
|
||
|
}
|
||
|
alreadyInMode := o.inMode
|
||
|
o.inMode = true
|
||
|
o.dir = dir
|
||
|
o.source = o.history.current
|
||
|
if alreadyInMode {
|
||
|
o.search(false)
|
||
|
} else {
|
||
|
o.searchRefresh(-1)
|
||
|
}
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) ExitSearchMode(revert bool) {
|
||
|
o.mutex.Lock()
|
||
|
defer o.mutex.Unlock()
|
||
|
|
||
|
if revert {
|
||
|
o.history.current = o.source
|
||
|
var redrawValue []rune
|
||
|
if o.history.current != nil {
|
||
|
redrawValue = o.history.showItem(o.history.current.Value)
|
||
|
}
|
||
|
o.buf.Set(redrawValue)
|
||
|
}
|
||
|
o.markStart, o.markEnd = 0, 0
|
||
|
o.state = searchStateFound
|
||
|
o.inMode = false
|
||
|
o.source = nil
|
||
|
o.data = nil
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) searchRefresh(x int) {
|
||
|
tWidth, _ := o.w.GetWidthHeight()
|
||
|
if x == -2 {
|
||
|
o.state = searchStateFailing
|
||
|
} else if x >= 0 {
|
||
|
o.state = searchStateFound
|
||
|
}
|
||
|
if x < 0 {
|
||
|
x = o.buf.idx
|
||
|
}
|
||
|
x = o.buf.CurrentWidth(x)
|
||
|
x += o.buf.PromptLen()
|
||
|
x = x % tWidth
|
||
|
|
||
|
if o.markStart > 0 {
|
||
|
o.buf.SetStyle(o.markStart, o.markEnd, "4")
|
||
|
}
|
||
|
|
||
|
lineCnt := o.buf.CursorLineCount()
|
||
|
buf := bytes.NewBuffer(nil)
|
||
|
buf.Write(bytes.Repeat([]byte("\n"), lineCnt))
|
||
|
buf.WriteString("\033[J")
|
||
|
if o.state == searchStateFailing {
|
||
|
buf.WriteString("failing ")
|
||
|
}
|
||
|
if o.dir == searchDirectionBackward {
|
||
|
buf.WriteString("bck")
|
||
|
} else if o.dir == searchDirectionForward {
|
||
|
buf.WriteString("fwd")
|
||
|
}
|
||
|
buf.WriteString("-i-search: ")
|
||
|
buf.WriteString(string(o.data)) // keyword
|
||
|
buf.WriteString("\033[4m \033[0m") // _
|
||
|
fmt.Fprintf(buf, "\r\033[%dA", lineCnt) // move prev
|
||
|
if x > 0 {
|
||
|
fmt.Fprintf(buf, "\033[%dC", x) // move forward
|
||
|
}
|
||
|
o.w.Write(buf.Bytes())
|
||
|
}
|
||
|
|
||
|
func (o *opSearch) RefreshIfNeeded() {
|
||
|
o.mutex.Lock()
|
||
|
defer o.mutex.Unlock()
|
||
|
|
||
|
if o.inMode {
|
||
|
o.searchRefresh(-1)
|
||
|
}
|
||
|
}
|