1
0
mirror of https://github.com/mum4k/termdash.git synced 2025-04-25 13:48:50 +08:00
termdash/container/draw_test.go
Jakub Sobon 2d4d903f87 Container handles resize correctly.
And making container thread unsafe, thread safety will be implemented at
the top.
2018-04-23 00:45:17 +01:00

535 lines
13 KiB
Go

// Copyright 2018 Google Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package container
import (
"image"
"testing"
"github.com/mum4k/termdash/canvas/testcanvas"
"github.com/mum4k/termdash/cell"
"github.com/mum4k/termdash/draw"
"github.com/mum4k/termdash/draw/testdraw"
"github.com/mum4k/termdash/terminal/faketerm"
"github.com/mum4k/termdash/widgetapi"
"github.com/mum4k/termdash/widgets/fakewidget"
)
func TestDrawWidget(t *testing.T) {
tests := []struct {
desc string
termSize image.Point
container func(ft *faketerm.Terminal) *Container
want func(size image.Point) *faketerm.Terminal
wantErr bool
}{
{
desc: "draws widget with container border",
termSize: image.Point{9, 5},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
PlaceWidget(fakewidget.New(widgetapi.Options{})),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(1, 1, 8, 4), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{2, 2}}
testdraw.MustText(cvs, "(7,3)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "draws widget without container border",
termSize: image.Point{9, 5},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
PlaceWidget(fakewidget.New(widgetapi.Options{})),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(0, 0, 9, 5), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{1, 1}}
testdraw.MustText(cvs, "(9,5)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "widget.Draw returns an error",
termSize: image.Point{5, 5}, // Too small for the widget's box.
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
PlaceWidget(fakewidget.New(widgetapi.Options{})),
)
},
want: func(size image.Point) *faketerm.Terminal {
return faketerm.MustNew(size)
},
wantErr: true,
},
{
desc: "container with border and no space isn't drawn",
termSize: image.Point{1, 1},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
testdraw.MustText(cvs, "⇄", draw.TextBounds{})
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "container without the requested space for its widget isn't drawn",
termSize: image.Point{1, 1},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
PlaceWidget(fakewidget.New(widgetapi.Options{
MinimumSize: image.Point{2, 2}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
testdraw.MustText(cvs, "⇄", draw.TextBounds{})
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "widget gets the requested aspect ratio",
termSize: image.Point{22, 22},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
PlaceWidget(fakewidget.New(widgetapi.Options{
Ratio: image.Point{1, 2}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(1, 1, 11, 21), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{2, 2}}
testdraw.MustText(cvs, "(10,20)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "horizontal left align for the widget",
termSize: image.Point{22, 22},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
HorizontalAlignLeft(),
PlaceWidget(fakewidget.New(widgetapi.Options{
Ratio: image.Point{1, 2}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(1, 1, 11, 21), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{2, 2}}
testdraw.MustText(cvs, "(10,20)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "horizontal center align for the widget",
termSize: image.Point{22, 22},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
HorizontalAlignCenter(),
PlaceWidget(fakewidget.New(widgetapi.Options{
Ratio: image.Point{1, 2}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(6, 1, 16, 21), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{7, 2}}
testdraw.MustText(cvs, "(10,20)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "horizontal right align for the widget",
termSize: image.Point{22, 22},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
HorizontalAlignRight(),
PlaceWidget(fakewidget.New(widgetapi.Options{
Ratio: image.Point{1, 2}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(11, 1, 21, 21), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{12, 2}}
testdraw.MustText(cvs, "(10,20)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "vertical top align for the widget",
termSize: image.Point{22, 22},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
VerticalAlignTop(),
PlaceWidget(fakewidget.New(widgetapi.Options{
Ratio: image.Point{2, 1}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(1, 1, 21, 11), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{2, 2}}
testdraw.MustText(cvs, "(20,10)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "vertical middle align for the widget",
termSize: image.Point{22, 22},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
VerticalAlignMiddle(),
PlaceWidget(fakewidget.New(widgetapi.Options{
Ratio: image.Point{2, 1}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(1, 6, 21, 16), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{2, 7}}
testdraw.MustText(cvs, "(20,10)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
{
desc: "vertical bottom align for the widget",
termSize: image.Point{22, 22},
container: func(ft *faketerm.Terminal) *Container {
return New(
ft,
Border(draw.LineStyleLight),
VerticalAlignBottom(),
PlaceWidget(fakewidget.New(widgetapi.Options{
Ratio: image.Point{2, 1}},
)),
)
},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
cvs := testcanvas.MustNew(ft.Area())
// Container border.
testdraw.MustBox(
cvs,
cvs.Area(),
draw.LineStyleLight,
cell.FgColor(cell.ColorYellow),
)
// Fake widget border.
testdraw.MustBox(cvs, image.Rect(1, 11, 21, 21), draw.LineStyleLight)
tb := draw.TextBounds{Start: image.Point{2, 12}}
testdraw.MustText(cvs, "(20,10)", tb)
testcanvas.MustApply(cvs, ft)
return ft
},
},
}
for _, tc := range tests {
t.Run(tc.desc, func(t *testing.T) {
got := faketerm.MustNew(tc.termSize)
c := tc.container(got)
err := c.Draw()
if (err != nil) != tc.wantErr {
t.Errorf("Draw => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
return
}
if diff := faketerm.Diff(tc.want(got.Size()), got); diff != "" {
t.Errorf("Draw => %v", diff)
}
})
}
}
func TestDrawHandlesTerminalResize(t *testing.T) {
termSize := image.Point{60, 10}
got, err := faketerm.New(termSize)
if err != nil {
t.Errorf("faketerm.New => unexpected error: %v", err)
}
cont := New(
got,
SplitVertical(
Left(
SplitHorizontal(
Top(
PlaceWidget(fakewidget.New(widgetapi.Options{})),
),
Bottom(
PlaceWidget(fakewidget.New(widgetapi.Options{})),
),
),
),
Right(
SplitVertical(
Left(
PlaceWidget(fakewidget.New(widgetapi.Options{})),
),
Right(
PlaceWidget(fakewidget.New(widgetapi.Options{})),
),
),
),
),
)
// The following tests aren't hermetic, they all access the same container
// and fake terminal in order to retain state between resizes.
tests := []struct {
desc string
resize *image.Point // if not nil, the fake terminal will be resized.
want func(size image.Point) *faketerm.Terminal
}{
{
desc: "handles the initial draw request",
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(0, 0, 30, 5)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(0, 5, 30, 10)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(30, 0, 45, 10)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(45, 0, 60, 10)),
widgetapi.Options{},
)
return ft
},
},
{
desc: "increase in size",
resize: &image.Point{80, 10},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(0, 0, 40, 5)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(0, 5, 40, 10)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(40, 0, 60, 10)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(60, 0, 80, 10)),
widgetapi.Options{},
)
return ft
},
},
{
desc: "decrease in size",
resize: &image.Point{50, 10},
want: func(size image.Point) *faketerm.Terminal {
ft := faketerm.MustNew(size)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(0, 0, 25, 5)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(0, 5, 25, 10)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(25, 0, 37, 10)),
widgetapi.Options{},
)
fakewidget.MustDraw(
ft,
testcanvas.MustNew(image.Rect(37, 0, 50, 10)),
widgetapi.Options{},
)
return ft
},
},
}
for _, tc := range tests {
t.Run(tc.desc, func(t *testing.T) {
if tc.resize != nil {
if err := got.Resize(*tc.resize); err != nil {
t.Fatalf("Resize => unexpected error: %v", err)
}
}
if err := cont.Draw(); err != nil {
t.Fatalf("Draw => unexpected error: %v", err)
}
if diff := faketerm.Diff(tc.want(got.Size()), got); diff != "" {
t.Errorf("Draw => %v", diff)
}
})
}
}