-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathdisplay.go
121 lines (96 loc) · 3.24 KB
/
display.go
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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
package cardputer
import (
"image"
"image/color"
"machine"
"tinygo.org/x/drivers/st7789"
)
const (
dispWidth = 240
dispHeight = 135
)
var Display = &display{}
type display struct {
device st7789.Device
bus machine.SPI
scroll int16
}
func (d *display) Init() {
// Configure SPI pins
LCDMOSI.Configure(machine.PinConfig{Mode: machine.PinSPI})
//LCDMISO.Configure(machine.PinConfig{Mode: machine.PinSPI})
// configure SPI
machine.SPI0.Configure(machine.SPIConfig{
SCK: LCDSCK,
SDO: LCDMOSI,
// TODO: Speed, mode, etc?
})
d.device = st7789.New(machine.SPI0, LCDReset, LCDRS, LCDCS, LCDBacklight)
d.device.Configure(st7789.Config{
Width: dispWidth,
Height: dispHeight,
//Rotation:
//RowOffset int16
//ColumnOffset int16
//FrameRate FrameRate
//VSyncLines int16
})
}
func (*display) Bounds() image.Rectangle {
return image.Rect(0, 0, dispWidth, dispHeight)
// if the display doesn't use 0,0 as the upper left corner, might have to change this.
// if 0,0 is bottom left
// return image.Rect(0,-135,240,0)
}
func (d *display) Set(x, y int, c color.Color) {
d.device.SetPixel(int16(x), int16(y), colorToRGBA(c))
}
func (d *display) ColorModel() color.Model {
// TODO fix
return color.RGBAModel
}
func (d *display) Fill(r image.Rectangle, c color.Color) {
d.device.FillRectangle(int16(r.Min.X), int16(r.Min.Y), int16(r.Dx()), int16(r.Dy()), colorToRGBA(c))
}
// Blit copies pixels from img into display, aligning img.Bounds().Min to 'at' within display.
func (d *display) Blit(img image.Image, at image.Point) {
// must convert img to a slice of []color.RGBA first
d.device.FillRectangleWithBuffer(int16(img.Bounds().Min.X), int16(img.Bounds().Min.Y), int16(img.Bounds().Dx()), int16(img.Bounds().Dy()), imageToRGBASlice(img))
// would be better to make st7789.Device actually implement Blit and have it iterate over pixel data
// so we still get hte performance benefit of Blit operation, without the cost of memory
}
// we can't implement this (can't read the display!), but need to fake it so we implement draw.Draw
// we could buffer the Set() calls, but pretty sure we'll run out of RAM.
func (d *display) At(x, y int) color.Color {
return color.Alpha{0}
}
func (d *display) Scroll(amount int) {
d.scroll = (d.scroll + int16(amount)) % dispHeight // modulus may be wrong...
d.device.SetScroll(d.scroll)
// and then clear scrolled area? Leave that to fansiterm...
}
func colorToRGBA(c color.Color) color.RGBA {
if rgba, ok := c.(color.RGBA); ok {
return rgba
}
r, g, b, a := c.RGBA()
return color.RGBA{uint8(r / 0x101), uint8(g / 0x101), uint8(b / 0x101), uint8(a / 0x101)}
}
func imageToRGBASlice(img image.Image) []color.RGBA {
out := make([]color.RGBA, img.Bounds().Dx()*img.Bounds().Dy())
stride := img.Bounds().Dx()
if rgbaImg, ok := img.(*image.RGBA); ok {
for y := img.Bounds().Min.Y; y < img.Bounds().Max.Y; y++ {
for x := img.Bounds().Min.X; x < img.Bounds().Max.X; x++ {
out[y*stride+x] = rgbaImg.RGBAAt(x-img.Bounds().Min.X, y-img.Bounds().Min.Y)
}
}
return out
}
for y := img.Bounds().Min.Y; y < img.Bounds().Max.Y; y++ {
for x := img.Bounds().Min.X; x < img.Bounds().Max.X; x++ {
out[y*stride+x] = colorToRGBA(img.At(x-img.Bounds().Min.X, y-img.Bounds().Min.Y))
}
}
return out
}