package chart import ( "math" util "github.com/wcharczuk/go-chart/util" ) var ( // Draw contains helpers for drawing common objects. Draw = &draw{} ) type draw struct{} // LineSeries draws a line series with a renderer. func (d draw) LineSeries(r Renderer, canvasBox Box, xrange, yrange Range, style Style, vs ValuesProvider) { if vs.Len() == 0 { return } cb := canvasBox.Bottom cl := canvasBox.Left v0x, v0y := vs.GetValues(0) x0 := cl + xrange.Translate(v0x) y0 := cb - yrange.Translate(v0y) yv0 := yrange.Translate(0) var vx, vy float64 var x, y int if style.ShouldDrawStroke() && style.ShouldDrawFill() { style.GetFillOptions().WriteDrawingOptionsToRenderer(r) r.MoveTo(x0, y0) for i := 1; i < vs.Len(); i++ { vx, vy = vs.GetValues(i) x = cl + xrange.Translate(vx) y = cb - yrange.Translate(vy) r.LineTo(x, y) } r.LineTo(x, util.Math.MinInt(cb, cb-yv0)) r.LineTo(x0, util.Math.MinInt(cb, cb-yv0)) r.LineTo(x0, y0) r.Fill() } if style.ShouldDrawStroke() { style.GetStrokeOptions().WriteDrawingOptionsToRenderer(r) r.MoveTo(x0, y0) for i := 1; i < vs.Len(); i++ { vx, vy = vs.GetValues(i) x = cl + xrange.Translate(vx) y = cb - yrange.Translate(vy) r.LineTo(x, y) } r.Stroke() } if style.ShouldDrawDot() { defaultDotWidth := style.GetDotWidth() style.GetDotOptions().WriteDrawingOptionsToRenderer(r) for i := 0; i < vs.Len(); i++ { vx, vy = vs.GetValues(i) x = cl + xrange.Translate(vx) y = cb - yrange.Translate(vy) dotWidth := defaultDotWidth if style.DotWidthProvider != nil { dotWidth = style.DotWidthProvider(xrange, yrange, i, vx, vy) } if style.DotColorProvider != nil { dotColor := style.DotColorProvider(xrange, yrange, i, vx, vy) r.SetFillColor(dotColor) r.SetStrokeColor(dotColor) } r.Circle(dotWidth, x, y) r.FillStroke() } } } // BoundedSeries draws a series that implements BoundedValuesProvider. func (d draw) BoundedSeries(r Renderer, canvasBox Box, xrange, yrange Range, style Style, bbs BoundedValuesProvider, drawOffsetIndexes ...int) { drawOffsetIndex := 0 if len(drawOffsetIndexes) > 0 { drawOffsetIndex = drawOffsetIndexes[0] } cb := canvasBox.Bottom cl := canvasBox.Left v0x, v0y1, v0y2 := bbs.GetBoundedValues(0) x0 := cl + xrange.Translate(v0x) y0 := cb - yrange.Translate(v0y1) var vx, vy1, vy2 float64 var x, y int xvalues := make([]float64, bbs.Len()) xvalues[0] = v0x y2values := make([]float64, bbs.Len()) y2values[0] = v0y2 style.GetFillAndStrokeOptions().WriteToRenderer(r) r.MoveTo(x0, y0) for i := 1; i < bbs.Len(); i++ { vx, vy1, vy2 = bbs.GetBoundedValues(i) xvalues[i] = vx y2values[i] = vy2 x = cl + xrange.Translate(vx) y = cb - yrange.Translate(vy1) if i > drawOffsetIndex { r.LineTo(x, y) } else { r.MoveTo(x, y) } } y = cb - yrange.Translate(vy2) r.LineTo(x, y) for i := bbs.Len() - 1; i >= drawOffsetIndex; i-- { vx, vy2 = xvalues[i], y2values[i] x = cl + xrange.Translate(vx) y = cb - yrange.Translate(vy2) r.LineTo(x, y) } r.Close() r.FillStroke() } // HistogramSeries draws a value provider as boxes from 0. func (d draw) HistogramSeries(r Renderer, canvasBox Box, xrange, yrange Range, style Style, vs ValuesProvider, barWidths ...int) { if vs.Len() == 0 { return } //calculate bar width? seriesLength := vs.Len() barWidth := int(math.Floor(float64(xrange.GetDomain()) / float64(seriesLength))) if len(barWidths) > 0 { barWidth = barWidths[0] } cb := canvasBox.Bottom cl := canvasBox.Left //foreach datapoint, draw a box. for index := 0; index < seriesLength; index++ { vx, vy := vs.GetValues(index) y0 := yrange.Translate(0) x := cl + xrange.Translate(vx) y := yrange.Translate(vy) d.Box(r, Box{ Top: cb - y0, Left: x - (barWidth >> 1), Right: x + (barWidth >> 1), Bottom: cb - y, }, style) } } func (d draw) CandlestickSeries(r Renderer, canvasBox Box, xrange, yrange Range, style Style, cs CandlestickSeries) { if cs.Len() == 0 { return } candleValues := cs.GetCandleValues() cb := canvasBox.Bottom cl := canvasBox.Left var cv CandleValue for index := 0; index < len(candleValues); index++ { cv = candleValues[index] y0 := yrange.Translate(cv.Open) y1 := yrange.Translate(cv.Close) x0 := cl + xrange.Translate(util.Time.ToFloat64(util.Date.On(util.NYSEOpen(), cv.Timestamp))) x1 := cl + xrange.Translate(util.Time.ToFloat64(util.Date.On(util.NYSEClose(), cv.Timestamp))) x := x0 + ((x1 - x0) >> 1) // draw open / close box. if cv.Open < cv.Close { d.Box(r, Box{ Top: cb - y0, Left: x0, Right: x1, Bottom: cb - y1, }, style.InheritFrom(Style{FillColor: ColorAlternateGreen})) } else { d.Box(r, Box{ Top: cb - y1, Left: x0, Right: x1, Bottom: cb - y0, }, style.InheritFrom(Style{FillColor: ColorRed})) } // draw high / low t bars y0 = yrange.Translate(cv.High) y1 = yrange.Translate(cv.Low) style.InheritFrom(Style{StrokeColor: DefaultStrokeColor}).WriteToRenderer(r) r.MoveTo(x0, cb-y0) r.LineTo(x1, cb-y0) r.Stroke() r.MoveTo(x, cb-y0) r.LineTo(x, cb-y1) r.Stroke() r.MoveTo(x0, cb-y1) r.LineTo(x1, cb-y1) r.Stroke() } } // MeasureAnnotation measures how big an annotation would be. func (d draw) MeasureAnnotation(r Renderer, canvasBox Box, style Style, lx, ly int, label string) Box { style.WriteToRenderer(r) defer r.ResetStyle() textBox := r.MeasureText(label) textWidth := int(textBox.Width()) textHeight := int(textBox.Height()) halfTextHeight := textHeight >> 1 pt := style.Padding.GetTop(DefaultAnnotationPadding.Top) pl := style.Padding.GetLeft(DefaultAnnotationPadding.Left) pr := style.Padding.GetRight(DefaultAnnotationPadding.Right) pb := style.Padding.GetBottom(DefaultAnnotationPadding.Bottom) strokeWidth := style.GetStrokeWidth() top := ly - (pt + halfTextHeight) right := lx + pl + pr + textWidth + DefaultAnnotationDeltaWidth + int(strokeWidth) bottom := ly + (pb + halfTextHeight) return Box{ Top: top, Left: lx, Right: right, Bottom: bottom, } } // Annotation draws an anotation with a renderer. func (d draw) Annotation(r Renderer, canvasBox Box, style Style, lx, ly int, label string) { style.GetTextOptions().WriteToRenderer(r) defer r.ResetStyle() textBox := r.MeasureText(label) textWidth := int(textBox.Width()) halfTextHeight := int(textBox.Height()) >> 1 style.GetFillAndStrokeOptions().WriteToRenderer(r) pt := style.Padding.GetTop(DefaultAnnotationPadding.Top) pl := style.Padding.GetLeft(DefaultAnnotationPadding.Left) pr := style.Padding.GetRight(DefaultAnnotationPadding.Right) pb := style.Padding.GetBottom(DefaultAnnotationPadding.Bottom) textX := lx + pl + DefaultAnnotationDeltaWidth textY := ly + halfTextHeight ltx := lx + DefaultAnnotationDeltaWidth lty := ly - (pt + halfTextHeight) rtx := lx + pl + pr + textWidth + DefaultAnnotationDeltaWidth rty := ly - (pt + halfTextHeight) rbx := lx + pl + pr + textWidth + DefaultAnnotationDeltaWidth rby := ly + (pb + halfTextHeight) lbx := lx + DefaultAnnotationDeltaWidth lby := ly + (pb + halfTextHeight) r.MoveTo(lx, ly) r.LineTo(ltx, lty) r.LineTo(rtx, rty) r.LineTo(rbx, rby) r.LineTo(lbx, lby) r.LineTo(lx, ly) r.Close() r.FillStroke() style.GetTextOptions().WriteToRenderer(r) r.Text(label, textX, textY) } // Box draws a box with a given style. func (d draw) Box(r Renderer, b Box, s Style) { s.GetFillAndStrokeOptions().WriteToRenderer(r) defer r.ResetStyle() r.MoveTo(b.Left, b.Top) r.LineTo(b.Right, b.Top) r.LineTo(b.Right, b.Bottom) r.LineTo(b.Left, b.Bottom) r.LineTo(b.Left, b.Top) r.FillStroke() } func (d draw) BoxRotated(r Renderer, b Box, thetaDegrees float64, s Style) { d.Box2d(r, b.Corners().Rotate(thetaDegrees), s) } func (d draw) Box2d(r Renderer, bc Box2d, s Style) { s.GetFillAndStrokeOptions().WriteToRenderer(r) defer r.ResetStyle() r.MoveTo(int(bc.TopLeft.X), int(bc.TopLeft.Y)) r.LineTo(int(bc.TopRight.X), int(bc.TopRight.Y)) r.LineTo(int(bc.BottomRight.X), int(bc.BottomRight.Y)) r.LineTo(int(bc.BottomLeft.X), int(bc.BottomLeft.Y)) r.Close() r.FillStroke() } // DrawText draws text with a given style. func (d draw) Text(r Renderer, text string, x, y int, style Style) { style.GetTextOptions().WriteToRenderer(r) defer r.ResetStyle() r.Text(text, x, y) } func (d draw) MeasureText(r Renderer, text string, style Style) Box2d { style.GetTextOptions().WriteToRenderer(r) defer r.ResetStyle() return r.MeasureText(text) } // TextWithin draws the text within a given box. func (d draw) TextWithin(r Renderer, text string, box Box, style Style) { style.GetTextOptions().WriteToRenderer(r) defer r.ResetStyle() lines := Text.WrapFit(r, text, box.Width(), style) linesBox := Text.MeasureLines(r, lines, style) y := box.Top switch style.GetTextVerticalAlign() { case TextVerticalAlignBottom, TextVerticalAlignBaseline: // i have to build better baseline handling into measure text y = y - int(linesBox.Height()) case TextVerticalAlignMiddle, TextVerticalAlignMiddleBaseline: y = (y - int(linesBox.Height())) >> 1 } var tx, ty int for _, line := range lines { lineBox := r.MeasureText(line) switch style.GetTextHorizontalAlign() { case TextHorizontalAlignCenter: tx = box.Left + ((int(box.Width()) - int(lineBox.Width())) >> 1) case TextHorizontalAlignRight: tx = box.Right - int(lineBox.Width()) default: tx = box.Left } if style.TextRotationDegrees == 0 { ty = y + int(lineBox.Height()) } else { ty = y } r.Text(line, tx, ty) y += int(lineBox.Height()) + style.GetTextLineSpacing() } }