summaryrefslogtreecommitdiffstats
path: root/pkg/plot/heatmap.go
blob: 2667b40e998664455ae4cd5221d026a9a7b2150c (plain)
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
64
65
66
67
68
69
70
71
package plot

import (
	"bytes"

	"github.com/sgreben/jp/pkg/data"
	"github.com/sgreben/jp/pkg/draw"
)

// HeatMap is a heatmap
type HeatMap struct{ draw.Heatmap }

// NewHeatMap returns a new line chart
func NewHeatMap(buffer *draw.Buffer) *HeatMap { return &HeatMap{draw.Heatmap{buffer}} }

func (c *HeatMap) 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 *HeatMap) Draw(heatmap *data.Heatmap) string {
	var scaleY, scaleX float64

	minX := heatmap.X[0].LeftInclusive
	maxX := heatmap.X[len(heatmap.X)-1].Right
	minY := heatmap.Y[0].LeftInclusive
	maxY := heatmap.Y[len(heatmap.Y)-1].Right
	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)
	chartHeight := c.Size().Height - paddingY
	scaleX = float64(chartWidth) / (maxX - minX)
	scaleY = float64(chartHeight) / (maxY - minY)

	for i := range heatmap.Z {
		for j := range heatmap.Z[i] {
			x0 := int((heatmap.X[j].LeftInclusive-minX)*scaleX + float64(paddingX+1))
			y0 := int((heatmap.Y[i].LeftInclusive-minY)*scaleY + float64(paddingY))
			x1 := int((heatmap.X[j].Right-minX)*scaleX + float64(paddingX+1))
			y1 := int((heatmap.Y[i].Right-minY)*scaleY + float64(paddingY))
			z := heatmap.Z[i][j]
			for x := x0; x < x1; x++ {
				for y := y0; y < y1; y++ {
					c.Set(y, x, z)
				}
			}
		}
	}
	c.drawAxes(paddingX, paddingY, minX, maxX, minY, maxY)

	b := bytes.NewBuffer(nil)
	c.GetBuffer().Render(b)
	return b.String()
}