mirror of
https://github.com/jesseduffield/lazygit.git
synced 2024-12-12 11:15:00 +02:00
e33fe37a99
We've been sometimes using lo and sometimes using my slices package, and we need to pick one for consistency. Lo is more extensive and better maintained so we're going with that. My slices package was a superset of go's own slices package so in some places I've just used the official one (the methods were just wrappers anyway). I've also moved the remaining methods into the utils package.
169 lines
3.9 KiB
Go
169 lines
3.9 KiB
Go
package utils
|
|
|
|
import (
|
|
"strings"
|
|
|
|
"github.com/mattn/go-runewidth"
|
|
"github.com/samber/lo"
|
|
)
|
|
|
|
type Alignment int
|
|
|
|
const (
|
|
AlignLeft Alignment = iota
|
|
AlignRight
|
|
)
|
|
|
|
type ColumnConfig struct {
|
|
Width int
|
|
Alignment Alignment
|
|
}
|
|
|
|
// WithPadding pads a string as much as you want
|
|
func WithPadding(str string, padding int, alignment Alignment) string {
|
|
uncoloredStr := Decolorise(str)
|
|
width := runewidth.StringWidth(uncoloredStr)
|
|
if padding < width {
|
|
return str
|
|
}
|
|
space := strings.Repeat(" ", padding-width)
|
|
if alignment == AlignLeft {
|
|
return str + space
|
|
} else {
|
|
return space + str
|
|
}
|
|
}
|
|
|
|
// defaults to left-aligning each column. If you want to set the alignment of
|
|
// each column, pass in a slice of Alignment values.
|
|
func RenderDisplayStrings(displayStringsArr [][]string, columnAlignments []Alignment) string {
|
|
displayStringsArr = excludeBlankColumns(displayStringsArr)
|
|
padWidths := getPadWidths(displayStringsArr)
|
|
columnConfigs := make([]ColumnConfig, len(padWidths))
|
|
for i, padWidth := range padWidths {
|
|
// gracefully handle when columnAlignments is shorter than padWidths
|
|
alignment := AlignLeft
|
|
if len(columnAlignments) > i {
|
|
alignment = columnAlignments[i]
|
|
}
|
|
|
|
columnConfigs[i] = ColumnConfig{
|
|
Width: padWidth,
|
|
Alignment: alignment,
|
|
}
|
|
}
|
|
output := getPaddedDisplayStrings(displayStringsArr, columnConfigs)
|
|
|
|
return output
|
|
}
|
|
|
|
// NOTE: this mutates the input slice for the sake of performance
|
|
func excludeBlankColumns(displayStringsArr [][]string) [][]string {
|
|
if len(displayStringsArr) == 0 {
|
|
return displayStringsArr
|
|
}
|
|
|
|
// if all rows share a blank column, we want to remove that column
|
|
toRemove := []int{}
|
|
outer:
|
|
for i := range displayStringsArr[0] {
|
|
for _, strings := range displayStringsArr {
|
|
if strings[i] != "" {
|
|
continue outer
|
|
}
|
|
}
|
|
toRemove = append(toRemove, i)
|
|
}
|
|
|
|
if len(toRemove) == 0 {
|
|
return displayStringsArr
|
|
}
|
|
|
|
// remove the columns
|
|
for i, strings := range displayStringsArr {
|
|
for j := len(toRemove) - 1; j >= 0; j-- {
|
|
strings = append(strings[:toRemove[j]], strings[toRemove[j]+1:]...)
|
|
}
|
|
displayStringsArr[i] = strings
|
|
}
|
|
|
|
return displayStringsArr
|
|
}
|
|
|
|
func getPaddedDisplayStrings(stringArrays [][]string, columnConfigs []ColumnConfig) string {
|
|
builder := strings.Builder{}
|
|
for i, stringArray := range stringArrays {
|
|
if len(stringArray) == 0 {
|
|
continue
|
|
}
|
|
for j, columnConfig := range columnConfigs {
|
|
if len(stringArray)-1 < j {
|
|
continue
|
|
}
|
|
builder.WriteString(WithPadding(stringArray[j], columnConfig.Width, columnConfig.Alignment))
|
|
builder.WriteString(" ")
|
|
}
|
|
if len(stringArray)-1 < len(columnConfigs) {
|
|
continue
|
|
}
|
|
builder.WriteString(stringArray[len(columnConfigs)])
|
|
|
|
if i < len(stringArrays)-1 {
|
|
builder.WriteString("\n")
|
|
}
|
|
}
|
|
return builder.String()
|
|
}
|
|
|
|
func getPadWidths(stringArrays [][]string) []int {
|
|
maxWidth := MaxFn(stringArrays, func(stringArray []string) int {
|
|
return len(stringArray)
|
|
})
|
|
|
|
if maxWidth-1 < 0 {
|
|
return []int{}
|
|
}
|
|
return lo.Map(lo.Range(maxWidth-1), func(i int, _ int) int {
|
|
return MaxFn(stringArrays, func(stringArray []string) int {
|
|
uncoloredStr := Decolorise(stringArray[i])
|
|
|
|
return runewidth.StringWidth(uncoloredStr)
|
|
})
|
|
})
|
|
}
|
|
|
|
func MaxFn[T any](items []T, fn func(T) int) int {
|
|
max := 0
|
|
for _, item := range items {
|
|
if fn(item) > max {
|
|
max = fn(item)
|
|
}
|
|
}
|
|
return max
|
|
}
|
|
|
|
// TruncateWithEllipsis returns a string, truncated to a certain length, with an ellipsis
|
|
func TruncateWithEllipsis(str string, limit int) string {
|
|
if runewidth.StringWidth(str) > limit && limit <= 3 {
|
|
return strings.Repeat(".", limit)
|
|
}
|
|
return runewidth.Truncate(str, limit, "...")
|
|
}
|
|
|
|
func SafeTruncate(str string, limit int) string {
|
|
if len(str) > limit {
|
|
return str[0:limit]
|
|
} else {
|
|
return str
|
|
}
|
|
}
|
|
|
|
const COMMIT_HASH_SHORT_SIZE = 8
|
|
|
|
func ShortSha(sha string) string {
|
|
if len(sha) < COMMIT_HASH_SHORT_SIZE {
|
|
return sha
|
|
}
|
|
return sha[:COMMIT_HASH_SHORT_SIZE]
|
|
}
|