Implement the algorithm
This commit is contained in:
parent
9852fce5a1
commit
f79e1f3446
4 changed files with 328 additions and 9 deletions
186
tick.go
186
tick.go
|
@ -5,7 +5,7 @@ import (
|
||||||
"math"
|
"math"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
util "github.com/wcharczuk/go-chart/util"
|
"github.com/wcharczuk/go-chart/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
// TicksProvider is a type that provides ticks.
|
// TicksProvider is a type that provides ticks.
|
||||||
|
@ -115,3 +115,187 @@ func GenerateContinuousTicks(r Renderer, ra Range, isVertical bool, style Style,
|
||||||
|
|
||||||
return ticks
|
return ticks
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GeneratePrettyContinuousTicks generates a set of ticks at visually pleasing intervals.
|
||||||
|
// Based on http://vis.stanford.edu/files/2010-TickLabels-InfoVis.pdf by Justin Talbot et. al.
|
||||||
|
func GeneratePrettyContinuousTicks(r Renderer, ra Range, isVertical bool, style Style, vf ValueFormatter) []Tick {
|
||||||
|
if vf == nil {
|
||||||
|
vf = FloatValueFormatter
|
||||||
|
}
|
||||||
|
|
||||||
|
prettyStepsPriorityList := []float64{1, 5, 2, 2.5, 4, 3}
|
||||||
|
paramWeights := map[string]float64{
|
||||||
|
"simplicity": 0.2,
|
||||||
|
"coverage": 0.25,
|
||||||
|
"density": 0.5,
|
||||||
|
"legibility": 0.05,
|
||||||
|
}
|
||||||
|
|
||||||
|
rangeMin, rangeMax := ra.GetMin(), ra.GetMax()
|
||||||
|
|
||||||
|
renderedLabelExample := vf(rangeMin)
|
||||||
|
style.GetTextOptions().WriteToRenderer(r)
|
||||||
|
renderedLabelSizePx := r.MeasureText(renderedLabelExample)
|
||||||
|
|
||||||
|
var actualLabelSizePx, desiredPaddedLabelSizePx float64
|
||||||
|
if isVertical {
|
||||||
|
actualLabelSizePx = math.Max(float64(renderedLabelSizePx.Height()), 1)
|
||||||
|
desiredPaddedLabelSizePx = actualLabelSizePx + DefaultMinimumTickVerticalSpacing
|
||||||
|
} else {
|
||||||
|
actualLabelSizePx = math.Max(float64(renderedLabelSizePx.Width()), 1)
|
||||||
|
desiredPaddedLabelSizePx = actualLabelSizePx + DefaultMinimumTickHorizontalSpacing
|
||||||
|
}
|
||||||
|
availableSpacePx := float64(ra.GetDomain())
|
||||||
|
desiredTicksCount := math.Min(
|
||||||
|
math.Max(math.Floor(availableSpacePx/desiredPaddedLabelSizePx), 2), // less than 2 leads to incorrect density calculation
|
||||||
|
DefaultTickCountSanityCheck)
|
||||||
|
|
||||||
|
prettyStepsCount := float64(len(prettyStepsPriorityList))
|
||||||
|
|
||||||
|
var bestTickMin, bestTickMax, bestTickStep float64
|
||||||
|
bestScore := -2.0
|
||||||
|
stepsToSkip := 1.0
|
||||||
|
|
||||||
|
OUTER:
|
||||||
|
for {
|
||||||
|
for prettyStepIndex, prettyStep := range prettyStepsPriorityList {
|
||||||
|
simplicityMax := calculateSimplicityMax(float64(prettyStepIndex), prettyStepsCount, stepsToSkip)
|
||||||
|
|
||||||
|
if paramWeights["simplicity"]*simplicityMax+
|
||||||
|
paramWeights["coverage"]+
|
||||||
|
paramWeights["density"]+
|
||||||
|
paramWeights["legibility"] < bestScore {
|
||||||
|
break OUTER
|
||||||
|
}
|
||||||
|
|
||||||
|
ticksCount := 2.0
|
||||||
|
for {
|
||||||
|
densityMax := calculateDensityMax(ticksCount, desiredTicksCount)
|
||||||
|
|
||||||
|
if paramWeights["simplicity"]*simplicityMax+
|
||||||
|
paramWeights["coverage"]+
|
||||||
|
paramWeights["density"]*densityMax+
|
||||||
|
paramWeights["legibility"] < bestScore {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
delta := (rangeMax - rangeMin) / (ticksCount + 1) / stepsToSkip / prettyStep
|
||||||
|
stepSizeMultiplierLog := math.Ceil(math.Log10(delta))
|
||||||
|
|
||||||
|
for {
|
||||||
|
tickStep := stepsToSkip * prettyStep * math.Pow(10, stepSizeMultiplierLog)
|
||||||
|
coverageMax := calculateCoverageMax(rangeMin, rangeMax, tickStep*(ticksCount-1))
|
||||||
|
|
||||||
|
if paramWeights["simplicity"]*simplicityMax+
|
||||||
|
paramWeights["coverage"]*coverageMax+
|
||||||
|
paramWeights["density"]*densityMax+
|
||||||
|
paramWeights["legibility"] < bestScore {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
minStart := math.Floor(rangeMax/tickStep)*stepsToSkip - (ticksCount-1)*stepsToSkip
|
||||||
|
maxStart := math.Ceil(rangeMin/tickStep) * stepsToSkip
|
||||||
|
|
||||||
|
if minStart > maxStart {
|
||||||
|
stepSizeMultiplierLog += 1
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for start := minStart; start <= maxStart; start++ {
|
||||||
|
tickMin := start * (tickStep / stepsToSkip)
|
||||||
|
tickMax := tickMin + tickStep*(ticksCount-1)
|
||||||
|
|
||||||
|
coverage := calculateCoverage(rangeMin, rangeMax, tickMin, tickMax)
|
||||||
|
simplicity := calculateSimplicity(prettyStepsCount, float64(prettyStepIndex), stepsToSkip, tickMin, tickMax, tickStep)
|
||||||
|
density := calculateDensity(ticksCount, desiredTicksCount, rangeMin, rangeMax, tickMin, tickMax)
|
||||||
|
legibility := 1.0 // format is out of our control (provided by ValueFormatter)
|
||||||
|
// font size is out of our control (provided by Style)
|
||||||
|
// orientation is out of our control
|
||||||
|
if actualLabelSizePx*ticksCount > availableSpacePx {
|
||||||
|
legibility = math.Inf(-1) // overlap is unacceptable
|
||||||
|
}
|
||||||
|
|
||||||
|
score := paramWeights["simplicity"]*simplicity +
|
||||||
|
paramWeights["coverage"]*coverage +
|
||||||
|
paramWeights["density"]*density +
|
||||||
|
paramWeights["legibility"]*legibility
|
||||||
|
|
||||||
|
// original algorithm allows ticks outside value range, but it breaks rendering in this library
|
||||||
|
if score > bestScore && tickMin >= rangeMin && tickMax <= rangeMax {
|
||||||
|
bestTickMin = tickMin
|
||||||
|
bestTickMax = tickMax
|
||||||
|
bestTickStep = tickStep
|
||||||
|
bestScore = score
|
||||||
|
}
|
||||||
|
}
|
||||||
|
stepSizeMultiplierLog++
|
||||||
|
}
|
||||||
|
ticksCount++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
stepsToSkip++
|
||||||
|
}
|
||||||
|
|
||||||
|
var ticks []Tick
|
||||||
|
if bestTickStep == 0 {
|
||||||
|
return ticks
|
||||||
|
}
|
||||||
|
|
||||||
|
if ra.IsDescending() {
|
||||||
|
for tickValue := bestTickMax; tickValue > bestTickMin-bestTickStep/2; tickValue -= bestTickStep {
|
||||||
|
ticks = append(ticks, Tick{
|
||||||
|
Value: tickValue,
|
||||||
|
Label: vf(tickValue),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for tickValue := bestTickMin; tickValue < bestTickMax+bestTickStep/2; tickValue += bestTickStep {
|
||||||
|
ticks = append(ticks, Tick{
|
||||||
|
Value: tickValue,
|
||||||
|
Label: vf(tickValue),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ticks
|
||||||
|
}
|
||||||
|
|
||||||
|
func calculateSimplicity(prettyStepsCount, prettyStepIndex, stepsToSkip, tickMin, tickMax, tickStep float64) float64 {
|
||||||
|
var hasZeroTick float64
|
||||||
|
if tickMin <= 0 && tickMax >= 0 && math.Mod(tickMin, tickStep) < 10e-10 {
|
||||||
|
hasZeroTick = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1 - prettyStepIndex/(prettyStepsCount-1) - stepsToSkip + hasZeroTick
|
||||||
|
}
|
||||||
|
|
||||||
|
func calculateSimplicityMax(prettyStepIndex, prettyStepsCount, stepsToSkip float64) float64 {
|
||||||
|
return 2 - prettyStepIndex/(prettyStepsCount-1) - stepsToSkip
|
||||||
|
}
|
||||||
|
|
||||||
|
func calculateCoverage(rangeMin, rangeMax, tickMin, tickMax float64) float64 {
|
||||||
|
return 1 - 0.5*(math.Pow(rangeMax-tickMax, 2)+math.Pow(rangeMin-tickMin, 2))/math.Pow(0.1*(rangeMax-rangeMin), 2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func calculateCoverageMax(rangeMin, rangeMax, span float64) float64 {
|
||||||
|
if span <= rangeMax-rangeMin {
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1 - math.Pow((rangeMax-rangeMin)/2, 2)/math.Pow(0.1*(rangeMax-rangeMin), 2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func calculateDensity(ticksCount, desiredTicksCount, rangeMin, rangeMax, tickMin, tickMax float64) float64 {
|
||||||
|
ticksDensity := (ticksCount - 1) / (tickMax - tickMin)
|
||||||
|
desiredTicksDensity := (desiredTicksCount - 1) / (math.Max(tickMax, rangeMax) - math.Min(tickMin, rangeMin))
|
||||||
|
|
||||||
|
return 2 - math.Max(ticksDensity/desiredTicksDensity, desiredTicksDensity/ticksDensity)
|
||||||
|
}
|
||||||
|
|
||||||
|
func calculateDensityMax(ticksCount, desiredTicksCount float64) float64 {
|
||||||
|
if ticksCount >= desiredTicksCount {
|
||||||
|
return 2 - (ticksCount-1)/(desiredTicksCount-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
129
tick_test.go
129
tick_test.go
|
@ -1,9 +1,10 @@
|
||||||
package chart
|
package chart
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"math"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
assert "github.com/blend/go-sdk/assert"
|
"github.com/blend/go-sdk/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestGenerateContinuousTicks(t *testing.T) {
|
func TestGenerateContinuousTicks(t *testing.T) {
|
||||||
|
@ -58,3 +59,129 @@ func TestGenerateContinuousTicksDescending(t *testing.T) {
|
||||||
assert.Equal(1.0, ticks[len(ticks)-2].Value)
|
assert.Equal(1.0, ticks[len(ticks)-2].Value)
|
||||||
assert.Equal(0.0, ticks[len(ticks)-1].Value)
|
assert.Equal(0.0, ticks[len(ticks)-1].Value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestGenerateContinuousPrettyTicks(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
f, err := GetDefaultFont()
|
||||||
|
assert.Nil(err)
|
||||||
|
|
||||||
|
r, err := PNG(1024, 1024)
|
||||||
|
assert.Nil(err)
|
||||||
|
r.SetFont(f)
|
||||||
|
|
||||||
|
ra := &ContinuousRange{
|
||||||
|
Min: 37.5,
|
||||||
|
Max: 60.1,
|
||||||
|
Domain: 256,
|
||||||
|
}
|
||||||
|
|
||||||
|
vf := FloatValueFormatter
|
||||||
|
|
||||||
|
ticks := GeneratePrettyContinuousTicks(r, ra, false, Style{}, vf)
|
||||||
|
assert.NotEmpty(ticks)
|
||||||
|
assert.Equal(ticks, []Tick{
|
||||||
|
{Label: "38.00", Value: 38},
|
||||||
|
{Label: "40.00", Value: 40},
|
||||||
|
{Label: "42.00", Value: 42},
|
||||||
|
{Label: "44.00", Value: 44},
|
||||||
|
{Label: "46.00", Value: 46},
|
||||||
|
{Label: "48.00", Value: 48},
|
||||||
|
{Label: "50.00", Value: 50},
|
||||||
|
{Label: "52.00", Value: 52},
|
||||||
|
{Label: "54.00", Value: 54},
|
||||||
|
{Label: "56.00", Value: 56},
|
||||||
|
{Label: "58.00", Value: 58},
|
||||||
|
{Label: "60.00", Value: 60}})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGeneratePrettyTicksForVerySmallRange(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
f, err := GetDefaultFont()
|
||||||
|
assert.Nil(err)
|
||||||
|
|
||||||
|
r, err := PNG(1024, 1024)
|
||||||
|
assert.Nil(err)
|
||||||
|
r.SetFont(f)
|
||||||
|
|
||||||
|
ra := &ContinuousRange{
|
||||||
|
Min: 1e-100,
|
||||||
|
Max: 1e-99,
|
||||||
|
Domain: 256,
|
||||||
|
}
|
||||||
|
|
||||||
|
vf := FloatValueFormatter
|
||||||
|
|
||||||
|
ticks := GeneratePrettyContinuousTicks(r, ra, false, Style{}, vf)
|
||||||
|
assert.NotEmpty(ticks)
|
||||||
|
assert.Len(ticks, 9)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGeneratePrettyTicksForVeryLargeRange(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
f, err := GetDefaultFont()
|
||||||
|
assert.Nil(err)
|
||||||
|
|
||||||
|
r, err := PNG(1024, 1024)
|
||||||
|
assert.Nil(err)
|
||||||
|
r.SetFont(f)
|
||||||
|
|
||||||
|
ra := &ContinuousRange{
|
||||||
|
Min: 1e-100,
|
||||||
|
Max: 1e+100,
|
||||||
|
Domain: 256,
|
||||||
|
}
|
||||||
|
|
||||||
|
vf := FloatValueFormatter
|
||||||
|
|
||||||
|
ticks := GeneratePrettyContinuousTicks(r, ra, false, Style{}, vf)
|
||||||
|
assert.NotEmpty(ticks)
|
||||||
|
assert.Len(ticks, 10)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGeneratePrettyTicksForVerySmallDomain(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
f, err := GetDefaultFont()
|
||||||
|
assert.Nil(err)
|
||||||
|
|
||||||
|
r, err := PNG(1024, 1024)
|
||||||
|
assert.Nil(err)
|
||||||
|
r.SetFont(f)
|
||||||
|
|
||||||
|
ra := &ContinuousRange{
|
||||||
|
Min: 0.0,
|
||||||
|
Max: 10.0,
|
||||||
|
Domain: 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
vf := FloatValueFormatter
|
||||||
|
|
||||||
|
ticks := GeneratePrettyContinuousTicks(r, ra, false, Style{}, vf)
|
||||||
|
assert.Empty(ticks)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGeneratePrettyTicksForVeryLargeDomain(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
f, err := GetDefaultFont()
|
||||||
|
assert.Nil(err)
|
||||||
|
|
||||||
|
r, err := PNG(1024, 1024)
|
||||||
|
assert.Nil(err)
|
||||||
|
r.SetFont(f)
|
||||||
|
|
||||||
|
ra := &ContinuousRange{
|
||||||
|
Min: 0.0,
|
||||||
|
Max: 10.0,
|
||||||
|
Domain: math.MaxInt32,
|
||||||
|
}
|
||||||
|
|
||||||
|
vf := FloatValueFormatter
|
||||||
|
|
||||||
|
ticks := GeneratePrettyContinuousTicks(r, ra, false, Style{}, vf)
|
||||||
|
assert.NotEmpty(ticks)
|
||||||
|
assert.Len(ticks, 1001)
|
||||||
|
}
|
||||||
|
|
12
xaxis.go
12
xaxis.go
|
@ -3,7 +3,7 @@ package chart
|
||||||
import (
|
import (
|
||||||
"math"
|
"math"
|
||||||
|
|
||||||
util "github.com/wcharczuk/go-chart/util"
|
"github.com/wcharczuk/go-chart/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
// XAxis represents the horizontal axis.
|
// XAxis represents the horizontal axis.
|
||||||
|
@ -15,9 +15,10 @@ type XAxis struct {
|
||||||
ValueFormatter ValueFormatter
|
ValueFormatter ValueFormatter
|
||||||
Range Range
|
Range Range
|
||||||
|
|
||||||
TickStyle Style
|
TickStyle Style
|
||||||
Ticks []Tick
|
Ticks []Tick
|
||||||
TickPosition TickPosition
|
TickPosition TickPosition
|
||||||
|
EnablePrettyTicks bool
|
||||||
|
|
||||||
GridLines []GridLine
|
GridLines []GridLine
|
||||||
GridMajorStyle Style
|
GridMajorStyle Style
|
||||||
|
@ -66,6 +67,9 @@ func (xa XAxis) GetTicks(r Renderer, ra Range, defaults Style, vf ValueFormatter
|
||||||
return tp.GetTicks(r, defaults, vf)
|
return tp.GetTicks(r, defaults, vf)
|
||||||
}
|
}
|
||||||
tickStyle := xa.Style.InheritFrom(defaults)
|
tickStyle := xa.Style.InheritFrom(defaults)
|
||||||
|
if xa.EnablePrettyTicks {
|
||||||
|
return GeneratePrettyContinuousTicks(r, ra, false, tickStyle, vf)
|
||||||
|
}
|
||||||
return GenerateContinuousTicks(r, ra, false, tickStyle, vf)
|
return GenerateContinuousTicks(r, ra, false, tickStyle, vf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
10
yaxis.go
10
yaxis.go
|
@ -3,7 +3,7 @@ package chart
|
||||||
import (
|
import (
|
||||||
"math"
|
"math"
|
||||||
|
|
||||||
util "github.com/wcharczuk/go-chart/util"
|
"github.com/wcharczuk/go-chart/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
// YAxis is a veritcal rule of the range.
|
// YAxis is a veritcal rule of the range.
|
||||||
|
@ -22,8 +22,9 @@ type YAxis struct {
|
||||||
ValueFormatter ValueFormatter
|
ValueFormatter ValueFormatter
|
||||||
Range Range
|
Range Range
|
||||||
|
|
||||||
TickStyle Style
|
TickStyle Style
|
||||||
Ticks []Tick
|
Ticks []Tick
|
||||||
|
EnablePrettyTicks bool
|
||||||
|
|
||||||
GridLines []GridLine
|
GridLines []GridLine
|
||||||
GridMajorStyle Style
|
GridMajorStyle Style
|
||||||
|
@ -71,6 +72,9 @@ func (ya YAxis) GetTicks(r Renderer, ra Range, defaults Style, vf ValueFormatter
|
||||||
return tp.GetTicks(r, defaults, vf)
|
return tp.GetTicks(r, defaults, vf)
|
||||||
}
|
}
|
||||||
tickStyle := ya.Style.InheritFrom(defaults)
|
tickStyle := ya.Style.InheritFrom(defaults)
|
||||||
|
if ya.EnablePrettyTicks {
|
||||||
|
return GeneratePrettyContinuousTicks(r, ra, true, tickStyle, vf)
|
||||||
|
}
|
||||||
return GenerateContinuousTicks(r, ra, true, tickStyle, vf)
|
return GenerateContinuousTicks(r, ra, true, tickStyle, vf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue