223 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			223 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2017 Zack Guo <zack.y.guo@gmail.com>. All rights reserved.
 | |
| // Use of this source code is governed by a MIT license that can
 | |
| // be found in the LICENSE file.
 | |
| 
 | |
| package termui
 | |
| 
 | |
| import (
 | |
| 	"regexp"
 | |
| 	"strings"
 | |
| 
 | |
| 	tm "github.com/nsf/termbox-go"
 | |
| )
 | |
| import rw "github.com/mattn/go-runewidth"
 | |
| 
 | |
| /* ---------------Port from termbox-go --------------------- */
 | |
| 
 | |
| // Attribute is printable cell's color and style.
 | |
| type Attribute uint16
 | |
| 
 | |
| // 8 basic clolrs
 | |
| const (
 | |
| 	ColorDefault Attribute = iota
 | |
| 	ColorBlack
 | |
| 	ColorRed
 | |
| 	ColorGreen
 | |
| 	ColorYellow
 | |
| 	ColorBlue
 | |
| 	ColorMagenta
 | |
| 	ColorCyan
 | |
| 	ColorWhite
 | |
| )
 | |
| 
 | |
| //Have a constant that defines number of colors
 | |
| const NumberofColors = 8
 | |
| 
 | |
| // Text style
 | |
| const (
 | |
| 	AttrBold Attribute = 1 << (iota + 9)
 | |
| 	AttrUnderline
 | |
| 	AttrReverse
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	dot  = "…"
 | |
| 	dotw = rw.StringWidth(dot)
 | |
| )
 | |
| 
 | |
| /* ----------------------- End ----------------------------- */
 | |
| 
 | |
| func toTmAttr(x Attribute) tm.Attribute {
 | |
| 	return tm.Attribute(x)
 | |
| }
 | |
| 
 | |
| func str2runes(s string) []rune {
 | |
| 	return []rune(s)
 | |
| }
 | |
| 
 | |
| // Here for backwards-compatibility.
 | |
| func trimStr2Runes(s string, w int) []rune {
 | |
| 	return TrimStr2Runes(s, w)
 | |
| }
 | |
| 
 | |
| // TrimStr2Runes trims string to w[-1 rune], appends …, and returns the runes
 | |
| // of that string if string is grather then n. If string is small then w,
 | |
| // return the runes.
 | |
| func TrimStr2Runes(s string, w int) []rune {
 | |
| 	if w <= 0 {
 | |
| 		return []rune{}
 | |
| 	}
 | |
| 
 | |
| 	sw := rw.StringWidth(s)
 | |
| 	if sw > w {
 | |
| 		return []rune(rw.Truncate(s, w, dot))
 | |
| 	}
 | |
| 	return str2runes(s)
 | |
| }
 | |
| 
 | |
| // TrimStrIfAppropriate trim string to "s[:-1] + …"
 | |
| // if string > width otherwise return string
 | |
| func TrimStrIfAppropriate(s string, w int) string {
 | |
| 	if w <= 0 {
 | |
| 		return ""
 | |
| 	}
 | |
| 
 | |
| 	sw := rw.StringWidth(s)
 | |
| 	if sw > w {
 | |
| 		return rw.Truncate(s, w, dot)
 | |
| 	}
 | |
| 
 | |
| 	return s
 | |
| }
 | |
| 
 | |
| func strWidth(s string) int {
 | |
| 	return rw.StringWidth(s)
 | |
| }
 | |
| 
 | |
| func charWidth(ch rune) int {
 | |
| 	return rw.RuneWidth(ch)
 | |
| }
 | |
| 
 | |
| var whiteSpaceRegex = regexp.MustCompile(`\s`)
 | |
| 
 | |
| // StringToAttribute converts text to a termui attribute. You may specifiy more
 | |
| // then one attribute like that: "BLACK, BOLD, ...". All whitespaces
 | |
| // are ignored.
 | |
