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] }