1
0
mirror of https://github.com/mum4k/termdash.git synced 2025-04-25 13:48:50 +08:00

Constructor of Gauge now returns an error.

This commit is contained in:
Jakub Sobon 2019-02-14 23:56:06 -05:00
parent 1ab50bcefc
commit a6f5d69569
No known key found for this signature in database
GPG Key ID: F2451A77FB05D3B7
5 changed files with 144 additions and 85 deletions

View File

@ -77,6 +77,11 @@ func layout(ctx context.Context, t terminalapi.Terminal) (*container.Container,
), ),
} }
g, err := newGauge(ctx)
if err != nil {
return nil, err
}
heartLC, err := newHeartbeat(ctx) heartLC, err := newHeartbeat(ctx)
if err != nil { if err != nil {
return nil, err return nil, err
@ -87,7 +92,7 @@ func layout(ctx context.Context, t terminalapi.Terminal) (*container.Container,
container.Border(draw.LineStyleLight), container.Border(draw.LineStyleLight),
container.BorderTitle("A Gauge"), container.BorderTitle("A Gauge"),
container.BorderColor(cell.ColorNumber(39)), container.BorderColor(cell.ColorNumber(39)),
container.PlaceWidget(newGauge(ctx)), container.PlaceWidget(g),
), ),
container.Bottom( container.Bottom(
container.Border(draw.LineStyleLight), container.Border(draw.LineStyleLight),
@ -285,8 +290,11 @@ func newSparkLines(ctx context.Context) (*sparkline.SparkLine, *sparkline.SparkL
} }
// newGauge creates a demo Gauge widget. // newGauge creates a demo Gauge widget.
func newGauge(ctx context.Context) *gauge.Gauge { func newGauge(ctx context.Context) (*gauge.Gauge, error) {
g := gauge.New() g, err := gauge.New()
if err != nil {
return nil, err
}
const start = 35 const start = 35
progress := start progress := start
@ -301,7 +309,7 @@ func newGauge(ctx context.Context) *gauge.Gauge {
} }
return nil return nil
}) })
return g return g, nil
} }
// newDonut creates a demo Donut widget. // newDonut creates a demo Donut widget.

View File

@ -77,14 +77,18 @@ type Gauge struct {
} }
// New returns a new Gauge. // New returns a new Gauge.
func New(opts ...Option) *Gauge { func New(opts ...Option) (*Gauge, error) {
opt := newOptions() opt := newOptions()
for _, o := range opts { for _, o := range opts {
o.set(opt) o.set(opt)
} }
if err := opt.validate(); err != nil {
return nil, err
}
return &Gauge{ return &Gauge{
opts: opt, opts: opt,
} }, nil
} }
// Absolute sets the progress in absolute numbers, i.e. 7 out of 10. // Absolute sets the progress in absolute numbers, i.e. 7 out of 10.

View File

