fzf/src/util/util.go

191 lines
3.6 KiB
Go
Raw Normal View History

2015-01-12 03:56:17 +00:00
package util
2015-01-01 19:49:30 +00:00
2015-01-12 03:56:17 +00:00
import (
"math"
2015-01-12 03:56:17 +00:00
"os"
"strings"
2015-01-12 03:56:17 +00:00
"time"
2017-06-01 23:27:17 +00:00
"github.com/mattn/go-isatty"
"github.com/rivo/uniseg"
2015-01-12 03:56:17 +00:00
)
2015-01-08 13:07:04 +00:00
// StringWidth returns string width where each CR/LF character takes 1 column
func StringWidth(s string) int {
return uniseg.StringWidth(s) + strings.Count(s, "\n") + strings.Count(s, "\r")
}
// RunesWidth returns runes width
func RunesWidth(runes []rune, prefixWidth int, tabstop int, limit int) (int, int) {
width := 0
gr := uniseg.NewGraphemes(string(runes))
idx := 0
for gr.Next() {
rs := gr.Runes()
var w int
if len(rs) == 1 && rs[0] == '\t' {
w = tabstop - (prefixWidth+width)%tabstop
} else {
w = StringWidth(string(rs))
}
width += w
if width > limit {
return width, idx
}
idx += len(rs)
2017-01-07 16:30:31 +00:00
}
return width, -1
2017-01-07 16:30:31 +00:00
}
// Truncate returns the truncated runes and its width
func Truncate(input string, limit int) ([]rune, int) {
runes := []rune{}
width := 0
gr := uniseg.NewGraphemes(input)
for gr.Next() {
rs := gr.Runes()
w := StringWidth(string(rs))
if width+w > limit {
return runes, width
}
width += w
runes = append(runes, rs...)
}
return runes, width
}
2015-01-11 18:01:24 +00:00
// Max returns the largest integer
func Max(first int, second int) int {
if first >= second {
return first
2015-01-01 19:49:30 +00:00
}
return second
2015-01-01 19:49:30 +00:00
}
2016-09-07 00:58:18 +00:00
// Max16 returns the largest integer
func Max16(first int16, second int16) int16 {
if first >= second {
return first
}
return second
}
2016-09-07 00:58:18 +00:00
// Max32 returns the largest 32-bit integer
func Max32(first int32, second int32) int32 {
if first > second {
return first
}
return second
}
// Min returns the smallest integer
func Min(first int, second int) int {
if first <= second {
return first
}
return second
}
2016-09-07 00:58:18 +00:00
// Min32 returns the smallest 32-bit integer
func Min32(first int32, second int32) int32 {
if first <= second {
2015-01-08 17:37:08 +00:00
return first
}
return second
}
2015-03-18 16:59:14 +00:00
// Constrain32 limits the given 32-bit integer with the upper and lower bounds
func Constrain32(val int32, min int32, max int32) int32 {
if val < min {
return min
}
if val > max {
return max
}
return val
}
2015-01-12 03:56:17 +00:00
// Constrain limits the given integer with the upper and lower bounds
func Constrain(val int, min int, max int) int {
if val < min {
return min
2015-01-01 19:49:30 +00:00
}
2015-01-12 03:56:17 +00:00
if val > max {
return max
}
return val
2015-01-01 19:49:30 +00:00
}
2015-01-08 13:07:04 +00:00
func AsUint16(val int) uint16 {
if val > math.MaxUint16 {
return math.MaxUint16
} else if val < 0 {
return 0
}
return uint16(val)
}
2015-01-11 18:01:24 +00:00
// DurWithin limits the given time.Duration with the upper and lower bounds
2015-01-08 13:07:04 +00:00
func DurWithin(
val time.Duration, min time.Duration, max time.Duration) time.Duration {
if val < min {
return min
}
if val > max {
return max
}
return val
}
2015-01-12 03:56:17 +00:00
// IsTty returns true if stdin is a terminal
2015-01-12 03:56:17 +00:00
func IsTty() bool {
return isatty.IsTerminal(os.Stdin.Fd())
}
// ToTty returns true if stdout is a terminal
func ToTty() bool {
return isatty.IsTerminal(os.Stdout.Fd())
}
// Once returns a function that returns the specified boolean value only once
func Once(nextResponse bool) func() bool {
state := nextResponse
return func() bool {
prevState := state
state = false
return prevState
}
}
// RepeatToFill repeats the given string to fill the given width
func RepeatToFill(str string, length int, limit int) string {
times := limit / length
rest := limit % length
output := strings.Repeat(str, times)
if rest > 0 {
for _, r := range str {
rest -= uniseg.StringWidth(string(r))
if rest < 0 {
break
}
output += string(r)
if rest == 0 {
break
}
}
}
return output
}
// ToKebabCase converts the given CamelCase string to kebab-case
func ToKebabCase(s string) string {
name := ""
for i, r := range s {
if i > 0 && r >= 'A' && r <= 'Z' {
name += "-"
}
name += string(r)
}
return strings.ToLower(name)
}