Compare commits
5 Commits
Author | SHA1 | Date | |
---|---|---|---|
4331b2ac29 | |||
![]() |
5be0ec3905 | ||
![]() |
7f027b08ed | ||
![]() |
5949bf193f | ||
![]() |
708c1e5d7a |
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
table_header_test.go
|
2
go.mod
2
go.mod
@ -6,11 +6,13 @@ require (
|
||||
git.milar.in/milarin/gmath v0.0.2
|
||||
git.milar.in/milarin/slices v0.0.3
|
||||
github.com/fatih/color v1.13.0
|
||||
github.com/mattn/go-runewidth v0.0.13
|
||||
)
|
||||
|
||||
require (
|
||||
git.milar.in/milarin/channel v0.0.9 // indirect
|
||||
github.com/mattn/go-colorable v0.1.9 // indirect
|
||||
github.com/mattn/go-isatty v0.0.14 // indirect
|
||||
github.com/rivo/uniseg v0.3.4 // indirect
|
||||
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c // indirect
|
||||
)
|
||||
|
5
go.sum
5
go.sum
@ -11,6 +11,11 @@ github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope
|
||||
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
|
||||
github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y=
|
||||
github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
|
||||
github.com/mattn/go-runewidth v0.0.13 h1:lTGmDsbAYt5DmK6OnoV7EuIF1wEIFAcxld6ypU4OSgU=
|
||||
github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
|
||||
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
|
||||
github.com/rivo/uniseg v0.3.4 h1:3Z3Eu6FGHZWSfNKJTOUiPatWwfc7DzJRU04jFUqJODw=
|
||||
github.com/rivo/uniseg v0.3.4/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
|
||||
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c h1:F1jZWGFhYfh0Ci55sIpILtKKK8p3i2/krTr0H1rg74I=
|
||||
|
85
table.go
85
table.go
@ -6,7 +6,6 @@ import (
|
||||
|
||||
"git.milar.in/milarin/gmath"
|
||||
"git.milar.in/milarin/slices"
|
||||
"github.com/fatih/color"
|
||||
)
|
||||
|
||||
type Table struct {
|
||||
@ -61,31 +60,21 @@ func (t *Table) AddRow(row ...interface{}) {
|
||||
}
|
||||
}
|
||||
|
||||
func (t *Table) String() string {
|
||||
func (t *Table) StringNoHead() string {
|
||||
b := new(strings.Builder)
|
||||
|
||||
b.WriteRune('┌')
|
||||
b.WriteRune('┏')
|
||||
for i, colwidth := range t.colwidth {
|
||||
b.WriteString(strings.Repeat("─", colwidth))
|
||||
b.WriteString(strings.Repeat("━", colwidth))
|
||||
if i < len(t.colwidth)-1 {
|
||||
b.WriteRune('┬')
|
||||
b.WriteRune('┯')
|
||||
}
|
||||
}
|
||||
b.WriteRune('┐')
|
||||
b.WriteRune('┓')
|
||||
b.WriteRune('\n')
|
||||
|
||||
bold := color.New(color.Bold)
|
||||
t.printRow(b, bold.Sprint, t.head, t.headheight)
|
||||
|
||||
if len(t.data) > 0 {
|
||||
t.addNextCellLine(b)
|
||||
} else {
|
||||
t.addLastCellLine(b)
|
||||
return b.String()
|
||||
}
|
||||
|
||||
for i, row := range t.data {
|
||||
t.printRow(b, fmt.Sprint, row, t.rowheight[i])
|
||||
t.printRow(b, row, t.rowheight[i])
|
||||
if t.rowSep && i < len(t.data)-1 {
|
||||
t.addNextCellLine(b)
|
||||
}
|
||||
@ -96,9 +85,46 @@ func (t *Table) String() string {
|
||||
return b.String()
|
||||
}
|
||||
|
||||
func (t *Table) printRow(b *strings.Builder, printFunc func(...interface{}) string, rowData [][]string, rowHeight int) {
|
||||
func (t *Table) String() string {
|
||||
b := new(strings.Builder)
|
||||
|
||||
b.WriteRune('┏')
|
||||
for i, colwidth := range t.colwidth {
|
||||
b.WriteString(strings.Repeat("━", colwidth))
|
||||
if i < len(t.colwidth)-1 {
|
||||
b.WriteRune('┯')
|
||||
}
|
||||
}
|
||||
b.WriteRune('┓')
|
||||
b.WriteRune('\n')
|
||||
|
||||
t.printRow(b, t.head, t.headheight)
|
||||
|
||||
b.WriteRune('┣')
|
||||
for i, colwidth := range t.colwidth {
|
||||
b.WriteString(strings.Repeat("━", colwidth))
|
||||
if i < len(t.colwidth)-1 {
|
||||
b.WriteRune('┿')
|
||||
}
|
||||
}
|
||||
b.WriteRune('┫')
|
||||
b.WriteRune('\n')
|
||||
|
||||
for i, row := range t.data {
|
||||
t.printRow(b, row, t.rowheight[i])
|
||||
if t.rowSep && i < len(t.data)-1 {
|
||||
t.addNextCellLine(b)
|
||||
}
|
||||
}
|
||||
|
||||
t.addLastCellLine(b)
|
||||
|
||||
return b.String()
|
||||
}
|
||||
|
||||
func (t *Table) printRow(b *strings.Builder, rowData [][]string, rowHeight int) {
|
||||
for lineIndex := 0; lineIndex < rowHeight; lineIndex++ {
|
||||
b.WriteRune('│')
|
||||
b.WriteRune('┃')
|
||||
for colIndex := 0; colIndex < t.maxcols; colIndex++ {
|
||||
line := ""
|
||||
|
||||
@ -109,33 +135,34 @@ func (t *Table) printRow(b *strings.Builder, printFunc func(...interface{}) stri
|
||||
}
|
||||
}
|
||||
|
||||
b.WriteString(printFunc(padStringRight(line, ' ', t.colwidth[colIndex])))
|
||||
b.WriteRune('│')
|
||||
b.WriteString(fmt.Sprint(padStringRight(line, ' ', t.colwidth[colIndex])))
|
||||
if colIndex < t.maxcols-1 {
|
||||
b.WriteRune('│')
|
||||
}
|
||||
}
|
||||
b.WriteRune('\n')
|
||||
b.WriteString("┃\n")
|
||||
}
|
||||
}
|
||||
|
||||
func (t *Table) addNextCellLine(b *strings.Builder) {
|
||||
b.WriteRune('├')
|
||||
b.WriteRune('┠')
|
||||
for i, colwidth := range t.colwidth {
|
||||
b.WriteString(strings.Repeat("─", colwidth))
|
||||
if i < len(t.colwidth)-1 {
|
||||
b.WriteRune('┼')
|
||||
}
|
||||
}
|
||||
b.WriteRune('┤')
|
||||
b.WriteRune('┨')
|
||||
b.WriteRune('\n')
|
||||
}
|
||||
|
||||
func (t *Table) addLastCellLine(b *strings.Builder) {
|
||||
b.WriteRune('└')
|
||||
b.WriteRune('┗')
|
||||
for i, colwidth := range t.colwidth {
|
||||
b.WriteString(strings.Repeat("─", colwidth))
|
||||
b.WriteString(strings.Repeat("━", colwidth))
|
||||
if i < len(t.colwidth)-1 {
|
||||
b.WriteRune('┴')
|
||||
b.WriteRune('┷')
|
||||
}
|
||||
}
|
||||
b.WriteRune('┘')
|
||||
b.WriteRune('\n')
|
||||
b.WriteString("┛\n")
|
||||
}
|
||||
|
44
table_header.go
Normal file
44
table_header.go
Normal file
@ -0,0 +1,44 @@
|
||||
package tprint
|
||||
|
||||
import (
|
||||
"strings"
|
||||
|
||||
"git.milar.in/milarin/slices"
|
||||
)
|
||||
|
||||
func FormatHeaderTable(header string, table *Table) string {
|
||||
return formatHeaderTableStr(header, table.String())
|
||||
}
|
||||
|
||||
func FormatHeaderTableNoHead(header string, table *Table) string {
|
||||
return formatHeaderTableStr(header, table.StringNoHead())
|
||||
}
|
||||
|
||||
// TODO formatHeaderTableStr is a quick hack
|
||||
// a better solution would be to support nested tables.
|
||||
// in that case, a header table is just a table with a singe col, single header row and single data row
|
||||
|
||||
func formatHeaderTableStr(header string, tab string) string {
|
||||
b := new(strings.Builder)
|
||||
|
||||
splits := strings.Split(tab, "\n")
|
||||
tabwidth := strLen(splits[0])
|
||||
|
||||
b.WriteRune('┏')
|
||||
b.WriteString(strings.Repeat("━", tabwidth-2))
|
||||
b.WriteString("┓\n┃")
|
||||
b.WriteString(padStringCenter(header, ' ', tabwidth-2))
|
||||
b.WriteString("┃\n")
|
||||
|
||||
b.WriteRune('┣')
|
||||
b.WriteString(splits[0][3 : len(splits[0])-3])
|
||||
b.WriteString("┫\n")
|
||||
|
||||
slices.Each(splits[1:], func(s string) {
|
||||
b.WriteString(s)
|
||||
b.WriteRune('\n')
|
||||
})
|
||||
|
||||
ret := b.String()
|
||||
return ret[:len(ret)-1]
|
||||
}
|
32
utils.go
32
utils.go
@ -3,6 +3,8 @@ package tprint
|
||||
import (
|
||||
"strings"
|
||||
"unicode"
|
||||
|
||||
"github.com/mattn/go-runewidth"
|
||||
)
|
||||
|
||||
func maxLengthStr(a, b string) string {
|
||||
@ -23,20 +25,24 @@ func strLen(str string) int {
|
||||
l := 0
|
||||
|
||||
runes := []rune(str)
|
||||
colored := false
|
||||
|
||||
for i := 0; i < len(runes); i++ {
|
||||
if unicode.IsControl(runes[i]) {
|
||||
if colored {
|
||||
i += 3
|
||||
} else {
|
||||
i += 4
|
||||
rn := runes[i]
|
||||
|
||||
// skip control sequences
|
||||
if unicode.IsControl(rn) {
|
||||
for j := i; j < len(runes)-1 && runes[j] != 'm'; j++ {
|
||||
i = j
|
||||
}
|
||||
colored = !colored
|
||||
i++
|
||||
continue
|
||||
}
|
||||
|
||||
l++
|
||||
if rn <= 0xFF || rn >= '─' && rn <= '╿' {
|
||||
l++
|
||||
} else {
|
||||
l += runewidth.RuneWidth(rn)
|
||||
}
|
||||
}
|
||||
|
||||
return l
|
||||
@ -46,3 +52,13 @@ func padStringRight(str string, pad rune, length int) string {
|
||||
padding := length - strLen(str)
|
||||
return str + strings.Repeat(string(pad), padding)
|
||||
}
|
||||
|
||||
func padStringLeft(str string, pad rune, length int) string {
|
||||
padding := length - strLen(str)
|
||||
return strings.Repeat(string(pad), padding) + str
|
||||
}
|
||||
|
||||
func padStringCenter(str string, pad rune, length int) string {
|
||||
l := strLen(str)
|
||||
return padStringLeft(padStringRight(str, pad, (length-l)/2+l), pad, length)
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user