1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
|
package plot
import (
"bytes"
"github.com/sgreben/jp/pkg/draw"
)
// ScatterChart is a scatter plot
type ScatterChart struct{ draw.Canvas }
// NewScatterChart returns a new line chart
func NewScatterChart(canvas draw.Canvas) *ScatterChart { return &ScatterChart{canvas} }
func (c *ScatterChart) drawAxes(paddingX, paddingY int, minX, maxX, minY, maxY float64) {
buffer := c.GetBuffer()
// X axis
buffer.SetRow(1, paddingX, buffer.Width, draw.HorizontalLine)
// Y axis
buffer.SetColumn(1, buffer.Height, paddingX, draw.VerticalLine)
// Corner
buffer.Set(1, paddingX, draw.CornerBottomLeft)
// Labels
buffer.WriteRight(1, 1, Ff(minY))
buffer.WriteLeft(buffer.Height-1, paddingX, Ff(maxY))
buffer.WriteRight(0, paddingX, Ff(minX))
buffer.WriteLeft(0, buffer.Width, Ff(maxX))
}
// Draw implements Chart
func (c *ScatterChart) Draw(data *DataTable) string {
var scaleY, scaleX float64
minX, maxX, minY, maxY := minMax(data)
minLabelWidth := len(Ff(minY))
maxLabelWidth := len(Ff(maxY))
paddingX := minLabelWidth + 1
paddingY := 2
if minLabelWidth < maxLabelWidth {
paddingX = maxLabelWidth + 1
}
chartWidth := c.Size().Width - (paddingX+1)*c.RuneSize().Width
chartHeight := c.Size().Height - paddingY*c.RuneSize().Height
scaleX = float64(chartWidth) / (maxX - minX)
scaleY = float64(chartHeight) / (maxY - minY)
for _, point := range data.Rows {
if len(point) < 2 {
continue
}
x := int((point[0]-minX)*scaleX + float64((paddingX+1)*c.RuneSize().Width))
y := int((point[1]-minY)*scaleY + float64(paddingY*c.RuneSize().Height))
c.Set(y, x)
}
c.drawAxes(paddingX, paddingY, minX, maxX, minY, maxY)
b := bytes.NewBuffer(nil)
c.GetBuffer().Render(b)
return b.String()
}
|