test: add test for series
This commit is contained in:
parent
1c89ed29be
commit
51682069d7
7 changed files with 370 additions and 18 deletions
11
bar_chart.go
11
bar_chart.go
|
|
@ -23,10 +23,17 @@
|
||||||
package charts
|
package charts
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/golang/freetype/truetype"
|
||||||
"github.com/wcharczuk/go-chart/v2"
|
"github.com/wcharczuk/go-chart/v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
func barChartRender(opt ChartOption, result *basicRenderResult) ([]*markPointRenderOption, error) {
|
type barChartOption struct {
|
||||||
|
SeriesList SeriesList
|
||||||
|
Theme string
|
||||||
|
Font *truetype.Font
|
||||||
|
}
|
||||||
|
|
||||||
|
func barChartRender(opt barChartOption, result *basicRenderResult) ([]*markPointRenderOption, error) {
|
||||||
|
|
||||||
d, err := NewDraw(DrawOption{
|
d, err := NewDraw(DrawOption{
|
||||||
Parent: result.d,
|
Parent: result.d,
|
||||||
|
|
@ -56,7 +63,7 @@ func barChartRender(opt ChartOption, result *basicRenderResult) ([]*markPointRen
|
||||||
barMaxHeight := result.getYRange(0).Size
|
barMaxHeight := result.getYRange(0).Size
|
||||||
theme := NewTheme(opt.Theme)
|
theme := NewTheme(opt.Theme)
|
||||||
|
|
||||||
seriesNames := opt.Legend.Data
|
seriesNames := opt.SeriesList.Names()
|
||||||
|
|
||||||
r := d.Render
|
r := d.Render
|
||||||
|
|
||||||
|
|
|
||||||
36
chart.go
36
chart.go
|
|
@ -57,7 +57,7 @@ type ChartOption struct {
|
||||||
Parent *Draw
|
Parent *Draw
|
||||||
Padding chart.Box
|
Padding chart.Box
|
||||||
Box chart.Box
|
Box chart.Box
|
||||||
SeriesList []Series
|
SeriesList SeriesList
|
||||||
BackgroundColor drawing.Color
|
BackgroundColor drawing.Color
|
||||||
Children []ChartOption
|
Children []ChartOption
|
||||||
}
|
}
|
||||||
|
|
@ -126,12 +126,16 @@ func (o *ChartOption) FillDefault(theme string) {
|
||||||
if o.Legend.Left == "" {
|
if o.Legend.Left == "" {
|
||||||
o.Legend.Left = PositionCenter
|
o.Legend.Left = PositionCenter
|
||||||
}
|
}
|
||||||
|
// legend与series name的关联
|
||||||
if len(o.Legend.Data) == 0 {
|
if len(o.Legend.Data) == 0 {
|
||||||
names := make([]string, len(o.SeriesList))
|
o.Legend.Data = o.SeriesList.Names()
|
||||||
for index, item := range o.SeriesList {
|
} else {
|
||||||
names[index] = item.Name
|
seriesCount := len(o.SeriesList)
|
||||||
|
for index, name := range o.Legend.Data {
|
||||||
|
if index < seriesCount {
|
||||||
|
o.SeriesList[index].Name = name
|
||||||
|
}
|
||||||
}
|
}
|
||||||
o.Legend.Data = names
|
|
||||||
}
|
}
|
||||||
if o.Legend.Style.Font == nil {
|
if o.Legend.Style.Font == nil {
|
||||||
o.Legend.Style.Font = o.Font
|
o.Legend.Style.Font = o.Font
|
||||||
|
|
@ -262,7 +266,11 @@ func Render(opt ChartOption) (*Draw, error) {
|
||||||
if !isPieChart {
|
if !isPieChart {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
err := pieChartRender(opt, result)
|
err := pieChartRender(pieChartOption{
|
||||||
|
SeriesList: opt.SeriesList,
|
||||||
|
Theme: opt.Theme,
|
||||||
|
Font: opt.Font,
|
||||||
|
}, result)
|
||||||
return err
|
return err
|
||||||
},
|
},
|
||||||
// bar render
|
// bar render
|
||||||
|
|
@ -271,9 +279,11 @@ func Render(opt ChartOption) (*Draw, error) {
|
||||||
if isPieChart || len(barSeries) == 0 {
|
if isPieChart || len(barSeries) == 0 {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
o := opt
|
options, err := barChartRender(barChartOption{
|
||||||
o.SeriesList = barSeries
|
SeriesList: barSeries,
|
||||||
options, err := barChartRender(o, result)
|
Theme: opt.Theme,
|
||||||
|
Font: opt.Font,
|
||||||
|
}, result)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
@ -286,9 +296,11 @@ func Render(opt ChartOption) (*Draw, error) {
|
||||||
if isPieChart || len(lineSeries) == 0 {
|
if isPieChart || len(lineSeries) == 0 {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
o := opt
|
options, err := lineChartRender(lineChartOption{
|
||||||
o.SeriesList = lineSeries
|
Theme: opt.Theme,
|
||||||
options, err := lineChartRender(o, result)
|
SeriesList: lineSeries,
|
||||||
|
Font: opt.Font,
|
||||||
|
}, result)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -23,11 +23,18 @@
|
||||||
package charts
|
package charts
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/golang/freetype/truetype"
|
||||||
"github.com/wcharczuk/go-chart/v2"
|
"github.com/wcharczuk/go-chart/v2"
|
||||||
"github.com/wcharczuk/go-chart/v2/drawing"
|
"github.com/wcharczuk/go-chart/v2/drawing"
|
||||||
)
|
)
|
||||||
|
|
||||||
func lineChartRender(opt ChartOption, result *basicRenderResult) ([]*markPointRenderOption, error) {
|
type lineChartOption struct {
|
||||||
|
Theme string
|
||||||
|
SeriesList SeriesList
|
||||||
|
Font *truetype.Font
|
||||||
|
}
|
||||||
|
|
||||||
|
func lineChartRender(opt lineChartOption, result *basicRenderResult) ([]*markPointRenderOption, error) {
|
||||||
|
|
||||||
theme := NewTheme(opt.Theme)
|
theme := NewTheme(opt.Theme)
|
||||||
|
|
||||||
|
|
@ -40,7 +47,7 @@ func lineChartRender(opt ChartOption, result *basicRenderResult) ([]*markPointRe
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
seriesNames := opt.Legend.Data
|
seriesNames := opt.SeriesList.Names()
|
||||||
|
|
||||||
r := d.Render
|
r := d.Render
|
||||||
xRange := result.xRange
|
xRange := result.xRange
|
||||||
|
|
|
||||||
11
pie_chart.go
11
pie_chart.go
|
|
@ -26,6 +26,7 @@ import (
|
||||||
"math"
|
"math"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/golang/freetype/truetype"
|
||||||
"github.com/wcharczuk/go-chart/v2"
|
"github.com/wcharczuk/go-chart/v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -40,7 +41,13 @@ func getPieStyle(theme *Theme, index int) chart.Style {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func pieChartRender(opt ChartOption, result *basicRenderResult) error {
|
type pieChartOption struct {
|
||||||
|
Theme string
|
||||||
|
Font *truetype.Font
|
||||||
|
SeriesList SeriesList
|
||||||
|
}
|
||||||
|
|
||||||
|
func pieChartRender(opt pieChartOption, result *basicRenderResult) error {
|
||||||
d, err := NewDraw(DrawOption{
|
d, err := NewDraw(DrawOption{
|
||||||
Parent: result.d,
|
Parent: result.d,
|
||||||
}, PaddingOption(chart.Box{
|
}, PaddingOption(chart.Box{
|
||||||
|
|
@ -91,7 +98,7 @@ func pieChartRender(opt ChartOption, result *basicRenderResult) error {
|
||||||
}
|
}
|
||||||
labelRadius := radius + float64(labelLineWidth)
|
labelRadius := radius + float64(labelLineWidth)
|
||||||
|
|
||||||
seriesNames := opt.Legend.Data
|
seriesNames := opt.SeriesList.Names()
|
||||||
|
|
||||||
if len(values) == 1 {
|
if len(values) == 1 {
|
||||||
getPieStyle(theme, 0).WriteToRenderer(r)
|
getPieStyle(theme, 0).WriteToRenderer(r)
|
||||||
|
|
|
||||||
|
|
@ -94,6 +94,7 @@ type Series struct {
|
||||||
MarkPoint SeriesMarkPoint
|
MarkPoint SeriesMarkPoint
|
||||||
MarkLine SeriesMarkLine
|
MarkLine SeriesMarkLine
|
||||||
}
|
}
|
||||||
|
type SeriesList []Series
|
||||||
|
|
||||||
type PieSeriesOption struct {
|
type PieSeriesOption struct {
|
||||||
Radius string
|
Radius string
|
||||||
|
|
@ -158,6 +159,14 @@ func (s *Series) Summary() seriesSummary {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (sl SeriesList) Names() []string {
|
||||||
|
names := make([]string, len(sl))
|
||||||
|
for index, s := range sl {
|
||||||
|
names[index] = s.Name
|
||||||
|
}
|
||||||
|
return names
|
||||||
|
}
|
||||||
|
|
||||||
type LabelFormatter func(index int, value float64, percent float64) string
|
type LabelFormatter func(index int, value float64, percent float64) string
|
||||||
|
|
||||||
func NewPieLabelFormatter(seriesNames []string, layout string) LabelFormatter {
|
func NewPieLabelFormatter(seriesNames []string, layout string) LabelFormatter {
|
||||||
|
|
|
||||||
158
series_test.go
Normal file
158
series_test.go
Normal file
|
|
@ -0,0 +1,158 @@
|
||||||
|
// MIT License
|
||||||
|
|
||||||
|
// Copyright (c) 2022 Tree Xie
|
||||||
|
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
|
// in the Software without restriction, including without limitation the rights
|
||||||
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
// copies of the Software, and to permit persons to whom the Software is
|
||||||
|
// furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
// The above copyright notice and this permission notice shall be included in all
|
||||||
|
// copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
// SOFTWARE.
|
||||||
|
|
||||||
|
package charts
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNewSeriesFromValues(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
assert.Equal(Series{
|
||||||
|
Data: []SeriesData{
|
||||||
|
{
|
||||||
|
Value: 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Value: 2,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Type: ChartTypeBar,
|
||||||
|
}, NewSeriesFromValues([]float64{
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
}, ChartTypeBar))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewSeriesDataFromValues(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
assert.Equal([]SeriesData{
|
||||||
|
{
|
||||||
|
Value: 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Value: 2,
|
||||||
|
},
|
||||||
|
}, NewSeriesDataFromValues([]float64{
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewPieSeriesList(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
assert.Equal([]Series{
|
||||||
|
{
|
||||||
|
Type: ChartTypePie,
|
||||||
|
Label: SeriesLabel{
|
||||||
|
Show: true,
|
||||||
|
},
|
||||||
|
Radius: "30%",
|
||||||
|
Data: []SeriesData{
|
||||||
|
{
|
||||||
|
Value: 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Type: ChartTypePie,
|
||||||
|
Label: SeriesLabel{
|
||||||
|
Show: true,
|
||||||
|
},
|
||||||
|
Radius: "30%",
|
||||||
|
Data: []SeriesData{
|
||||||
|
{
|
||||||
|
Value: 2,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}, NewPieSeriesList([]float64{
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
}, PieSeriesOption{
|
||||||
|
Radius: "30%",
|
||||||
|
LabelShow: true,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSeriesSummary(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
s := Series{
|
||||||
|
Data: NewSeriesDataFromValues([]float64{
|
||||||
|
1,
|
||||||
|
3,
|
||||||
|
5,
|
||||||
|
7,
|
||||||
|
9,
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
assert.Equal(seriesSummary{
|
||||||
|
MaxIndex: 4,
|
||||||
|
MaxValue: 9,
|
||||||
|
MinIndex: 0,
|
||||||
|
MinValue: 1,
|
||||||
|
AverageValue: 5,
|
||||||
|
}, s.Summary())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetSeriesNames(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
sl := SeriesList{
|
||||||
|
{
|
||||||
|
Name: "a",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "b",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
assert.Equal([]string{
|
||||||
|
"a",
|
||||||
|
"b",
|
||||||
|
}, sl.Names())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewPieLabelFormatter(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
fn := NewPieLabelFormatter([]string{
|
||||||
|
"a",
|
||||||
|
"b",
|
||||||
|
}, "")
|
||||||
|
assert.Equal("a: 35%", fn(0, 1.2, 0.35))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewValueLabelFormater(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
fn := NewValueLabelFormater([]string{
|
||||||
|
"a",
|
||||||
|
"b",
|
||||||
|
}, "")
|
||||||
|
assert.Equal("1.2", fn(0, 1.2, 0.35))
|
||||||
|
}
|
||||||
152
theme_test.go
Normal file
152
theme_test.go
Normal file
|
|
@ -0,0 +1,152 @@
|
||||||
|
// MIT License
|
||||||
|
|
||||||
|
// Copyright (c) 2022 Tree Xie
|
||||||
|
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
|
// in the Software without restriction, including without limitation the rights
|
||||||
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
// copies of the Software, and to permit persons to whom the Software is
|
||||||
|
// furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
// The above copyright notice and this permission notice shall be included in all
|
||||||
|
// copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
// SOFTWARE.
|
||||||
|
|
||||||
|
package charts
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/wcharczuk/go-chart/v2/drawing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestTheme(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
darkTheme := NewTheme(ThemeDark)
|
||||||
|
lightTheme := NewTheme(ThemeLight)
|
||||||
|
|
||||||
|
assert.True(darkTheme.IsDark())
|
||||||
|
assert.False(lightTheme.IsDark())
|
||||||
|
|
||||||
|
assert.Equal(drawing.Color{
|
||||||
|
R: 185,
|
||||||
|
G: 184,
|
||||||
|
B: 206,
|
||||||
|
A: 255,
|
||||||
|
}, darkTheme.GetAxisStrokeColor())
|
||||||
|
assert.Equal(drawing.Color{
|
||||||
|
R: 110,
|
||||||
|
G: 112,
|
||||||
|
B: 121,
|
||||||
|
A: 255,
|
||||||
|
}, lightTheme.GetAxisStrokeColor())
|
||||||
|
|
||||||
|
assert.Equal(drawing.Color{
|
||||||
|
R: 72,
|
||||||
|
G: 71,
|
||||||
|
B: 83,
|
||||||
|
A: 255,
|
||||||
|
}, darkTheme.GetAxisSplitLineColor())
|
||||||
|
assert.Equal(drawing.Color{
|
||||||
|
R: 224,
|
||||||
|
G: 230,
|
||||||
|
B: 242,
|
||||||
|
A: 255,
|
||||||
|
}, lightTheme.GetAxisSplitLineColor())
|
||||||
|
|
||||||
|
assert.Equal([]drawing.Color{
|
||||||
|
{
|
||||||
|
R: 84,
|
||||||
|
G: 112,
|
||||||
|
B: 198,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 145,
|
||||||
|
G: 204,
|
||||||
|
B: 117,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 250,
|
||||||
|
G: 200,
|
||||||
|
B: 88,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 238,
|
||||||
|
G: 102,
|
||||||
|
B: 102,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 115,
|
||||||
|
G: 192,
|
||||||
|
B: 222,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
}, darkTheme.GetSeriesColors())
|
||||||
|
assert.Equal([]drawing.Color{
|
||||||
|
{
|
||||||
|
R: 84,
|
||||||
|
G: 112,
|
||||||
|
B: 198,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 145,
|
||||||
|
G: 204,
|
||||||
|
B: 117,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 250,
|
||||||
|
G: 200,
|
||||||
|
B: 88,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 238,
|
||||||
|
G: 102,
|
||||||
|
B: 102,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
R: 115,
|
||||||
|
G: 192,
|
||||||
|
B: 222,
|
||||||
|
A: 255,
|
||||||
|
},
|
||||||
|
}, lightTheme.GetSeriesColors())
|
||||||
|
|
||||||
|
assert.Equal(drawing.Color{
|
||||||
|
R: 16,
|
||||||
|
G: 12,
|
||||||
|
B: 42,
|
||||||
|
A: 255,
|
||||||
|
}, darkTheme.GetBackgroundColor())
|
||||||
|
assert.Equal(drawing.ColorWhite, lightTheme.GetBackgroundColor())
|
||||||
|
|
||||||
|
assert.Equal(drawing.Color{
|
||||||
|
R: 238,
|
||||||
|
G: 238,
|
||||||
|
B: 238,
|
||||||
|
A: 255,
|
||||||
|
}, darkTheme.GetTextColor())
|
||||||
|
assert.Equal(drawing.Color{
|
||||||
|
R: 70,
|
||||||
|
G: 70,
|
||||||
|
B: 70,
|
||||||
|
A: 255,
|
||||||
|
}, lightTheme.GetTextColor())
|
||||||
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue