2016-07-10 04:11:47 -04:00
|
|
|
package chart
|
|
|
|
|
2016-07-10 13:43:04 -04:00
|
|
|
import (
|
|
|
|
"math"
|
|
|
|
"sort"
|
|
|
|
)
|
2016-07-10 04:11:47 -04:00
|
|
|
|
|
|
|
// XAxis represents the horizontal axis.
|
|
|
|
type XAxis struct {
|
2016-07-10 13:43:04 -04:00
|
|
|
Name string
|
|
|
|
Style Style
|
|
|
|
ValueFormatter ValueFormatter
|
|
|
|
Range Range
|
|
|
|
Ticks []Tick
|
2016-07-12 22:14:14 -04:00
|
|
|
|
|
|
|
GridLines []GridLine
|
|
|
|
GridMajorStyle Style
|
|
|
|
GridMinorStyle Style
|
2016-07-10 13:43:04 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetName returns the name.
|
|
|
|
func (xa XAxis) GetName() string {
|
|
|
|
return xa.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetStyle returns the style.
|
|
|
|
func (xa XAxis) GetStyle() Style {
|
|
|
|
return xa.Style
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetTicks returns the ticks for a series. It coalesces between user provided ticks and
|
|
|
|
// generated ticks.
|
2016-07-13 01:04:30 -04:00
|
|
|
func (xa XAxis) GetTicks(r Renderer, ra Range, defaults Style, vf ValueFormatter) []Tick {
|
2016-07-10 13:43:04 -04:00
|
|
|
if len(xa.Ticks) > 0 {
|
|
|
|
return xa.Ticks
|
|
|
|
}
|
2016-07-13 01:04:30 -04:00
|
|
|
return xa.generateTicks(r, ra, defaults, vf)
|
2016-07-10 13:43:04 -04:00
|
|
|
}
|
|
|
|
|
2016-07-13 01:04:30 -04:00
|
|
|
func (xa XAxis) generateTicks(r Renderer, ra Range, defaults Style, vf ValueFormatter) []Tick {
|
|
|
|
step := xa.getTickStep(r, ra, defaults, vf)
|
2016-07-11 02:06:14 -04:00
|
|
|
return GenerateTicksWithStep(ra, step, vf)
|
|
|
|
}
|
|
|
|
|
2016-07-13 01:04:30 -04:00
|
|
|
func (xa XAxis) getTickStep(r Renderer, ra Range, defaults Style, vf ValueFormatter) float64 {
|
|
|
|
tickCount := xa.getTickCount(r, ra, defaults, vf)
|
2016-07-11 02:06:14 -04:00
|
|
|
step := ra.Delta() / float64(tickCount)
|
|
|
|
return step
|
2016-07-10 13:43:04 -04:00
|
|
|
}
|
|
|
|
|
2016-07-13 01:04:30 -04:00
|
|
|
func (xa XAxis) getTickCount(r Renderer, ra Range, defaults Style, vf ValueFormatter) int {
|
|
|
|
r.SetFont(xa.Style.GetFont(defaults.GetFont()))
|
|
|
|
r.SetFontSize(xa.Style.GetFontSize(defaults.GetFontSize(DefaultFontSize)))
|
2016-07-10 13:43:04 -04:00
|
|
|
|
|
|
|
// take a cut at determining the 'widest' value.
|
|
|
|
l0 := vf(ra.Min)
|
|
|
|
ln := vf(ra.Max)
|
|
|
|
ll := l0
|
|
|
|
if len(ln) > len(l0) {
|
|
|
|
ll = ln
|
|
|
|
}
|
2016-07-11 21:48:51 -04:00
|
|
|
llb := r.MeasureText(ll)
|
2016-07-12 19:47:52 -04:00
|
|
|
textWidth := llb.Width()
|
2016-07-10 13:43:04 -04:00
|
|
|
width := textWidth + DefaultMinimumTickHorizontalSpacing
|
|
|
|
count := int(math.Ceil(float64(ra.Domain) / float64(width)))
|
|
|
|
return count
|
|
|
|
}
|
|
|
|
|
2016-07-12 22:14:14 -04:00
|
|
|
// GetGridLines returns the gridlines for the axis.
|
|
|
|
func (xa XAxis) GetGridLines(ticks []Tick) []GridLine {
|
|
|
|
if len(xa.GridLines) > 0 {
|
|
|
|
return xa.GridLines
|
|
|
|
}
|
2016-07-12 23:34:59 -04:00
|
|
|
return GenerateGridLines(ticks, true)
|
2016-07-12 22:14:14 -04:00
|
|
|
}
|
|
|
|
|
2016-07-11 21:48:51 -04:00
|
|
|
// Measure returns the bounds of the axis.
|
2016-07-12 23:34:59 -04:00
|
|
|
func (xa XAxis) Measure(r Renderer, canvasBox Box, ra Range, defaults Style, ticks []Tick) Box {
|
|
|
|
r.SetStrokeColor(xa.Style.GetStrokeColor(defaults.StrokeColor))
|
|
|
|
r.SetStrokeWidth(xa.Style.GetStrokeWidth(defaults.StrokeWidth))
|
|
|
|
r.SetFont(xa.Style.GetFont(defaults.GetFont()))
|
|
|
|
r.SetFontColor(xa.Style.GetFontColor(DefaultAxisColor))
|
|
|
|
r.SetFontSize(xa.Style.GetFontSize(defaults.GetFontSize()))
|
2016-07-11 21:48:51 -04:00
|
|
|
|
|
|
|
sort.Sort(Ticks(ticks))
|
|
|
|
|
|
|
|
var left, right, top, bottom = math.MaxInt32, 0, math.MaxInt32, 0
|
|
|
|
for _, t := range ticks {
|
|
|
|
v := t.Value
|
|
|
|
lx := ra.Translate(v)
|
|
|
|
tb := r.MeasureText(t.Label)
|
|
|
|
|
|
|
|
tx := canvasBox.Left + lx
|
2016-07-12 19:47:52 -04:00
|
|
|
ty := canvasBox.Bottom + DefaultXAxisMargin + tb.Height()
|
2016-07-11 21:48:51 -04:00
|
|
|
|
|
|
|
top = MinInt(top, canvasBox.Bottom)
|
2016-07-12 19:47:52 -04:00
|
|
|
left = MinInt(left, tx-(tb.Width()>>1))
|
|
|
|
right = MaxInt(right, tx+(tb.Width()>>1))
|
2016-07-11 21:48:51 -04:00
|
|
|
bottom = MaxInt(bottom, ty)
|
|
|
|
}
|
|
|
|
|
|
|
|
return Box{
|
|
|
|
Top: top,
|
|
|
|
Left: left,
|
|
|
|
Right: right,
|
|
|
|
Bottom: bottom,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-10 04:11:47 -04:00
|
|
|
// Render renders the axis
|
2016-07-12 23:34:59 -04:00
|
|
|
func (xa XAxis) Render(r Renderer, canvasBox Box, ra Range, defaults Style, ticks []Tick) {
|
|
|
|
r.SetStrokeColor(xa.Style.GetStrokeColor(defaults.StrokeColor))
|
|
|
|
r.SetStrokeWidth(xa.Style.GetStrokeWidth(defaults.StrokeWidth))
|
|
|
|
r.SetFont(xa.Style.GetFont(defaults.GetFont()))
|
|
|
|
r.SetFontColor(xa.Style.GetFontColor(DefaultAxisColor))
|
|
|
|
r.SetFontSize(xa.Style.GetFontSize(defaults.GetFontSize()))
|
2016-07-10 13:43:04 -04:00
|
|
|
|
|
|
|
r.MoveTo(canvasBox.Left, canvasBox.Bottom)
|
|
|
|
r.LineTo(canvasBox.Right, canvasBox.Bottom)
|
|
|
|
r.Stroke()
|
|
|
|
|
|
|
|
sort.Sort(Ticks(ticks))
|
2016-07-11 03:02:31 -04:00
|
|
|
|
2016-07-10 04:11:47 -04:00
|
|
|
for _, t := range ticks {
|
|
|
|
v := t.Value
|
2016-07-11 21:48:51 -04:00
|
|
|
lx := ra.Translate(v)
|
|
|
|
tb := r.MeasureText(t.Label)
|
|
|
|
tx := canvasBox.Left + lx
|
2016-07-12 19:47:52 -04:00
|
|
|
ty := canvasBox.Bottom + DefaultXAxisMargin + tb.Height()
|
|
|
|
r.Text(t.Label, tx-tb.Width()>>1, ty)
|
2016-07-11 21:48:51 -04:00
|
|
|
|
|
|
|
r.MoveTo(tx, canvasBox.Bottom)
|
|
|
|
r.LineTo(tx, canvasBox.Bottom+DefaultVerticalTickHeight)
|
|
|
|
r.Stroke()
|
2016-07-10 04:11:47 -04:00
|
|
|
}
|
2016-07-12 22:14:14 -04:00
|
|
|
|
|
|
|
if xa.GridMajorStyle.Show || xa.GridMinorStyle.Show {
|
|
|
|
for _, gl := range xa.GetGridLines(ticks) {
|
|
|
|
if (gl.IsMinor && xa.GridMinorStyle.Show) ||
|
|
|
|
(!gl.IsMinor && xa.GridMajorStyle.Show) {
|
|
|
|
gl.Render(r, canvasBox, ra)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-07-10 04:11:47 -04:00
|
|
|
}
|