2020-11-04 16:42:44 +03:00
|
|
|
package progress
|
|
|
|
|
|
|
|
import (
|
2020-11-18 22:27:41 +03:00
|
|
|
"fmt"
|
2020-11-04 16:42:44 +03:00
|
|
|
"strings"
|
2021-07-08 20:02:52 +03:00
|
|
|
"sync"
|
|
|
|
"time"
|
2020-11-04 16:42:44 +03:00
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
tea "github.com/charmbracelet/bubbletea"
|
|
|
|
"github.com/fogleman/ease"
|
2020-11-04 16:42:44 +03:00
|
|
|
"github.com/lucasb-eyer/go-colorful"
|
2020-11-18 22:27:41 +03:00
|
|
|
"github.com/muesli/reflow/ansi"
|
2020-11-04 16:42:44 +03:00
|
|
|
"github.com/muesli/termenv"
|
|
|
|
)
|
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
// Internal ID management. Used during animating to assure that frame messages
|
|
|
|
// can only be received by progress components that sent them.
|
|
|
|
var (
|
|
|
|
lastID int
|
|
|
|
idMtx sync.Mutex
|
|
|
|
)
|
|
|
|
|
|
|
|
// Return the next ID we should use on the model.
|
|
|
|
func nextID() int {
|
|
|
|
idMtx.Lock()
|
|
|
|
defer idMtx.Unlock()
|
|
|
|
lastID++
|
|
|
|
return lastID
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
defaultWidth = 40
|
|
|
|
defaultFPS = time.Second / 60
|
|
|
|
defaultTransitionDuration = time.Millisecond * 350
|
|
|
|
)
|
2020-11-19 02:38:34 +03:00
|
|
|
|
2020-11-18 23:59:10 +03:00
|
|
|
var color func(string) termenv.Color = termenv.ColorProfile().Color
|
|
|
|
|
2020-11-19 00:20:00 +03:00
|
|
|
// Option is used to set options in NewModel. For example:
|
|
|
|
//
|
|
|
|
// progress := NewModel(
|
|
|
|
// WithRamp("#ff0000", "#0000ff"),
|
2020-11-19 01:47:58 +03:00
|
|
|
// WithoutPercentage(),
|
2020-11-19 00:20:00 +03:00
|
|
|
// )
|
2021-07-08 20:02:52 +03:00
|
|
|
//
|
2020-11-19 02:33:45 +03:00
|
|
|
type Option func(*Model) error
|
2020-11-19 00:20:00 +03:00
|
|
|
|
2020-11-20 18:47:01 +03:00
|
|
|
// WithDefaultGradient sets a gradient fill with default colors.
|
|
|
|
func WithDefaultGradient() Option {
|
2020-12-02 01:01:08 +03:00
|
|
|
return WithGradient("#5A56E0", "#EE6FF8")
|
2020-11-19 00:20:00 +03:00
|
|
|
}
|
|
|
|
|
2020-11-20 18:47:01 +03:00
|
|
|
// WithGradient sets a gradient fill blending between two colors.
|
|
|
|
func WithGradient(colorA, colorB string) Option {
|
2020-11-19 02:33:45 +03:00
|
|
|
return func(m *Model) error {
|
|
|
|
return m.setRamp(colorA, colorB, false)
|
2020-11-19 00:20:00 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-20 18:47:01 +03:00
|
|
|
// WithDefaultScaledGradient sets a gradient with default colors, and scales the
|
2020-11-19 02:04:06 +03:00
|
|
|
// gradient to fit the filled portion of the ramp.
|
2020-11-20 18:47:01 +03:00
|
|
|
func WithDefaultScaledGradient() Option {
|
2020-12-02 01:01:08 +03:00
|
|
|
return WithScaledGradient("#5A56E0", "#EE6FF8")
|
2020-11-19 02:04:06 +03:00
|
|
|
}
|
|
|
|
|
2020-11-20 18:47:01 +03:00
|
|
|
// WithScaledGradient scales the gradient to fit the width of the filled portion of
|
2020-11-19 01:47:58 +03:00
|
|
|
// the progress bar.
|
2020-11-20 18:47:01 +03:00
|
|
|
func WithScaledGradient(colorA, colorB string) Option {
|
2020-11-19 02:33:45 +03:00
|
|
|
return func(m *Model) error {
|
|
|
|
return m.setRamp(colorA, colorB, true)
|
2020-11-19 02:04:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-19 02:38:34 +03:00
|
|
|
// WithSolidFill sets the progress to use a solid fill with the given color.
|
2020-11-19 02:04:06 +03:00
|
|
|
func WithSolidFill(color string) Option {
|
2020-11-19 02:33:45 +03:00
|
|
|
return func(m *Model) error {
|
2020-11-19 02:04:06 +03:00
|
|
|
m.FullColor = color
|
|
|
|
m.useRamp = false
|
2020-11-19 02:33:45 +03:00
|
|
|
return nil
|
2020-11-19 01:47:58 +03:00
|
|
|
}
|
|
|
|
}
|
2020-11-04 16:42:44 +03:00
|
|
|
|
2020-11-19 01:47:58 +03:00
|
|
|
// WithoutPercentage hides the numeric percentage.
|
|
|
|
func WithoutPercentage() Option {
|
2020-11-19 02:33:45 +03:00
|
|
|
return func(m *Model) error {
|
2020-11-19 01:47:58 +03:00
|
|
|
m.ShowPercentage = false
|
2020-11-19 02:33:45 +03:00
|
|
|
return nil
|
2020-11-19 01:47:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-19 02:26:30 +03:00
|
|
|
// WithWidth sets the initial width of the progress bar. Note that you can also
|
|
|
|
// set the width via the Width property, which can come in handy if you're
|
|
|
|
// waiting for a tea.WindowSizeMsg.
|
|
|
|
func WithWidth(w int) Option {
|
2020-11-19 02:33:45 +03:00
|
|
|
return func(m *Model) error {
|
2020-11-19 02:26:30 +03:00
|
|
|
m.Width = w
|
2020-11-19 02:33:45 +03:00
|
|
|
return nil
|
2020-11-19 02:26:30 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
// FrameMsg indicates that an animation step should occur.
|
|
|
|
type FrameMsg struct {
|
|
|
|
id int
|
|
|
|
}
|
|
|
|
|
2020-11-19 02:17:27 +03:00
|
|
|
// Model stores values we'll use when rendering the progress bar.
|
2020-11-19 01:47:58 +03:00
|
|
|
type Model struct {
|
2021-07-08 20:02:52 +03:00
|
|
|
// The internal identifier for this model.
|
|
|
|
id int
|
2020-11-04 16:42:44 +03:00
|
|
|
|
2020-11-19 00:20:00 +03:00
|
|
|
// Total width of the progress bar, including percentage, if set.
|
2020-11-05 14:41:53 +03:00
|
|
|
Width int
|
2020-11-04 16:42:44 +03:00
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
// "Filled" sections of the progress bar.
|
2020-11-19 01:47:58 +03:00
|
|
|
Full rune
|
|
|
|
FullColor string
|
2020-11-04 16:42:44 +03:00
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
// "Empty" sections of progress bar.
|
2020-11-19 01:47:58 +03:00
|
|
|
Empty rune
|
|
|
|
EmptyColor string
|
2020-11-04 16:42:44 +03:00
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
// Settings for rendering the numeric percentage.
|
2020-11-19 02:49:26 +03:00
|
|
|
ShowPercentage bool
|
|
|
|
PercentFormat string // a fmt string for a float
|
|
|
|
PercentageStyle *termenv.Style
|
2020-11-18 22:27:41 +03:00
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
// Animation options.
|
|
|
|
FPS time.Duration
|
|
|
|
TransitionDuration time.Duration
|
|
|
|
|
|
|
|
// Values for the internal animation state.
|
|
|
|
progress float64
|
|
|
|
startPercent float64
|
|
|
|
endPercent float64
|
|
|
|
transitionStart time.Time
|
|
|
|
|
|
|
|
// Gradient settings
|
2020-11-19 01:47:58 +03:00
|
|
|
useRamp bool
|
|
|
|
rampColorA colorful.Color
|
|
|
|
rampColorB colorful.Color
|
2020-11-19 00:20:00 +03:00
|
|
|
|
2020-11-19 01:47:58 +03:00
|
|
|
// When true, we scale the gradient to fit the width of the filled section
|
|
|
|
// of the progress bar. When false, the width of the gradient will be set
|
|
|
|
// to the full width of the progress bar.
|
|
|
|
scaleRamp bool
|
2020-11-04 16:42:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewModel returns a model with default values.
|
2021-07-08 20:02:52 +03:00
|
|
|
func NewModel(opts ...Option) (Model, error) {
|
|
|
|
m := Model{
|
|
|
|
id: nextID(),
|
|
|
|
Width: defaultWidth,
|
|
|
|
Full: '█',
|
|
|
|
FullColor: "#7571F9",
|
|
|
|
Empty: '░',
|
|
|
|
EmptyColor: "#606060",
|
|
|
|
ShowPercentage: true,
|
|
|
|
PercentFormat: " %3.0f%%",
|
|
|
|
FPS: defaultFPS,
|
|
|
|
TransitionDuration: defaultTransitionDuration,
|
2020-11-04 16:42:44 +03:00
|
|
|
}
|
2020-11-19 00:20:00 +03:00
|
|
|
|
|
|
|
for _, opt := range opts {
|
2021-07-08 20:02:52 +03:00
|
|
|
if err := opt(&m); err != nil {
|
|
|
|
return Model{}, err
|
2020-11-19 02:33:45 +03:00
|
|
|
}
|
2020-11-19 00:20:00 +03:00
|
|
|
}
|
|
|
|
|
2020-11-19 02:33:45 +03:00
|
|
|
return m, nil
|
2020-11-04 16:42:44 +03:00
|
|
|
}
|
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
// View renders the an animated progress bar in its current state. To render
|
|
|
|
// a static progress bar based on your own calculations use ViewAs instead.
|
|
|
|
func (m Model) View() string {
|
|
|
|
return m.ViewAs(m.progress)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ViewAs renders the progress bar with a given percentage.
|
|
|
|
func (m Model) ViewAs(percent float64) string {
|
2020-11-19 02:17:27 +03:00
|
|
|
b := strings.Builder{}
|
2020-11-19 01:47:58 +03:00
|
|
|
if m.ShowPercentage {
|
2021-03-12 03:43:13 +03:00
|
|
|
percentage := fmt.Sprintf(m.PercentFormat, percent*100) //nolint:gomnd
|
2020-11-19 02:49:26 +03:00
|
|
|
if m.PercentageStyle != nil {
|
|
|
|
percentage = m.PercentageStyle.Styled(percentage)
|
|
|
|
}
|
|
|
|
m.bar(&b, percent, ansi.PrintableRuneWidth(percentage))
|
|
|
|
b.WriteString(percentage)
|
2020-11-19 02:17:27 +03:00
|
|
|
} else {
|
|
|
|
m.bar(&b, percent, 0)
|
2020-11-18 22:27:41 +03:00
|
|
|
}
|
2020-11-19 02:17:27 +03:00
|
|
|
return b.String()
|
2020-11-18 22:27:41 +03:00
|
|
|
}
|
|
|
|
|
2021-07-08 20:02:52 +03:00
|
|
|
func (m Model) Update(msg tea.Msg) (Model, tea.Cmd) {
|
|
|
|
switch msg := msg.(type) {
|
|
|
|
case FrameMsg:
|
|
|
|
if msg.id != m.id {
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
elapsed := time.Since(m.transitionStart)
|
|
|
|
totalDuration := m.transitionStart.
|
|
|
|
Add(m.TransitionDuration).
|
|
|
|
Sub(m.transitionStart)
|
|
|
|
|
|
|
|
f := func(d time.Duration) float64 { return float64(int64(d)) }
|
|
|
|
|
|
|
|
segmentProgress := f(elapsed) / f(totalDuration)
|
|
|
|
if segmentProgress >= 1.0 {
|
|
|
|
m.progress = m.endPercent
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
segmentSize := m.endPercent - m.startPercent
|
|
|
|
m.progress = ease.OutQuad(segmentProgress)*segmentSize + m.startPercent
|
|
|
|
return m, m.nextFrame()
|
|
|
|
|
|
|
|
default:
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Model) SetPercent(p float64) tea.Cmd {
|
|
|
|
m.startPercent = m.endPercent
|
|
|
|
m.endPercent = p
|
|
|
|
m.transitionStart = time.Now()
|
|
|
|
return m.nextFrame()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m Model) nextFrame() tea.Cmd {
|
|
|
|
return tea.Tick(m.FPS, func(time.Time) tea.Msg {
|
|
|
|
return FrameMsg{m.id}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-11-19 02:17:27 +03:00
|
|
|
func (m Model) bar(b *strings.Builder, percent float64, textWidth int) {
|
2020-11-19 01:47:58 +03:00
|
|
|
var (
|
|
|
|
tw = m.Width - textWidth // total width
|
|
|
|
fw = int(float64(tw) * percent) // filled width
|
|
|
|
p float64
|
|
|
|
)
|
|
|
|
|
|
|
|
if m.useRamp {
|
|
|
|
// Gradient fill
|
|
|
|
for i := 0; i < fw; i++ {
|
|
|
|
if m.scaleRamp {
|
|
|
|
p = float64(i) / float64(fw)
|
|
|
|
} else {
|
|
|
|
p = float64(i) / float64(tw)
|
|
|
|
}
|
|
|
|
c := m.rampColorA.BlendLuv(m.rampColorB, p).Hex()
|
|
|
|
b.WriteString(termenv.
|
|
|
|
String(string(m.Full)).
|
|
|
|
Foreground(color(c)).
|
|
|
|
String(),
|
|
|
|
)
|
2020-11-04 16:42:44 +03:00
|
|
|
}
|
2020-11-19 01:47:58 +03:00
|
|
|
} else {
|
|
|
|
// Solid fill
|
|
|
|
s := termenv.String(string(m.Full)).Foreground(color(m.FullColor)).String()
|
|
|
|
b.WriteString(strings.Repeat(s, fw))
|
2020-11-04 16:42:44 +03:00
|
|
|
}
|
|
|
|
|
2020-11-19 01:47:58 +03:00
|
|
|
// Empty fill
|
|
|
|
e := termenv.String(string(m.Empty)).Foreground(color(m.EmptyColor)).String()
|
|
|
|
b.WriteString(strings.Repeat(e, tw-fw))
|
2020-11-04 16:42:44 +03:00
|
|
|
}
|
2020-11-19 02:04:06 +03:00
|
|
|
|
2020-11-19 02:33:45 +03:00
|
|
|
func (m *Model) setRamp(colorA, colorB string, scaled bool) error {
|
|
|
|
a, err := colorful.Hex(colorA)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := colorful.Hex(colorB)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-11-19 02:04:06 +03:00
|
|
|
m.useRamp = true
|
|
|
|
m.scaleRamp = scaled
|
|
|
|
m.rampColorA = a
|
|
|
|
m.rampColorB = b
|
2020-11-19 02:33:45 +03:00
|
|
|
return nil
|
2020-11-19 02:04:06 +03:00
|
|
|
}
|