readline/runebuf.go

281 lines
4.7 KiB
Go
Raw Normal View History

2015-09-20 18:14:29 +03:00
package readline
import (
"bytes"
"io"
)
type RuneBuffer struct {
2015-09-22 13:16:24 +03:00
buf []rune
idx int
prompt []byte
w io.Writer
2015-09-20 18:14:29 +03:00
}
func NewRuneBuffer(w io.Writer, prompt string) *RuneBuffer {
rb := &RuneBuffer{
2015-09-21 17:27:40 +03:00
prompt: []byte(prompt),
w: w,
2015-09-20 18:14:29 +03:00
}
return rb
}
func (r *RuneBuffer) Runes() []rune {
return r.buf
}
func (r *RuneBuffer) Pos() int {
return r.idx
}
func (r *RuneBuffer) Len() int {
return len(r.buf)
}
func (r *RuneBuffer) MoveToLineStart() {
if r.idx == 0 {
return
}
2015-09-22 12:41:14 +03:00
r.idx = 0
r.Refresh()
2015-09-20 18:14:29 +03:00
}
2015-09-21 08:13:30 +03:00
func (r *RuneBuffer) MoveBackward() {
2015-09-20 18:14:29 +03:00
if r.idx == 0 {
return
}
r.idx--
2015-09-22 12:41:14 +03:00
r.Refresh()
2015-09-20 18:14:29 +03:00
}
2015-09-23 08:52:26 +03:00
func (r *RuneBuffer) WriteString(s string) {
r.WriteRunes([]rune(s))
2015-09-20 18:14:29 +03:00
}
2015-09-23 08:52:26 +03:00
func (r *RuneBuffer) WriteRune(s rune) {
r.WriteRunes([]rune{s})
2015-09-20 18:14:29 +03:00
}
2015-09-23 08:52:26 +03:00
func (r *RuneBuffer) WriteRunes(s []rune) {
tail := append(s, r.buf[r.idx:]...)
r.buf = append(r.buf[:r.idx], tail...)
r.idx++
r.Refresh()
2015-09-20 18:14:29 +03:00
}
2015-09-21 08:13:30 +03:00
func (r *RuneBuffer) MoveForward() {
2015-09-20 18:14:29 +03:00
if r.idx == len(r.buf) {
return
}
r.idx++
2015-09-22 12:41:14 +03:00
r.Refresh()
2015-09-20 18:14:29 +03:00
}
func (r *RuneBuffer) Delete() {
if r.idx == len(r.buf) {
return
}
r.buf = append(r.buf[:r.idx], r.buf[r.idx+1:]...)
2015-09-22 12:41:14 +03:00
r.Refresh()
2015-09-20 18:14:29 +03:00
}
func (r *RuneBuffer) DeleteWord() {
if r.idx == len(r.buf) {
return
}
2015-09-21 17:51:48 +03:00
init := r.idx
2015-09-23 06:59:39 +03:00
for init < len(r.buf) && IsWordBreak(r.buf[init]) {
2015-09-21 17:51:48 +03:00
init++
}
for i := init + 1; i < len(r.buf); i++ {
2015-09-23 06:59:39 +03:00
if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {
2015-09-20 18:14:29 +03:00
r.buf = append(r.buf[:r.idx], r.buf[i-1:]...)
2015-09-22 12:41:14 +03:00
r.Refresh()
2015-09-20 18:14:29 +03:00
return
}
}
2015-09-21 17:51:48 +03:00
r.Kill()
2015-09-20 18:14:29 +03:00
}
func (r *RuneBuffer) MoveToPrevWord() {
if r.idx == 0 {
return
}
for i := r.idx - 1; i > 0; i-- {
2015-09-23 06:59:39 +03:00
if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {
2015-09-22 12:41:14 +03:00
r.idx = i
r.Refresh()
2015-09-20 18:14:29 +03:00
return
}
}
2015-09-22 12:41:14 +03:00
r.idx = 0
r.Refresh()
2015-09-20 18:14:29 +03:00
}
func (r *RuneBuffer) SetIdx(idx int) (change int) {
i := r.idx
r.idx = idx
return r.idx - i
}
2015-09-21 17:51:48 +03:00
func (r *RuneBuffer) Kill() {
r.buf = r.buf[:r.idx]
2015-09-22 12:41:14 +03:00
r.Refresh()
2015-09-21 17:51:48 +03:00
}
2015-09-23 08:03:13 +03:00
func (r *RuneBuffer) Transpose() {
2015-09-23 06:59:39 +03:00
if len(r.buf) < 2 {
if len(r.buf) == 1 {
r.idx++
r.Refresh()
}
return
}
if r.idx == 0 {
r.idx = 1
2015-09-23 08:03:13 +03:00
} else if r.idx >= len(r.buf) {
r.idx = len(r.buf) - 1
2015-09-23 06:59:39 +03:00
}
r.buf[r.idx], r.buf[r.idx-1] = r.buf[r.idx-1], r.buf[r.idx]
r.idx++
r.Refresh()
2015-09-23 06:46:56 +03:00
}
2015-09-20 18:14:29 +03:00
func (r *RuneBuffer) MoveToNextWord() {
for i := r.idx + 1; i < len(r.buf); i++ {
2015-09-23 06:59:39 +03:00
if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {
2015-09-22 12:41:14 +03:00
r.idx = i
r.Refresh()
2015-09-20 18:14:29 +03:00
return
}
}
2015-09-22 12:41:14 +03:00
r.idx = len(r.buf)
r.Refresh()
2015-09-20 18:14:29 +03:00
}
2015-09-21 16:00:48 +03:00
func (r *RuneBuffer) BackEscapeWord() {
if r.idx == 0 {
return
}
for i := r.idx - 1; i > 0; i-- {
2015-09-23 06:59:39 +03:00
if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {
2015-09-21 16:00:48 +03:00
r.buf = append(r.buf[:i], r.buf[r.idx:]...)
2015-09-22 12:41:14 +03:00
r.idx = i
r.Refresh()
2015-09-21 16:00:48 +03:00
return
}
}
r.buf = r.buf[:0]
2015-09-22 12:41:14 +03:00
r.idx = 0
r.Refresh()
2015-09-21 16:00:48 +03:00
}
func (r *RuneBuffer) Backspace() {
2015-09-20 18:14:29 +03:00
if r.idx == 0 {
return
}
r.idx--
r.buf = append(r.buf[:r.idx], r.buf[r.idx+1:]...)
2015-09-22 12:41:14 +03:00
r.Refresh()
2015-09-20 18:14:29 +03:00
}
func (r *RuneBuffer) MoveToLineEnd() {
if r.idx == len(r.buf) {
return
}
2015-09-22 12:41:14 +03:00
r.idx = len(r.buf)
r.Refresh()
2015-09-20 18:14:29 +03:00
}
2015-09-22 18:01:15 +03:00
func (r *RuneBuffer) LineCount() int {
return LineCount(RunesWidth(r.buf) + len(r.prompt))
}
func (r *RuneBuffer) IdxLine() int {
totalWidth := RunesWidth(r.buf[:r.idx]) + len(r.prompt)
w := getWidth()
line := 0
for totalWidth >= w {
totalWidth -= w
line++
}
return line
}
func (r *RuneBuffer) CursorLineCount() int {
return r.LineCount() - r.IdxLine()
}
2015-09-22 12:41:14 +03:00
func (r *RuneBuffer) Refresh() {
r.w.Write(r.Output())
2015-09-20 18:14:29 +03:00
}
2015-09-22 12:41:14 +03:00
func (r *RuneBuffer) Output() []byte {
2015-09-20 18:14:29 +03:00
buf := bytes.NewBuffer(nil)
2015-09-22 12:36:52 +03:00
buf.Write(r.CleanOutput())
buf.Write(r.prompt)
2015-09-20 18:14:29 +03:00
buf.Write([]byte(string(r.buf)))
2015-09-23 06:10:36 +03:00
if len(r.buf) > r.idx {
buf.Write(bytes.Repeat([]byte{'\b'}, len(r.buf)-r.idx))
}
2015-09-20 18:14:29 +03:00
return buf.Bytes()
}
2015-09-22 12:36:52 +03:00
func (r *RuneBuffer) CleanOutput() []byte {
buf := bytes.NewBuffer(nil)
2015-09-22 13:16:24 +03:00
buf.Write([]byte("\033[J")) // just like ^k :)
// TODO: calculate how many line before cursor.
2015-09-22 12:36:52 +03:00
for i := 0; i <= 100; i++ {
buf.WriteString("\033[2K\r\b")
2015-09-21 17:27:40 +03:00
}
2015-09-22 12:36:52 +03:00
return buf.Bytes()
}
2015-09-21 17:27:40 +03:00
2015-09-22 12:36:52 +03:00
func (r *RuneBuffer) Clean() {
r.w.Write(r.CleanOutput())
2015-09-21 17:27:40 +03:00
}
2015-09-20 18:14:29 +03:00
func (r *RuneBuffer) Reset() []rune {
ret := r.buf
r.buf = r.buf[:0]
r.idx = 0
return ret
}
2015-09-21 08:13:30 +03:00
2015-09-23 06:10:36 +03:00
func (r *RuneBuffer) SetStyle(start, end int, style string) {
idx := r.idx
if end < start {
panic("end < start")
}
// goto start
move := start - idx
if move > 0 {
r.w.Write([]byte(string(r.buf[r.idx : r.idx+move])))
} else {
r.w.Write(bytes.Repeat([]byte("\b"), -move))
}
r.w.Write([]byte("\033[" + style))
r.w.Write([]byte(string(r.buf[start:end])))
r.w.Write([]byte("\033[0m"))
if move > 0 {
r.w.Write(bytes.Repeat([]byte("\b"), -move+(end-start)))
} else if -move < end-start {
r.w.Write(bytes.Repeat([]byte("\b"), -move))
} else {
r.w.Write([]byte(string(r.buf[end:r.idx])))
}
}
2015-09-22 18:01:15 +03:00
func (r *RuneBuffer) SetWithIdx(idx int, buf []rune) {
2015-09-21 08:13:30 +03:00
r.buf = buf
2015-09-22 18:01:15 +03:00
r.idx = idx
2015-09-22 12:41:14 +03:00
r.Refresh()
2015-09-21 08:13:30 +03:00
}
2015-09-22 18:01:15 +03:00
func (r *RuneBuffer) Set(buf []rune) {
r.SetWithIdx(len(buf), buf)
}