Path: blob/main/vendor/github.com/chzyer/readline/runebuf.go
2875 views
package readline12import (3"bufio"4"bytes"5"io"6"strconv"7"strings"8"sync"9)1011type runeBufferBck struct {12buf []rune13idx int14}1516type RuneBuffer struct {17buf []rune18idx int19prompt []rune20w io.Writer2122hadClean bool23interactive bool24cfg *Config2526width int2728bck *runeBufferBck2930offset string3132lastKill []rune3334sync.Mutex35}3637func (r *RuneBuffer) pushKill(text []rune) {38r.lastKill = append([]rune{}, text...)39}4041func (r *RuneBuffer) OnWidthChange(newWidth int) {42r.Lock()43r.width = newWidth44r.Unlock()45}4647func (r *RuneBuffer) Backup() {48r.Lock()49r.bck = &runeBufferBck{r.buf, r.idx}50r.Unlock()51}5253func (r *RuneBuffer) Restore() {54r.Refresh(func() {55if r.bck == nil {56return57}58r.buf = r.bck.buf59r.idx = r.bck.idx60})61}6263func NewRuneBuffer(w io.Writer, prompt string, cfg *Config, width int) *RuneBuffer {64rb := &RuneBuffer{65w: w,66interactive: cfg.useInteractive(),67cfg: cfg,68width: width,69}70rb.SetPrompt(prompt)71return rb72}7374func (r *RuneBuffer) SetConfig(cfg *Config) {75r.Lock()76r.cfg = cfg77r.interactive = cfg.useInteractive()78r.Unlock()79}8081func (r *RuneBuffer) SetMask(m rune) {82r.Lock()83r.cfg.MaskRune = m84r.Unlock()85}8687func (r *RuneBuffer) CurrentWidth(x int) int {88r.Lock()89defer r.Unlock()90return runes.WidthAll(r.buf[:x])91}9293func (r *RuneBuffer) PromptLen() int {94r.Lock()95width := r.promptLen()96r.Unlock()97return width98}99100func (r *RuneBuffer) promptLen() int {101return runes.WidthAll(runes.ColorFilter(r.prompt))102}103104func (r *RuneBuffer) RuneSlice(i int) []rune {105r.Lock()106defer r.Unlock()107108if i > 0 {109rs := make([]rune, i)110copy(rs, r.buf[r.idx:r.idx+i])111return rs112}113rs := make([]rune, -i)114copy(rs, r.buf[r.idx+i:r.idx])115return rs116}117118func (r *RuneBuffer) Runes() []rune {119r.Lock()120newr := make([]rune, len(r.buf))121copy(newr, r.buf)122r.Unlock()123return newr124}125126func (r *RuneBuffer) Pos() int {127r.Lock()128defer r.Unlock()129return r.idx130}131132func (r *RuneBuffer) Len() int {133r.Lock()134defer r.Unlock()135return len(r.buf)136}137138func (r *RuneBuffer) MoveToLineStart() {139r.Refresh(func() {140if r.idx == 0 {141return142}143r.idx = 0144})145}146147func (r *RuneBuffer) MoveBackward() {148r.Refresh(func() {149if r.idx == 0 {150return151}152r.idx--153})154}155156func (r *RuneBuffer) WriteString(s string) {157r.WriteRunes([]rune(s))158}159160func (r *RuneBuffer) WriteRune(s rune) {161r.WriteRunes([]rune{s})162}163164func (r *RuneBuffer) WriteRunes(s []rune) {165r.Refresh(func() {166tail := append(s, r.buf[r.idx:]...)167r.buf = append(r.buf[:r.idx], tail...)168r.idx += len(s)169})170}171172func (r *RuneBuffer) MoveForward() {173r.Refresh(func() {174if r.idx == len(r.buf) {175return176}177r.idx++178})179}180181func (r *RuneBuffer) IsCursorInEnd() bool {182r.Lock()183defer r.Unlock()184return r.idx == len(r.buf)185}186187func (r *RuneBuffer) Replace(ch rune) {188r.Refresh(func() {189r.buf[r.idx] = ch190})191}192193func (r *RuneBuffer) Erase() {194r.Refresh(func() {195r.idx = 0196r.pushKill(r.buf[:])197r.buf = r.buf[:0]198})199}200201func (r *RuneBuffer) Delete() (success bool) {202r.Refresh(func() {203if r.idx == len(r.buf) {204return205}206r.pushKill(r.buf[r.idx : r.idx+1])207r.buf = append(r.buf[:r.idx], r.buf[r.idx+1:]...)208success = true209})210return211}212213func (r *RuneBuffer) DeleteWord() {214if r.idx == len(r.buf) {215return216}217init := r.idx218for init < len(r.buf) && IsWordBreak(r.buf[init]) {219init++220}221for i := init + 1; i < len(r.buf); i++ {222if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {223r.pushKill(r.buf[r.idx : i-1])224r.Refresh(func() {225r.buf = append(r.buf[:r.idx], r.buf[i-1:]...)226})227return228}229}230r.Kill()231}232233func (r *RuneBuffer) MoveToPrevWord() (success bool) {234r.Refresh(func() {235if r.idx == 0 {236return237}238239for i := r.idx - 1; i > 0; i-- {240if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {241r.idx = i242success = true243return244}245}246r.idx = 0247success = true248})249return250}251252func (r *RuneBuffer) KillFront() {253r.Refresh(func() {254if r.idx == 0 {255return256}257258length := len(r.buf) - r.idx259r.pushKill(r.buf[:r.idx])260copy(r.buf[:length], r.buf[r.idx:])261r.idx = 0262r.buf = r.buf[:length]263})264}265266func (r *RuneBuffer) Kill() {267r.Refresh(func() {268r.pushKill(r.buf[r.idx:])269r.buf = r.buf[:r.idx]270})271}272273func (r *RuneBuffer) Transpose() {274r.Refresh(func() {275if len(r.buf) == 1 {276r.idx++277}278279if len(r.buf) < 2 {280return281}282283if r.idx == 0 {284r.idx = 1285} else if r.idx >= len(r.buf) {286r.idx = len(r.buf) - 1287}288r.buf[r.idx], r.buf[r.idx-1] = r.buf[r.idx-1], r.buf[r.idx]289r.idx++290})291}292293func (r *RuneBuffer) MoveToNextWord() {294r.Refresh(func() {295for i := r.idx + 1; i < len(r.buf); i++ {296if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {297r.idx = i298return299}300}301302r.idx = len(r.buf)303})304}305306func (r *RuneBuffer) MoveToEndWord() {307r.Refresh(func() {308// already at the end, so do nothing309if r.idx == len(r.buf) {310return311}312// if we are at the end of a word already, go to next313if !IsWordBreak(r.buf[r.idx]) && IsWordBreak(r.buf[r.idx+1]) {314r.idx++315}316317// keep going until at the end of a word318for i := r.idx + 1; i < len(r.buf); i++ {319if IsWordBreak(r.buf[i]) && !IsWordBreak(r.buf[i-1]) {320r.idx = i - 1321return322}323}324r.idx = len(r.buf)325})326}327328func (r *RuneBuffer) BackEscapeWord() {329r.Refresh(func() {330if r.idx == 0 {331return332}333for i := r.idx - 1; i > 0; i-- {334if !IsWordBreak(r.buf[i]) && IsWordBreak(r.buf[i-1]) {335r.pushKill(r.buf[i:r.idx])336r.buf = append(r.buf[:i], r.buf[r.idx:]...)337r.idx = i338return339}340}341342r.buf = r.buf[:0]343r.idx = 0344})345}346347func (r *RuneBuffer) Yank() {348if len(r.lastKill) == 0 {349return350}351r.Refresh(func() {352buf := make([]rune, 0, len(r.buf)+len(r.lastKill))353buf = append(buf, r.buf[:r.idx]...)354buf = append(buf, r.lastKill...)355buf = append(buf, r.buf[r.idx:]...)356r.buf = buf357r.idx += len(r.lastKill)358})359}360361func (r *RuneBuffer) Backspace() {362r.Refresh(func() {363if r.idx == 0 {364return365}366367r.idx--368r.buf = append(r.buf[:r.idx], r.buf[r.idx+1:]...)369})370}371372func (r *RuneBuffer) MoveToLineEnd() {373r.Refresh(func() {374if r.idx == len(r.buf) {375return376}377378r.idx = len(r.buf)379})380}381382func (r *RuneBuffer) LineCount(width int) int {383if width == -1 {384width = r.width385}386return LineCount(width,387runes.WidthAll(r.buf)+r.PromptLen())388}389390func (r *RuneBuffer) MoveTo(ch rune, prevChar, reverse bool) (success bool) {391r.Refresh(func() {392if reverse {393for i := r.idx - 1; i >= 0; i-- {394if r.buf[i] == ch {395r.idx = i396if prevChar {397r.idx++398}399success = true400return401}402}403return404}405for i := r.idx + 1; i < len(r.buf); i++ {406if r.buf[i] == ch {407r.idx = i408if prevChar {409r.idx--410}411success = true412return413}414}415})416return417}418419func (r *RuneBuffer) isInLineEdge() bool {420if isWindows {421return false422}423sp := r.getSplitByLine(r.buf)424return len(sp[len(sp)-1]) == 0425}426427func (r *RuneBuffer) getSplitByLine(rs []rune) []string {428return SplitByLine(r.promptLen(), r.width, rs)429}430431func (r *RuneBuffer) IdxLine(width int) int {432r.Lock()433defer r.Unlock()434return r.idxLine(width)435}436437func (r *RuneBuffer) idxLine(width int) int {438if width == 0 {439return 0440}441sp := r.getSplitByLine(r.buf[:r.idx])442return len(sp) - 1443}444445func (r *RuneBuffer) CursorLineCount() int {446return r.LineCount(r.width) - r.IdxLine(r.width)447}448449func (r *RuneBuffer) Refresh(f func()) {450r.Lock()451defer r.Unlock()452453if !r.interactive {454if f != nil {455f()456}457return458}459460r.clean()461if f != nil {462f()463}464r.print()465}466467func (r *RuneBuffer) SetOffset(offset string) {468r.Lock()469r.offset = offset470r.Unlock()471}472473func (r *RuneBuffer) print() {474r.w.Write(r.output())475r.hadClean = false476}477478func (r *RuneBuffer) output() []byte {479buf := bytes.NewBuffer(nil)480buf.WriteString(string(r.prompt))481if r.cfg.EnableMask && len(r.buf) > 0 {482buf.Write([]byte(strings.Repeat(string(r.cfg.MaskRune), len(r.buf)-1)))483if r.buf[len(r.buf)-1] == '\n' {484buf.Write([]byte{'\n'})485} else {486buf.Write([]byte(string(r.cfg.MaskRune)))487}488if len(r.buf) > r.idx {489buf.Write(r.getBackspaceSequence())490}491492} else {493for _, e := range r.cfg.Painter.Paint(r.buf, r.idx) {494if e == '\t' {495buf.WriteString(strings.Repeat(" ", TabWidth))496} else {497buf.WriteRune(e)498}499}500if r.isInLineEdge() {501buf.Write([]byte(" \b"))502}503}504// cursor position505if len(r.buf) > r.idx {506buf.Write(r.getBackspaceSequence())507}508return buf.Bytes()509}510511func (r *RuneBuffer) getBackspaceSequence() []byte {512var sep = map[int]bool{}513514var i int515for {516if i >= runes.WidthAll(r.buf) {517break518}519520if i == 0 {521i -= r.promptLen()522}523i += r.width524525sep[i] = true526}527var buf []byte528for i := len(r.buf); i > r.idx; i-- {529// move input to the left of one530buf = append(buf, '\b')531if sep[i] {532// up one line, go to the start of the line and move cursor right to the end (r.width)533buf = append(buf, "\033[A\r"+"\033["+strconv.Itoa(r.width)+"C"...)534}535}536537return buf538539}540541func (r *RuneBuffer) Reset() []rune {542ret := runes.Copy(r.buf)543r.buf = r.buf[:0]544r.idx = 0545return ret546}547548func (r *RuneBuffer) calWidth(m int) int {549if m > 0 {550return runes.WidthAll(r.buf[r.idx : r.idx+m])551}552return runes.WidthAll(r.buf[r.idx+m : r.idx])553}554555func (r *RuneBuffer) SetStyle(start, end int, style string) {556if end < start {557panic("end < start")558}559560// goto start561move := start - r.idx562if move > 0 {563r.w.Write([]byte(string(r.buf[r.idx : r.idx+move])))564} else {565r.w.Write(bytes.Repeat([]byte("\b"), r.calWidth(move)))566}567r.w.Write([]byte("\033[" + style + "m"))568r.w.Write([]byte(string(r.buf[start:end])))569r.w.Write([]byte("\033[0m"))570// TODO: move back571}572573func (r *RuneBuffer) SetWithIdx(idx int, buf []rune) {574r.Refresh(func() {575r.buf = buf576r.idx = idx577})578}579580func (r *RuneBuffer) Set(buf []rune) {581r.SetWithIdx(len(buf), buf)582}583584func (r *RuneBuffer) SetPrompt(prompt string) {585r.Lock()586r.prompt = []rune(prompt)587r.Unlock()588}589590func (r *RuneBuffer) cleanOutput(w io.Writer, idxLine int) {591buf := bufio.NewWriter(w)592593if r.width == 0 {594buf.WriteString(strings.Repeat("\r\b", len(r.buf)+r.promptLen()))595buf.Write([]byte("\033[J"))596} else {597buf.Write([]byte("\033[J")) // just like ^k :)598if idxLine == 0 {599buf.WriteString("\033[2K")600buf.WriteString("\r")601} else {602for i := 0; i < idxLine; i++ {603io.WriteString(buf, "\033[2K\r\033[A")604}605io.WriteString(buf, "\033[2K\r")606}607}608buf.Flush()609return610}611612func (r *RuneBuffer) Clean() {613r.Lock()614r.clean()615r.Unlock()616}617618func (r *RuneBuffer) clean() {619r.cleanWithIdxLine(r.idxLine(r.width))620}621622func (r *RuneBuffer) cleanWithIdxLine(idxLine int) {623if r.hadClean || !r.interactive {624return625}626r.hadClean = true627r.cleanOutput(r.w, idxLine)628}629630631