| func StringToAttribute(text string) Attribute {
 | |
| 	text = whiteSpaceRegex.ReplaceAllString(strings.ToLower(text), "")
 | |
| 	attributes := strings.Split(text, ",")
 | |
| 	result := Attribute(0)
 | |
| 
 | |
| 	for _, theAttribute := range attributes {
 | |
| 		var match Attribute
 | |
| 		switch theAttribute {
 | |
| 		case "reset", "default":
 | |
| 			match = ColorDefault
 | |
| 
 | |
| 		case "black":
 | |
| 			match = ColorBlack
 | |
| 
 | |
| 		case "red":
 | |
| 			match = ColorRed
 | |
| 
 | |
| 		case "green":
 | |
| 			match = ColorGreen
 | |
| 
 | |
| 		case "yellow":
 | |
| 			match = ColorYellow
 | |
| 
 | |
| 		case "blue":
 | |
| 			match = ColorBlue
 | |
| 
 | |
| 		case "magenta":
 | |
| 			match = ColorMagenta
 | |
| 
 | |
| 		case "cyan":
 | |
| 			match = ColorCyan
 | |
| 
 | |
| 		case "white":
 | |
| 			match = ColorWhite
 | |
| 
 | |
| 		case "bold":
 | |
| 			match = AttrBold
 | |
| 
 | |
| 		case "underline":
 | |
| 			match = AttrUnderline
 | |
| 
 | |
| 		case "reverse":
 | |
| 			match = AttrReverse
 | |
| 		}
 | |
| 
 | |
| 		result |= match
 | |
| 	}
 | |
| 
 | |
| 	return result
 | |
| }
 | |
| 
 | |
| // TextCells returns a coloured text cells []Cell
 | |
| func TextCells(s string, fg, bg Attribute) []Cell {
 | |
| 	cs := make([]Cell, 0, len(s))
 | |
| 
 | |
| 	// sequence := MarkdownTextRendererFactory{}.TextRenderer(s).Render(fg, bg)
 | |
| 	// runes := []rune(sequence.NormalizedText)
 | |
| 	runes := str2runes(s)
 | |
| 
 | |
| 	for n := range runes {
 | |
| 		// point, _ := sequence.PointAt(n, 0, 0)
 | |
| 		// cs = append(cs, Cell{point.Ch, point.Fg, point.Bg})
 | |
| 		cs = append(cs, Cell{runes[n], fg, bg})
 | |
| 	}
 | |
| 	return cs
 | |
| }
 | |
| 
 | |
| // Width returns the actual screen space the cell takes (usually 1 or 2).
 | |
| func (c Cell) Width() int {
 | |
| 	return charWidth(c.Ch)
 | |
| }
 | |
| 
 | |
| // Copy return a copy of c
 | |
| func (c Cell) Copy() Cell {
 | |
| 	return c
 | |
| }
 | |
| 
 | |
| // TrimTxCells trims the overflowed text cells sequence.
 | |
| func TrimTxCells(cs []Cell, w int) []Cell {
 | |
| 	if len(cs) <= w {
 | |
| 		return cs
 | |
| 	}
 | |
| 	return cs[:w]
 | |
| }
 | |
| 
 | |
| // DTrimTxCls trims the overflowed text cells sequence and append dots at the end.
 | |
| func DTrimTxCls(cs []Cell, w int) []Cell {
 | |
| 	l := len(cs)
 | |
| 	if l <= 0 {
 | |
| 		return []Cell{}
 | |
| 	}
 | |
| 
 | |
| 	rt := make([]Cell, 0, w)
 | |
| 	csw := 0
 | |
| 	for i := 0; i < l && csw <= w; i++ {
 | |
| 		c := cs[i]
 | |
| 		cw := c.Width()
 | |
| 
 | |
| 		if cw+csw < w {
 | |
| 			rt = append(rt, c)
 | |
| 			csw += cw
 | |
| 		} else {
 | |
| 			rt = append(rt, Cell{'…', c.Fg, c.Bg})
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return rt
 | |
| }
 | |
| 
 | |
| func CellsToStr(cs []Cell) string {
 | |
| 	str := ""
 | |
| 	for _, c := range cs {
 | |
| 		str += string(c.Ch)
 | |
| 	}
 | |
| 	return str
 | |
| }
 |