@ -45,20 +45,31 @@ type absoluteCall struct {
func TestGauge(t *testing.T) { func TestGauge(t *testing.T) {
tests := []struct { tests := []struct {
desc string desc string
gauge *Gauge opts []Option
percent *percentCall // if set, the test case calls Gauge.Percent(). percent *percentCall // if set, the test case calls Gauge.Percent().
absolute *absoluteCall // if set the test case calls Gauge.Absolute(). absolute *absoluteCall // if set the test case calls Gauge.Absolute().
canvas image.Rectangle canvas image.Rectangle
opts []Option
want func(size image.Point) *faketerm.Terminal want func(size image.Point) *faketerm.Terminal
wantErr bool
wantUpdateErr bool // whether to expect an error on a call to Gauge.Percent() or Gauge.Absolute(). wantUpdateErr bool // whether to expect an error on a call to Gauge.Percent() or Gauge.Absolute().
wantDrawErr bool wantDrawErr bool
}{ }{
{
desc: "fails on negative height",
opts: []Option{
Height(-1),
},
canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal {
return faketerm.MustNew(size)
},
wantErr: true,
},
{ {
desc: "gauge showing percentage", desc: "gauge showing percentage",
gauge: New( opts: []Option{
Char('o'), Char('o'),
), },
percent: &percentCall{p: 35}, percent: &percentCall{p: 35},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -76,10 +87,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "draws resize needed character when canvas is smaller than requested", desc: "draws resize needed character when canvas is smaller than requested",
gauge: New( opts: []Option{
Char('o'), Char('o'),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
), },
percent: &percentCall{p: 35}, percent: &percentCall{p: 35},
canvas: image.Rect(0, 0, 1, 1), canvas: image.Rect(0, 0, 1, 1),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -93,11 +104,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "aligns the progress text top and left", desc: "aligns the progress text top and left",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HorizontalTextAlign(align.HorizontalLeft), HorizontalTextAlign(align.HorizontalLeft),
VerticalTextAlign(align.VerticalTop), VerticalTextAlign(align.VerticalTop),
), },
percent: &percentCall{p: 0}, percent: &percentCall{p: 0},
canvas: image.Rect(0, 0, 10, 4), canvas: image.Rect(0, 0, 10, 4),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -111,12 +122,12 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "aligns the progress text top and left with border", desc: "aligns the progress text top and left with border",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HorizontalTextAlign(align.HorizontalLeft), HorizontalTextAlign(align.HorizontalLeft),
VerticalTextAlign(align.VerticalTop), VerticalTextAlign(align.VerticalTop),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
), },
percent: &percentCall{p: 0}, percent: &percentCall{p: 0},
canvas: image.Rect(0, 0, 10, 4), canvas: image.Rect(0, 0, 10, 4),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -131,11 +142,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "aligns the progress text bottom and right", desc: "aligns the progress text bottom and right",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HorizontalTextAlign(align.HorizontalRight), HorizontalTextAlign(align.HorizontalRight),
VerticalTextAlign(align.VerticalBottom), VerticalTextAlign(align.VerticalBottom),
), },
percent: &percentCall{p: 0}, percent: &percentCall{p: 0},
canvas: image.Rect(0, 0, 10, 4), canvas: image.Rect(0, 0, 10, 4),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -149,12 +160,12 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "aligns the progress text bottom and right with border", desc: "aligns the progress text bottom and right with border",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HorizontalTextAlign(align.HorizontalRight), HorizontalTextAlign(align.HorizontalRight),
VerticalTextAlign(align.VerticalBottom), VerticalTextAlign(align.VerticalBottom),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
), },
percent: &percentCall{p: 0}, percent: &percentCall{p: 0},
canvas: image.Rect(0, 0, 10, 4), canvas: image.Rect(0, 0, 10, 4),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -169,11 +180,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge showing percentage with border", desc: "gauge showing percentage with border",
gauge: New( opts: []Option{
Char('o'), Char('o'),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
BorderTitle("title"), BorderTitle("title"),
), },
percent: &percentCall{p: 35}, percent: &percentCall{p: 35},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -194,12 +205,12 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "respects border options", desc: "respects border options",
gauge: New( opts: []Option{
Char('o'), Char('o'),
Border(draw.LineStyleLight, cell.FgColor(cell.ColorBlue)), Border(draw.LineStyleLight, cell.FgColor(cell.ColorBlue)),
BorderTitle("title"), BorderTitle("title"),
BorderTitleAlign(align.HorizontalRight), BorderTitleAlign(align.HorizontalRight),
), },
percent: &percentCall{p: 35}, percent: &percentCall{p: 35},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -222,9 +233,9 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge showing zero percentage", desc: "gauge showing zero percentage",
gauge: New( opts: []Option{
Char('o'), Char('o'),
), },
percent: &percentCall{}, percent: &percentCall{},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -238,9 +249,9 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge showing 100 percent", desc: "gauge showing 100 percent",
gauge: New( opts: []Option{
Char('o'), Char('o'),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -260,10 +271,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge showing 100 percent with border", desc: "gauge showing 100 percent with border",
gauge: New( opts: []Option{
Char('o'), Char('o'),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -284,9 +295,9 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge showing absolute progress", desc: "gauge showing absolute progress",
gauge: New( opts: []Option{
Char('o'), Char('o'),
), },
absolute: &absoluteCall{done: 20, total: 100}, absolute: &absoluteCall{done: 20, total: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -304,10 +315,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge without text progress", desc: "gauge without text progress",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
), },
percent: &percentCall{p: 35}, percent: &percentCall{p: 35},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -324,10 +335,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "passing option to Percent() overrides one provided to New()", desc: "passing option to Percent() overrides one provided to New()",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
), },
percent: &percentCall{p: 35, opts: []Option{ShowTextProgress()}}, percent: &percentCall{p: 35, opts: []Option{ShowTextProgress()}},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -345,10 +356,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "passing option to Absolute() overrides one provided to New()", desc: "passing option to Absolute() overrides one provided to New()",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
), },
absolute: &absoluteCall{done: 20, total: 100, opts: []Option{ShowTextProgress()}}, absolute: &absoluteCall{done: 20, total: 100, opts: []Option{ShowTextProgress()}},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -366,10 +377,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge takes full size of the canvas", desc: "gauge takes full size of the canvas",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 5, 2), canvas: image.Rect(0, 0, 5, 2),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -386,11 +397,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge with text label, half-width runes", desc: "gauge with text label, half-width runes",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
TextLabel("label"), TextLabel("label"),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -410,11 +421,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge with text label, full-width runes", desc: "gauge with text label, full-width runes",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
TextLabel("你好"), TextLabel("你好"),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -434,11 +445,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge with text label, full-width runes, gauge falls on rune boundary", desc: "gauge with text label, full-width runes, gauge falls on rune boundary",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
TextLabel("你好"), TextLabel("你好"),
), },
percent: &percentCall{p: 50}, percent: &percentCall{p: 50},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -461,11 +472,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge with text label, full-width runes, gauge extended to cover full rune", desc: "gauge with text label, full-width runes, gauge extended to cover full rune",
gauge: New( opts: []Option{
Char('o'), Char('o'),
HideTextProgress(), HideTextProgress(),
TextLabel("你好"), TextLabel("你好"),
), },
percent: &percentCall{p: 40}, percent: &percentCall{p: 40},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -488,10 +499,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "gauge with progress text and text label", desc: "gauge with progress text and text label",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("l"), TextLabel("l"),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -511,12 +522,12 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "text fully outside of gauge respects EmptyTextColor", desc: "text fully outside of gauge respects EmptyTextColor",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("l"), TextLabel("l"),
EmptyTextColor(cell.ColorMagenta), EmptyTextColor(cell.ColorMagenta),
FilledTextColor(cell.ColorBlue), FilledTextColor(cell.ColorBlue),
), },
percent: &percentCall{p: 10}, percent: &percentCall{p: 10},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -536,12 +547,12 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "text fully inside of gauge respects FilledTextColor", desc: "text fully inside of gauge respects FilledTextColor",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("l"), TextLabel("l"),
EmptyTextColor(cell.ColorMagenta), EmptyTextColor(cell.ColorMagenta),
FilledTextColor(cell.ColorBlue), FilledTextColor(cell.ColorBlue),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -561,12 +572,12 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "part of the text is inside and part outside of gauge", desc: "part of the text is inside and part outside of gauge",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("l"), TextLabel("l"),
EmptyTextColor(cell.ColorMagenta), EmptyTextColor(cell.ColorMagenta),
FilledTextColor(cell.ColorBlue), FilledTextColor(cell.ColorBlue),
), },
percent: &percentCall{p: 50}, percent: &percentCall{p: 50},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -589,10 +600,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "truncates text that is outside of gauge", desc: "truncates text that is outside of gauge",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("long label"), TextLabel("long label"),
), },
percent: &percentCall{p: 0}, percent: &percentCall{p: 0},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -608,11 +619,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "truncates text that is outside of gauge when drawn with border", desc: "truncates text that is outside of gauge when drawn with border",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("long label"), TextLabel("long label"),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
), },
percent: &percentCall{p: 0}, percent: &percentCall{p: 0},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -629,10 +640,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "truncates text that is inside of gauge", desc: "truncates text that is inside of gauge",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("long label"), TextLabel("long label"),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -652,11 +663,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "truncates text that is inside of gauge when drawn with border", desc: "truncates text that is inside of gauge when drawn with border",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("long label"), TextLabel("long label"),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
), },
percent: &percentCall{p: 100}, percent: &percentCall{p: 100},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -677,10 +688,10 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "truncates text that is inside and outside of gauge", desc: "truncates text that is inside and outside of gauge",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("long label"), TextLabel("long label"),
), },
percent: &percentCall{p: 50}, percent: &percentCall{p: 50},
canvas: image.Rect(0, 0, 10, 3), canvas: image.Rect(0, 0, 10, 3),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -703,11 +714,11 @@ func TestGauge(t *testing.T) {
}, },
{ {
desc: "truncates text that is inside and outside of gauge with border", desc: "truncates text that is inside and outside of gauge with border",
gauge: New( opts: []Option{
Char('o'), Char('o'),
TextLabel("long label"), TextLabel("long label"),
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
), },
percent: &percentCall{p: 50}, percent: &percentCall{p: 50},
canvas: image.Rect(0, 0, 10, 4), canvas: image.Rect(0, 0, 10, 4),
want: func(size image.Point) *faketerm.Terminal { want: func(size image.Point) *faketerm.Terminal {
@ -733,6 +744,14 @@ func TestGauge(t *testing.T) {
for _, tc := range tests { for _, tc := range tests {
t.Run(tc.desc, func(t *testing.T) { t.Run(tc.desc, func(t *testing.T) {
g, err := New(tc.opts...)
if (err != nil) != tc.wantErr {
t.Errorf("New => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
return
}
c, err := canvas.New(tc.canvas) c, err := canvas.New(tc.canvas)
if err != nil { if err != nil {
t.Fatalf("canvas.New => unexpected error: %v", err) t.Fatalf("canvas.New => unexpected error: %v", err)
@ -740,7 +759,7 @@ func TestGauge(t *testing.T) {
switch { switch {
case tc.percent != nil: case tc.percent != nil:
err := tc.gauge.Percent(tc.percent.p, tc.percent.opts...) err := g.Percent(tc.percent.p, tc.percent.opts...)
if (err != nil) != tc.wantUpdateErr { if (err != nil) != tc.wantUpdateErr {
t.Errorf("Percent => unexpected error: %v, wantUpdateErr: %v", err, tc.wantUpdateErr) t.Errorf("Percent => unexpected error: %v, wantUpdateErr: %v", err, tc.wantUpdateErr)
} }
@ -749,7 +768,7 @@ func TestGauge(t *testing.T) {
} }
case tc.absolute != nil: case tc.absolute != nil:
err := tc.gauge.Absolute(tc.absolute.done, tc.absolute.total, tc.absolute.opts...) err := g.Absolute(tc.absolute.done, tc.absolute.total, tc.absolute.opts...)
if (err != nil) != tc.wantUpdateErr { if (err != nil) != tc.wantUpdateErr {
t.Errorf("Absolute => unexpected error: %v, wantUpdateErr: %v", err, tc.wantUpdateErr) t.Errorf("Absolute => unexpected error: %v, wantUpdateErr: %v", err, tc.wantUpdateErr)
} }
@ -759,7 +778,7 @@ func TestGauge(t *testing.T) {
} }
err = tc.gauge.Draw(c) err = g.Draw(c)
if (err != nil) != tc.wantDrawErr { if (err != nil) != tc.wantDrawErr {
t.Errorf("Draw => unexpected error: %v, wantDrawErr: %v", err, tc.wantDrawErr) t.Errorf("Draw => unexpected error: %v, wantDrawErr: %v", err, tc.wantDrawErr)
} }
@ -785,13 +804,12 @@ func TestGauge(t *testing.T) {
func TestOptions(t *testing.T) { func TestOptions(t *testing.T) {
tests := []struct { tests := []struct {
desc string desc string
gauge *Gauge opts []Option
want widgetapi.Options want widgetapi.Options
}{ }{
{ {
desc: "reports correct minimum and maximum size", desc: "reports correct minimum and maximum size",
gauge: New(),
want: widgetapi.Options{ want: widgetapi.Options{
MaximumSize: image.Point{0, 0}, // Unlimited. MaximumSize: image.Point{0, 0}, // Unlimited.
MinimumSize: image.Point{1, 1}, MinimumSize: image.Point{1, 1},
@ -801,9 +819,9 @@ func TestOptions(t *testing.T) {
}, },
{ {
desc: "maximum size is limited when height is specified", desc: "maximum size is limited when height is specified",
gauge: New( opts: []Option{
Height(2), Height(2),
), },
want: widgetapi.Options{ want: widgetapi.Options{
MaximumSize: image.Point{0, 2}, MaximumSize: image.Point{0, 2},
MinimumSize: image.Point{1, 1}, MinimumSize: image.Point{1, 1},
@ -813,10 +831,10 @@ func TestOptions(t *testing.T) {
}, },
{ {
desc: "border is accounted for in maximum and minimum size", desc: "border is accounted for in maximum and minimum size",
gauge: New( opts: []Option{
Border(draw.LineStyleLight), Border(draw.LineStyleLight),
Height(2), Height(2),
), },
want: widgetapi.Options{ want: widgetapi.Options{
MaximumSize: image.Point{0, 4}, MaximumSize: image.Point{0, 4},
MinimumSize: image.Point{3, 3}, MinimumSize: image.Point{3, 3},
@ -828,7 +846,11 @@ func TestOptions(t *testing.T) {
for _, tc := range tests { for _, tc := range tests {
t.Run(tc.desc, func(t *testing.T) { t.Run(tc.desc, func(t *testing.T) {
got := tc.gauge.Options() g, err := New(tc.opts...)
if err != nil {
t.Fatalf("New => unexpected error: %v", err)
}
got := g.Options()
if diff := pretty.Compare(tc.want, got); diff != "" { if diff := pretty.Compare(tc.want, got); diff != "" {
t.Errorf("Options => unexpected diff (-want, +got):\n%s", diff) t.Errorf("Options => unexpected diff (-want, +got):\n%s", diff)

View File

@ -86,33 +86,48 @@ func main() {
defer t.Close() defer t.Close()
ctx, cancel := context.WithCancel(context.Background()) ctx, cancel := context.WithCancel(context.Background())
slim := gauge.New( slim, err := gauge.New(
gauge.Height(1), gauge.Height(1),
gauge.Border(draw.LineStyleLight), gauge.Border(draw.LineStyleLight),
gauge.BorderTitle("Percentage progress"), gauge.BorderTitle("Percentage progress"),
) )
if err != nil {
panic(err)
}
go playGauge(ctx, slim, 10, 500*time.Millisecond, playTypePercent) go playGauge(ctx, slim, 10, 500*time.Millisecond, playTypePercent)
absolute := gauge.New(
absolute, err := gauge.New(
gauge.Height(1), gauge.Height(1),
gauge.Color(cell.ColorBlue), gauge.Color(cell.ColorBlue),
gauge.Border(draw.LineStyleLight), gauge.Border(draw.LineStyleLight),
gauge.BorderTitle("Absolute progress"), gauge.BorderTitle("Absolute progress"),
) )
if err != nil {
panic(err)
}
go playGauge(ctx, absolute, 17, 500*time.Millisecond, playTypeAbsolute) go playGauge(ctx, absolute, 17, 500*time.Millisecond, playTypeAbsolute)
noProgress := gauge.New(
noProgress, err := gauge.New(
gauge.Height(1), gauge.Height(1),
gauge.Border(draw.LineStyleLight, cell.FgColor(cell.ColorMagenta)), gauge.Border(draw.LineStyleLight, cell.FgColor(cell.ColorMagenta)),
gauge.BorderTitle("Without progress text"), gauge.BorderTitle("Without progress text"),
gauge.HideTextProgress(), gauge.HideTextProgress(),
) )
if err != nil {
panic(err)
}
go playGauge(ctx, noProgress, 5, 250*time.Millisecond, playTypePercent) go playGauge(ctx, noProgress, 5, 250*time.Millisecond, playTypePercent)
withLabel := gauge.New(
withLabel, err := gauge.New(
gauge.Height(3), gauge.Height(3),
gauge.TextLabel("你好,世界! text label and no border"), gauge.TextLabel("你好,世界! text label and no border"),
gauge.Color(cell.ColorRed), gauge.Color(cell.ColorRed),
gauge.FilledTextColor(cell.ColorBlack), gauge.FilledTextColor(cell.ColorBlack),
gauge.EmptyTextColor(cell.ColorYellow), gauge.EmptyTextColor(cell.ColorYellow),
) )
if err != nil {
panic(err)
}
go playGauge(ctx, withLabel, 3, 500*time.Millisecond, playTypePercent) go playGauge(ctx, withLabel, 3, 500*time.Millisecond, playTypePercent)
c, err := container.New( c, err := container.New(

View File

@ -17,6 +17,8 @@ package gauge
// options.go contains configurable options for Gauge. // options.go contains configurable options for Gauge.
import ( import (
"fmt"
"github.com/mum4k/termdash/align" "github.com/mum4k/termdash/align"
"github.com/mum4k/termdash/cell" "github.com/mum4k/termdash/cell"
"github.com/mum4k/termdash/draw" "github.com/mum4k/termdash/draw"
@ -58,6 +60,14 @@ func newOptions() *options {
} }
} }
// validate validates the provided options.
func (o *options) validate() error {
if got, min := o.height, 0; got < min {
return fmt.Errorf("invalid Height %d, must be %d <= Height", got, min)
}
return nil
}
// option implements Option. // option implements Option.
type option func(*options) type option func(*options)