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.
|
|
|
|
|
2019-01-07 00:15:31 -05:00
|
|
|
package axes
|
|
|
|
|
|
|
|
import (
|
2019-01-08 00:24:48 -05:00
|
|
|
"image"
|
2019-01-07 00:15:31 -05:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/kylelemons/godebug/pretty"
|
|
|
|
)
|
|
|
|
|
|
|
|
type updateY struct {
|
|
|
|
minVal float64
|
|
|
|
maxVal float64
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestY(t *testing.T) {
|
|
|
|
tests := []struct {
|
2019-01-08 00:24:48 -05:00
|
|
|
desc string
|
|
|
|
minVal float64
|
|
|
|
maxVal float64
|
|
|
|
update *updateY
|
2019-01-26 22:23:55 -05:00
|
|
|
mode YScaleMode
|
2019-01-13 00:03:19 -05:00
|
|
|
cvsAr image.Rectangle
|
2019-01-08 00:24:48 -05:00
|
|
|
wantWidth int
|
|
|
|
want *YDetails
|
|
|
|
wantErr bool
|
2019-01-07 00:15:31 -05:00
|
|
|
}{
|
|
|
|
{
|
2019-01-08 00:24:48 -05:00
|
|
|
desc: "fails on canvas too small",
|
2019-01-07 00:15:31 -05:00
|
|
|
minVal: 0,
|
|
|
|
maxVal: 3,
|
2019-01-13 00:03:19 -05:00
|
|
|
cvsAr: image.Rect(0, 0, 3, 2),
|
2019-01-07 00:15:31 -05:00
|
|
|
wantWidth: 2,
|
2019-01-08 00:24:48 -05:00
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
{
|
2019-01-13 00:03:19 -05:00
|
|
|
desc: "fails on cvsWidth less than required width",
|
2019-01-08 00:24:48 -05:00
|
|
|
minVal: 0,
|
|
|
|
maxVal: 3,
|
2019-01-13 00:03:19 -05:00
|
|
|
cvsAr: image.Rect(0, 0, 2, 4),
|
2019-01-08 00:24:48 -05:00
|
|
|
wantWidth: 2,
|
|
|
|
wantErr: true,
|
|
|
|
},
|
2019-01-12 15:55:49 -05:00
|
|
|
{
|
|
|
|
desc: "fails when max is less than min",
|
|
|
|
minVal: 0,
|
|
|
|
maxVal: -1,
|
2019-01-13 00:03:19 -05:00
|
|
|
cvsAr: image.Rect(0, 0, 4, 4),
|
2019-01-12 15:55:49 -05:00
|
|
|
wantWidth: 3,
|
|
|
|
wantErr: true,
|
|
|
|
},
|
2019-01-08 00:24:48 -05:00
|
|
|
{
|
2019-01-13 00:03:19 -05:00
|
|
|
desc: "cvsWidth equals required width",
|
2019-01-08 00:24:48 -05:00
|
|
|
minVal: 0,
|
|
|
|
maxVal: 3,
|
2019-01-13 00:03:19 -05:00
|
|
|
cvsAr: image.Rect(0, 0, 3, 4),
|
2019-01-08 00:24:48 -05:00
|
|
|
wantWidth: 2,
|
2019-01-07 00:15:31 -05:00
|
|
|
want: &YDetails{
|
|
|
|
Width: 2,
|
2019-01-13 00:03:19 -05:00
|
|
|
Start: image.Point{1, 0},
|
|
|
|
End: image.Point{1, 2},
|
2019-01-26 22:23:55 -05:00
|
|
|
Scale: mustNewYScale(0, 3, 2, nonZeroDecimals, YScaleModeAnchored),
|
2019-01-08 00:24:48 -05:00
|
|
|
Labels: []*Label{
|
|
|
|
{NewValue(0, nonZeroDecimals), image.Point{0, 1}},
|
|
|
|
{NewValue(1.72, nonZeroDecimals), image.Point{0, 0}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2019-01-26 22:23:55 -05:00
|
|
|
{
|
|
|
|
desc: "success for anchored scale",
|
|
|
|
minVal: 1,
|
|
|
|
maxVal: 3,
|
|
|
|
mode: YScaleModeAnchored,
|
|
|
|
cvsAr: image.Rect(0, 0, 3, 4),
|
|
|
|
wantWidth: 2,
|
|
|
|
want: &YDetails{
|
|
|
|
Width: 2,
|
|
|
|
Start: image.Point{1, 0},
|
|
|
|
End: image.Point{1, 2},
|
|
|
|
Scale: mustNewYScale(0, 3, 2, nonZeroDecimals, YScaleModeAnchored),
|
|
|
|
Labels: []*Label{
|
|
|
|
{NewValue(0, nonZeroDecimals), image.Point{0, 1}},
|
|
|
|
{NewValue(1.72, nonZeroDecimals), image.Point{0, 0}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "success for adaptive scale",
|
|
|
|
minVal: 1,
|
|
|
|
maxVal: 6,
|
|
|
|
mode: YScaleModeAdaptive,
|
|
|
|
cvsAr: image.Rect(0, 0, 3, 4),
|
|
|
|
wantWidth: 2,
|
|
|
|
want: &YDetails{
|
|
|
|
Width: 2,
|
|
|
|
Start: image.Point{1, 0},
|
|
|
|
End: image.Point{1, 2},
|
|
|
|
Scale: mustNewYScale(1, 6, 2, nonZeroDecimals, YScaleModeAdaptive),
|
|
|
|
Labels: []*Label{
|
|
|
|
{NewValue(1, nonZeroDecimals), image.Point{0, 1}},
|
|
|
|
{NewValue(3.88, nonZeroDecimals), image.Point{0, 0}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2019-01-08 00:24:48 -05:00
|
|
|
{
|
2019-01-13 00:03:19 -05:00
|
|
|
desc: "cvsWidth just accommodates the longest label",
|
2019-01-08 00:24:48 -05:00
|
|
|
minVal: 0,
|
|
|
|
maxVal: 3,
|
2019-01-13 00:03:19 -05:00
|
|
|
cvsAr: image.Rect(0, 0, 6, 4),
|
2019-01-08 00:24:48 -05:00
|
|
|
wantWidth: 2,
|
|
|
|
want: &YDetails{
|
|
|
|
Width: 5,
|
2019-01-13 00:03:19 -05:00
|
|
|
Start: image.Point{4, 0},
|
|
|
|
End: image.Point{4, 2},
|
2019-01-26 22:23:55 -05:00
|
|
|
Scale: mustNewYScale(0, 3, 2, nonZeroDecimals, YScaleModeAnchored),
|
2019-01-08 00:24:48 -05:00
|
|
|
Labels: []*Label{
|
|
|
|
{NewValue(0, nonZeroDecimals), image.Point{3, 1}},
|
|
|
|
{NewValue(1.72, nonZeroDecimals), image.Point{0, 0}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2019-01-13 00:03:19 -05:00
|
|
|
desc: "cvsWidth is more than we need",
|
2019-01-08 00:24:48 -05:00
|
|
|
minVal: 0,
|
|
|
|
maxVal: 3,
|
2019-01-13 00:03:19 -05:00
|
|
|
cvsAr: image.Rect(0, 0, 7, 4),
|
2019-01-08 00:24:48 -05:00
|
|
|
wantWidth: 2,
|
|
|
|
want: &YDetails{
|
|
|
|
Width: 5,
|
2019-01-13 00:03:19 -05:00
|
|
|
Start: image.Point{4, 0},
|
|
|
|
End: image.Point{4, 2},
|
2019-01-26 22:23:55 -05:00
|
|
|
Scale: mustNewYScale(0, 3, 2, nonZeroDecimals, YScaleModeAnchored),
|
2019-01-08 00:24:48 -05:00
|
|
|
Labels: []*Label{
|
|
|
|
{NewValue(0, nonZeroDecimals), image.Point{3, 1}},
|
|
|
|
{NewValue(1.72, nonZeroDecimals), image.Point{0, 0}},
|
|
|
|
},
|
2019-01-07 00:15:31 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range tests {
|
|
|
|
t.Run(tc.desc, func(t *testing.T) {
|
2019-01-08 00:24:48 -05:00
|
|
|
y := NewY(tc.minVal, tc.maxVal)
|
2019-01-07 00:15:31 -05:00
|
|
|
if tc.update != nil {
|
2019-01-08 00:24:48 -05:00
|
|
|
y.Update(tc.update.minVal, tc.update.maxVal)
|
2019-01-07 00:15:31 -05:00
|
|
|
}
|
|
|
|
|
2019-01-08 00:24:48 -05:00
|
|
|
gotWidth := y.RequiredWidth()
|
2019-01-07 00:15:31 -05:00
|
|
|
if gotWidth != tc.wantWidth {
|
|
|
|
t.Errorf("RequiredWidth => got %v, want %v", gotWidth, tc.wantWidth)
|
|
|
|
}
|
|
|
|
|
2019-01-26 22:23:55 -05:00
|
|
|
got, err := y.Details(tc.cvsAr, tc.mode)
|
2019-01-08 00:24:48 -05:00
|
|
|
if (err != nil) != tc.wantErr {
|
|
|
|
t.Errorf("Details => unexpected error: %v, wantErr: %v", err, tc.wantErr)
|
2019-01-07 00:15:31 -05:00
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if diff := pretty.Compare(tc.want, got); diff != "" {
|
|
|
|
t.Errorf("Details => unexpected diff (-want, +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-01-12 15:55:49 -05:00
|
|
|
|
|
|
|
func TestNewXDetails(t *testing.T) {
|
|
|
|
tests := []struct {
|
2019-01-13 01:38:39 -05:00
|
|
|
desc string
|
|
|
|
numPoints int
|
|
|
|
yStart image.Point
|
|
|
|
cvsWidth int
|
|
|
|
cvsAr image.Rectangle
|
|
|
|
customLabels map[int]string
|
|
|
|
want *XDetails
|
|
|
|
wantErr bool
|
2019-01-12 15:55:49 -05:00
|
|
|
}{
|
|
|
|
{
|
|
|
|
desc: "fails when numPoints is negative",
|
|
|
|
numPoints: -1,
|
2019-01-13 00:03:19 -05:00
|
|
|
yStart: image.Point{0, 0},
|
|
|
|
cvsAr: image.Rect(0, 0, 2, 3),
|
2019-01-12 15:55:49 -05:00
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
{
|
2019-01-13 00:03:19 -05:00
|
|
|
desc: "fails when cvsAr isn't wide enough",
|
2019-01-12 15:55:49 -05:00
|
|
|
numPoints: 1,
|
2019-01-13 00:03:19 -05:00
|
|
|
yStart: image.Point{0, 0},
|
|
|
|
cvsAr: image.Rect(0, 0, 1, 3),
|
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "fails when cvsAr isn't tall enough",
|
|
|
|
numPoints: 1,
|
|
|
|
yStart: image.Point{0, 0},
|
|
|
|
cvsAr: image.Rect(0, 0, 3, 2),
|
2019-01-12 15:55:49 -05:00
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "works with no data points",
|
|
|
|
numPoints: 0,
|
2019-01-13 00:03:19 -05:00
|
|
|
yStart: image.Point{0, 0},
|
|
|
|
cvsAr: image.Rect(0, 0, 2, 3),
|
2019-01-12 15:55:49 -05:00
|
|
|
want: &XDetails{
|
2019-01-13 00:03:19 -05:00
|
|
|
Start: image.Point{0, 1},
|
|
|
|
End: image.Point{1, 1},
|
2019-01-12 15:55:49 -05:00
|
|
|
Scale: mustNewXScale(0, 1, nonZeroDecimals),
|
|
|
|
Labels: []*Label{
|
|
|
|
{
|
|
|
|
Value: NewValue(0, nonZeroDecimals),
|
2019-01-13 00:03:19 -05:00
|
|
|
Pos: image.Point{1, 2},
|
2019-01-12 15:55:49 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2019-01-13 00:03:19 -05:00
|
|
|
desc: "accounts for non-zero yStart",
|
2019-01-12 15:55:49 -05:00
|
|
|
numPoints: 0,
|
2019-01-13 00:03:19 -05:00
|
|
|
yStart: image.Point{2, 0},
|
|
|
|
cvsAr: image.Rect(0, 0, 4, 5),
|
2019-01-12 15:55:49 -05:00
|
|
|
want: &XDetails{
|
2019-01-13 00:03:19 -05:00
|
|
|
Start: image.Point{2, 3},
|
|
|
|
End: image.Point{3, 3},
|
2019-01-12 15:55:49 -05:00
|
|
|
Scale: mustNewXScale(0, 1, nonZeroDecimals),
|
|
|
|
Labels: []*Label{
|
|
|
|
{
|
|
|
|
Value: NewValue(0, nonZeroDecimals),
|
2019-01-13 00:03:19 -05:00
|
|
|
Pos: image.Point{3, 4},
|
2019-01-12 15:55:49 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range tests {
|
|
|
|
t.Run(tc.desc, func(t *testing.T) {
|
2019-01-13 01:38:39 -05:00
|
|
|
got, err := NewXDetails(tc.numPoints, tc.yStart, tc.cvsAr, tc.customLabels)
|
2019-01-12 15:55:49 -05:00
|
|
|
if (err != nil) != tc.wantErr {
|
|
|
|
t.Errorf("NewXDetails => unexpected error: %v, wantErr: %v", err, tc.wantErr)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if diff := pretty.Compare(tc.want, got); diff != "" {
|
|
|
|
t.Errorf("NewXDetails => unexpected diff (-want, +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-02-13 22:53:19 -05:00
|
|
|
|
|
|
|
func TestRequiredHeight(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
desc string
|
|
|
|
numPoints int
|
|
|
|
customLabels map[int]string
|
|
|
|
labelOrientation LabelOrientation
|
|
|
|
want int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
desc: "horizontal orientation",
|
|
|
|
want: 2,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "vertical orientation, no custom labels, need single row for max label",
|
|
|
|
numPoints: 9,
|
|
|
|
labelOrientation: LabelOrientationVertical,
|
|
|
|
want: 2,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "vertical orientation, no custom labels, need multiple row for max label",
|
|
|
|
numPoints: 100,
|
|
|
|
labelOrientation: LabelOrientationVertical,
|
|
|
|
want: 4,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "vertical orientation, custom labels but all shorter than max label",
|
|
|
|
numPoints: 100,
|
|
|
|
customLabels: map[int]string{1: "a", 2: "b"},
|
|
|
|
labelOrientation: LabelOrientationVertical,
|
|
|
|
want: 4,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
desc: "vertical orientation, custom labels and some longer than max label",
|
|
|
|
numPoints: 100,
|
|
|
|
customLabels: map[int]string{1: "a", 2: "bbbbb"},
|
|
|
|
labelOrientation: LabelOrientationVertical,
|
|
|
|
want: 6,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range tests {
|
|
|
|
t.Run(tc.desc, func(t *testing.T) {
|
|
|
|
got := RequiredHeight(tc.numPoints, tc.customLabels, tc.labelOrientation)
|
|
|
|
if got != tc.want {
|
|
|
|
t.Errorf("RequiredHeight => %d, want %d", got, tc.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|