2017-02-16 06:21:11 -06:00
|
|
|
// Copyright 2017 Zack Guo <zack.y.guo@gmail.com>. All rights reserved.
|
2015-06-24 10:40:18 -05:00
|
|
|
// Use of this source code is governed by a MIT license that can
|
|
|
|
// be found in the LICENSE file.
|
|
|
|
|
|
|
|
package termui
|
|
|
|
|
2016-02-11 08:16:52 -06:00
|
|
|
import (
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
tm "github.com/nsf/termbox-go"
|
|
|
|
)
|
2015-06-24 10:40:18 -05:00
|
|
|
import rw "github.com/mattn/go-runewidth"
|
|
|
|
|
|
|
|
/* ---------------Port from termbox-go --------------------- */
|
|
|
|
|
|
|
|
// Attribute is printable cell's color and style.
|
|
|
|
type Attribute uint16
|
|
|
|
|
2016-02-11 08:16:52 -06:00
|
|
|
// 8 basic clolrs
|
2015-06-24 10:40:18 -05:00
|
|
|
const (
|
|
|
|
ColorDefault Attribute = iota
|
|
|
|
ColorBlack
|
|
|
|
ColorRed
|
|
|
|
ColorGreen
|
|
|
|
ColorYellow
|
|
|
|
ColorBlue
|
|
|
|
ColorMagenta
|
|
|
|
ColorCyan
|
|
|
|
ColorWhite
|
|
|
|
)
|
|
|
|
|
2016-02-11 08:16:52 -06:00
|
|
|
//Have a constant that defines number of colors
|
|
|
|
const NumberofColors = 8
|
|
|
|
|
|
|
|
// Text style
|
2015-06-24 10:40:18 -05:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2016-02-11 08:16:52 -06:00
|
|
|
// Here for backwards-compatibility.
|
2015-06-24 10:40:18 -05:00
|
|
|
func trimStr2Runes(s string, w int) []rune {
|
2016-02-11 08:16:52 -06:00
|
|
|
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 {
|
2015-06-24 10:40:18 -05:00
|
|
|
if w <= 0 {
|
|
|
|
return []rune{}
|
|
|
|
}
|
2016-02-11 08:16:52 -06:00
|
|
|
|
2015-06-24 10:40:18 -05:00
|
|
|
sw := rw.StringWidth(s)
|
|
|
|
if sw > w {
|
|
|
|
return []rune(rw.Truncate(s, w, dot))
|
|
|
|
}
|
2016-02-11 08:16:52 -06:00
|
|
|
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
|
2015-06-24 10:40:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func strWidth(s string) int {
|
|
|
|
return rw.StringWidth(s)
|
|
|
|
}
|
|
|
|
|
|
|
|
func charWidth(ch rune) int {
|
|
|
|
return rw.RuneWidth(ch)
|
|
|
|
}
|
2016-02-11 08:16:52 -06:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|
2016-10-28 12:05:01 -05:00
|
|
|
|
|
|
|
func CellsToStr(cs []Cell) string {
|
|
|
|
str := ""
|
|
|
|
for _, c := range cs {
|
|
|
|
str += string(c.Ch)
|
|
|
|
}
|
|
|
|
return str
|
|
|
|
}
|