go-chart/moving_average_series.go
2016-07-14 13:11:03 -07:00

105 lines
2.6 KiB
Go

package chart
const (
// DefaultMovingAverageWindowSize is the default number of values to average.
DefaultMovingAverageWindowSize = 5
)
// MovingAverageSeries is a computed series.
type MovingAverageSeries struct {
Name string
Style Style
YAxis YAxisType
WindowSize int
InnerSeries ValueProvider
valueBuffer *RingBuffer
}
// GetName returns the name of the time series.
func (mas MovingAverageSeries) GetName() string {
return mas.Name
}
// GetStyle returns the line style.
func (mas MovingAverageSeries) GetStyle() Style {
return mas.Style
}
// GetYAxis returns which YAxis the series draws on.
func (mas MovingAverageSeries) GetYAxis() YAxisType {
return mas.YAxis
}
// Len returns the number of elements in the series.
func (mas *MovingAverageSeries) Len() int {
return mas.InnerSeries.Len()
}
// GetValue gets a value at a given index.
func (mas *MovingAverageSeries) GetValue(index int) (x float64, y float64) {
if mas.InnerSeries == nil {
return
}
if mas.valueBuffer == nil || index == 0 {
mas.valueBuffer = NewRingBufferWithCapacity(mas.GetWindowSize())
}
if mas.valueBuffer.Len() >= mas.GetWindowSize() {
mas.valueBuffer.Dequeue()
}
px, py := mas.InnerSeries.GetValue(index)
mas.valueBuffer.Enqueue(py)
x = px
y = mas.getAverage(mas.valueBuffer)
return
}
// GetLastValue computes the last moving average value but walking back window size samples,
// and recomputing the last moving average chunk.
func (mas MovingAverageSeries) GetLastValue() (x float64, y float64) {
if mas.InnerSeries == nil {
return
}
windowSize := mas.GetWindowSize()
seriesLength := mas.InnerSeries.Len()
startAt := seriesLength - windowSize
if startAt < 0 {
startAt = 0
}
vb := NewRingBufferWithCapacity(windowSize)
for index := startAt; index < seriesLength; index++ {
xn, yn := mas.InnerSeries.GetValue(index)
vb.Enqueue(yn)
x = xn
}
y = mas.getAverage(vb)
return
}
// GetWindowSize returns the window size.
func (mas MovingAverageSeries) GetWindowSize(defaults ...int) int {
if mas.WindowSize == 0 {
if len(defaults) > 0 {
return defaults[0]
}
return DefaultMovingAverageWindowSize
}
return mas.WindowSize
}
func (mas MovingAverageSeries) getAverage(valueBuffer *RingBuffer) float64 {
var accum float64
valueBuffer.Each(func(v interface{}) {
if typed, isTyped := v.(float64); isTyped {
accum += typed
}
})
return accum / float64(valueBuffer.Len())
}
// Render renders the series.
func (mas *MovingAverageSeries) Render(r Renderer, canvasBox Box, xrange, yrange Range, defaults Style) {
style := mas.Style.WithDefaultsFrom(defaults)
DrawLineSeries(r, canvasBox, xrange, yrange, style, mas)
}