1
0
mirror of https://github.com/mum4k/termdash.git synced 2025-04-28 13:48:51 +08:00
termdash/widgets/linechart/axes/scale_test.go

607 lines
14 KiB
Go
Raw Normal View History

2019-01-07 00:16:48 -05:00
// Copyright 2019 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 axes
import (
"fmt"
"testing"
"github.com/kylelemons/godebug/pretty"
)
2019-01-08 23:56:05 -05:00
// mustNewYScale returns a new YScale or panics.
func mustNewYScale(min, max float64, cvsHeight, nonZeroDecimals int) *YScale {
s, err := NewYScale(min, max, cvsHeight, nonZeroDecimals)
if err != nil {
panic(err)
}
return s
}
// mustNewXScale returns a new XScale or panics.
func mustNewXScale(numPoints int, axisWidth, nonZeroDecimals int) *XScale {
s, err := NewXScale(numPoints, axisWidth, nonZeroDecimals)
if err != nil {
panic(err)
}
return s
}
func TestYScale(t *testing.T) {
tests := []struct {
desc string
min float64
max float64
cvsHeight int
nonZeroDecimals int
want *YScale
2019-01-08 23:56:05 -05:00
wantErr bool
}{
2019-01-08 23:56:05 -05:00
{
desc: "fails when max is less than min",
min: 0,
max: -1,
cvsHeight: 4,
nonZeroDecimals: 2,
wantErr: true,
},
{
desc: "fails when canvas height too small",
min: 0,
max: 1,
cvsHeight: 0,
nonZeroDecimals: 2,
wantErr: true,
},
{
desc: "min and max are zero",
min: 0,
max: 0,
cvsHeight: 4,
nonZeroDecimals: 2,
want: &YScale{
Min: NewValue(0, 2),
Max: NewValue(0, 2),
Step: NewValue(0, 2),
CvsHeight: 4,
brailleHeight: 16,
},
},
{
desc: "zero based scale",
min: 0,
max: 10,
cvsHeight: 4,
nonZeroDecimals: 2,
want: &YScale{
Min: NewValue(0, 2),
Max: NewValue(10, 2),
Step: NewValue(float64(10)/15, 2),
CvsHeight: 4,
brailleHeight: 16,
},
},
{
desc: "min is negative",
min: -10,
max: 10,
cvsHeight: 4,
nonZeroDecimals: 2,
want: &YScale{
Min: NewValue(-10, 2),
Max: NewValue(10, 2),
Step: NewValue(float64(20)/15, 2),
CvsHeight: 4,
brailleHeight: 16,
},
},
{
desc: "greater than one",
min: 0,
max: 5,
cvsHeight: 1,
nonZeroDecimals: 1,
want: &YScale{
Min: NewValue(0, 1),
Max: NewValue(5, 1),
Step: NewValue(float64(5)/3, 1),
CvsHeight: 1,
brailleHeight: 4,
},
},
}
for _, tc := range tests {
t.Run(tc.desc, func(t *testing.T) {
2019-01-08 23:56:05 -05:00
got, err := NewYScale(tc.min, tc.max, tc.cvsHeight, tc.nonZeroDecimals)
if (err != nil) != tc.wantErr {
t.Errorf("NewYScale => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
return
}
if diff := pretty.Compare(tc.want, got); diff != "" {
t.Errorf("NewYScale => unexpected diff (-want, +got):\n%s", diff)
}
})
}
}
// pixelToValueTest is a test case for PixelToValue.
type pixelToValueTest struct {
pixel int
want float64
wantErr bool
}
// valueToPixelTest is a test case for ValueToPixel.
type valueToPixelTest struct {
value float64
want int
wantErr bool
}
// cellLabelTest is a test case for CellLabel.
type cellLabelTest struct {
cell int
want *Value
wantErr bool
}
2019-01-08 23:36:21 -05:00
func TestYScaleMethods(t *testing.T) {
tests := []struct {
desc string
min float64
max float64
cvsHeight int
nonZeroDecimals int
pixelToValueTests []pixelToValueTest
valueToPixelTests []valueToPixelTest
cellLabelTests []cellLabelTest
}{
{
desc: "fails on negative pixel",
min: 0,
max: 10,
cvsHeight: 4,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{-1, 0, true},
},
},
{
desc: "fails on pixel out of range",
min: 0,
max: 10,
cvsHeight: 4,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{16, 0, true},
},
},
{
desc: "fails on value or cell too small",
min: -1,
max: 0,
cvsHeight: 4,
nonZeroDecimals: 2,
valueToPixelTests: []valueToPixelTest{
{-2, 0, true},
},
cellLabelTests: []cellLabelTest{
{-1, nil, true},
},
},
{
desc: "fails on value or cell too large",
min: -1,
max: 0,
cvsHeight: 4,
nonZeroDecimals: 2,
valueToPixelTests: []valueToPixelTest{
{1, 0, true},
},
cellLabelTests: []cellLabelTest{
{4, nil, true},
},
},
2019-01-08 23:56:05 -05:00
{
desc: "works without data points",
min: 0,
max: 0,
cvsHeight: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{0, 0, false},
},
valueToPixelTests: []valueToPixelTest{
{0, 0, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(0, 2), false},
},
},
{
desc: "integer scale",
min: 0,
max: 6,
cvsHeight: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{3, 0, false},
{2, 2, false},
{1, 4, false},
{0, 6, false},
},
valueToPixelTests: []valueToPixelTest{
{0, 3, false},
{0.5, 3, false},
{1, 2, false},
{1.5, 2, false},
{2, 2, false},
{4, 1, false},
{6, 0, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(0, 2), false},
},
},
{
desc: "integer scale, multi-row canvas",
min: 0,
max: 14,
cvsHeight: 2,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{7, 0, false},
{6, 2, false},
{5, 4, false},
{4, 6, false},
{3, 8, false},
{2, 10, false},
{1, 12, false},
{0, 14, false},
},
valueToPixelTests: []valueToPixelTest{
{0, 7, false},
{1, 6, false},
{4, 5, false},
{6, 4, false},
{14, 0, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(8, 2), false},
{1, NewValue(0, 2), false},
},
},
{
desc: "negative integer scale",
min: -3,
max: 3,
cvsHeight: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{3, -3, false},
{2, -1, false},
{1, 1, false},
{0, 3, false},
},
valueToPixelTests: []valueToPixelTest{
{-3, 3, false},
{-2.5, 3, false},
{-2, 2, false},
{-1.5, 2, false},
{-1, 2, false},
{0, 1, false},
{1, 1, false},
{3, 0, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(-3, 2), false},
},
},
{
desc: "negative integer scale, max is zero",
min: -6,
max: 0,
cvsHeight: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{3, -6, false},
{2, -4, false},
{1, -2, false},
{0, 0, false},
},
valueToPixelTests: []valueToPixelTest{
{-6, 3, false},
{-4, 2, false},
{-2, 1, false},
{0, 0, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(-6, 2), false},
},
},
{
desc: "zero based float scale",
min: 0,
max: 0.3,
cvsHeight: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{3, 0, false},
{2, 0.1, false},
{1, 0.2, false},
{0, 0.3, false},
},
valueToPixelTests: []valueToPixelTest{
{0, 3, false},
{0.1, 2, false},
{0.2, 1, false},
{0.3, 0, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(0, 2), false},
},
},
}
for _, test := range tests {
2019-01-08 23:56:05 -05:00
scale, err := NewYScale(test.min, test.max, test.cvsHeight, test.nonZeroDecimals)
if err != nil {
t.Fatalf("NewYScale => unexpected error: %v", err)
}
t.Run(fmt.Sprintf("PixelToValue:%s", test.desc), func(t *testing.T) {
for _, tc := range test.pixelToValueTests {
got, err := scale.PixelToValue(tc.pixel)
if (err != nil) != tc.wantErr {
t.Errorf("PixelToValue => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
continue
}
if got != tc.want {
t.Errorf("PixelToValue(%v) => %v, want %v", tc.pixel, got, tc.want)
}
}
})
t.Run(fmt.Sprintf("ValueToPixel:%s", test.desc), func(t *testing.T) {
for _, tc := range test.valueToPixelTests {
got, err := scale.ValueToPixel(tc.value)
if (err != nil) != tc.wantErr {
t.Errorf("ValueToPixel => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
continue
}
if got != tc.want {
t.Errorf("ValueToPixel(%v) => %v, want %v", tc.value, got, tc.want)
}
}
})
t.Run(fmt.Sprintf("CellLabel:%s", test.desc), func(t *testing.T) {
for _, tc := range test.cellLabelTests {
got, err := scale.CellLabel(tc.cell)
if (err != nil) != tc.wantErr {
t.Errorf("CellLabel => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
continue
}
if diff := pretty.Compare(tc.want, got); diff != "" {
t.Errorf("CellLabel(%v) => unexpected diff (-want, +got):\n%s", tc.cell, diff)
}
}
})
}
}
2019-01-08 23:36:21 -05:00
func TestXScaleMethods(t *testing.T) {
tests := []struct {
desc string
numPoints int
axisWidth int
nonZeroDecimals int
pixelToValueTests []pixelToValueTest
valueToPixelTests []valueToPixelTest
cellLabelTests []cellLabelTest
wantErr bool
}{
{
desc: "fails when numPoints negative",
numPoints: -1,
axisWidth: 1,
wantErr: true,
},
{
desc: "fails when axisWidth zero",
numPoints: 1,
axisWidth: 0,
wantErr: true,
},
{
desc: "fails on negative pixel",
numPoints: 1,
axisWidth: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{-1, 0, true},
},
},
{
desc: "fails on pixel out of range",
numPoints: 1,
axisWidth: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{2, 0, true},
},
},
{
desc: "fails on value or cell too small",
numPoints: 1,
axisWidth: 1,
nonZeroDecimals: 2,
valueToPixelTests: []valueToPixelTest{
{-1, 0, true},
},
cellLabelTests: []cellLabelTest{
{-1, nil, true},
},
},
{
desc: "fails on value or cell too large",
numPoints: 1,
axisWidth: 1,
nonZeroDecimals: 2,
valueToPixelTests: []valueToPixelTest{
{1, 0, true},
},
cellLabelTests: []cellLabelTest{
{2, nil, true},
},
},
2019-01-08 23:56:05 -05:00
{
desc: "works without data points",
numPoints: 0,
axisWidth: 1,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{0, 0, false},
},
valueToPixelTests: []valueToPixelTest{
{0, 0, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(0, 2), false},
},
},
2019-01-08 23:36:21 -05:00
{
desc: "integer scale, all points fit",
numPoints: 6,
axisWidth: 3,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{0, 0, false},
{1, 1, false},
{2, 2, false},
{3, 3, false},
{4, 4, false},
{5, 5, false},
},
valueToPixelTests: []valueToPixelTest{
{0, 0, false},
{1, 1, false},
{2, 2, false},
{3, 3, false},
{4, 4, false},
{5, 5, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(0, 2), false},
},
},
{
desc: "float scale, multiple points per pixel",
numPoints: 12,
axisWidth: 3,
nonZeroDecimals: 2,
pixelToValueTests: []pixelToValueTest{
{0, 0, false},
{1, 2.21, false},
{2, 4.42, false},
{3, 6.63, false},
{4, 8.84, false},
{5, 11, false},
},
valueToPixelTests: []valueToPixelTest{
{0, 0, false},
{1, 0, false},
{2, 1, false},
{3, 1, false},
{4, 2, false},
{5, 2, false},
{6, 3, false},
{7, 3, false},
{8, 4, false},
{9, 4, false},
{10, 5, false},
{11, 5, false},
},
cellLabelTests: []cellLabelTest{
{0, NewValue(0, 2), false},
},
},
}
for _, test := range tests {
scale, err := NewXScale(test.numPoints, test.axisWidth, test.nonZeroDecimals)
if (err != nil) != test.wantErr {
t.Errorf("NewXScale => unexpected error: %v, wantErr: %v", err, test.wantErr)
}
if err != nil {
continue
}
t.Run(fmt.Sprintf("PixelToValue:%s", test.desc), func(t *testing.T) {
for _, tc := range test.pixelToValueTests {
got, err := scale.PixelToValue(tc.pixel)
if (err != nil) != tc.wantErr {
t.Errorf("PixelToValue => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
continue
}
if got != tc.want {
t.Errorf("PixelToValue(%v) => %v, want %v", tc.pixel, got, tc.want)
}
}
})
t.Run(fmt.Sprintf("ValueToPixel:%s", test.desc), func(t *testing.T) {
for _, tc := range test.valueToPixelTests {
got, err := scale.ValueToPixel(int(tc.value))
if (err != nil) != tc.wantErr {
t.Errorf("ValueToPixel => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
continue
}
if got != tc.want {
t.Errorf("ValueToPixel(%v) => %v, want %v", tc.value, got, tc.want)
}
}
})
t.Run(fmt.Sprintf("CellLabel:%s", test.desc), func(t *testing.T) {
for _, tc := range test.cellLabelTests {
got, err := scale.CellLabel(tc.cell)
if (err != nil) != tc.wantErr {
t.Errorf("CellLabel => unexpected error: %v, wantErr: %v", err, tc.wantErr)
}
if err != nil {
continue
}
if diff := pretty.Compare(tc.want, got); diff != "" {
t.Errorf("CellLabel(%v) => unexpected diff (-want, +got):\n%s", tc.cell, diff)
}
}
})
}
}