You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
devices/waveshare2in13v2/drawing.go

142 lines
2.9 KiB
Go

// Copyright 2021 The Periph Authors. All rights reserved.
// Use of this source code is governed under the Apache License, Version 2.0
// that can be found in the LICENSE file.
package waveshare2in13v2
import (
"bytes"
"image"
"image/draw"
"periph.io/x/devices/v3/ssd1306/image1bit"
)
// setMemoryArea configures the target drawing area (horizontal is in bytes,
// vertical in pixels).
func setMemoryArea(ctrl controller, area image.Rectangle) {
ctrl.sendCommand(dataEntryModeSetting)
ctrl.sendData([]byte{
// Y increment, X increment; update address counter in X direction
0b011,
})
ctrl.sendCommand(setRAMXAddressStartEndPosition)
ctrl.sendData([]byte{
// Start
byte(area.Min.X),
// End
byte(area.Max.X - 1),
})
ctrl.sendCommand(setRAMYAddressStartEndPosition)
ctrl.sendData([]byte{
// Start
byte(area.Min.Y % 0xFF),
byte(area.Min.Y / 0xFF),
// End
byte((area.Max.Y - 1) % 0xFF),
byte((area.Max.Y - 1) / 0xFF),
})
ctrl.sendCommand(setRAMXAddressCounter)
ctrl.sendData([]byte{byte(area.Min.X)})
ctrl.sendCommand(setRAMYAddressCounter)
ctrl.sendData([]byte{
byte(area.Min.Y % 0xFF),
byte(area.Min.Y / 0xFF),
})
}
type drawOpts struct {
cmd byte
devSize image.Point
buffer *image1bit.VerticalLSB
dstRect image.Rectangle
src image.Image
srcPts image.Point
}
type drawSpec struct {
// Destination on display in pixels, normalized to fit into actual size.
DstRect image.Rectangle
// Area to send to device; horizontally in bytes (thus aligned to
// 8 pixels), vertically in pixels.
MemRect image.Rectangle
}
func (o *drawOpts) spec() drawSpec {
s := drawSpec{
DstRect: image.Rectangle{Max: o.devSize}.Intersect(o.dstRect),
}
s.MemRect = image.Rect(
s.DstRect.Min.X/8, s.DstRect.Min.Y,
(s.DstRect.Max.X+7)/8, s.DstRect.Max.Y,
)
return s
}
// drawImage sends an image to the controller after setting up the registers.
func drawImage(ctrl controller, opts *drawOpts) {
s := opts.spec()
if s.MemRect.Empty() {
return
}
draw.Src.Draw(opts.buffer, s.DstRect, opts.src, opts.srcPts)
setMemoryArea(ctrl, s.MemRect)
ctrl.sendCommand(opts.cmd)
rowData := make([]byte, s.MemRect.Dx())
for y := s.MemRect.Min.Y; y < s.MemRect.Max.Y; y++ {
for x := 0; x < len(rowData); x++ {
rowData[x] = 0
for bit := 0; bit < 8; bit++ {
if opts.buffer.BitAt(((s.MemRect.Min.X+x)*8)+bit, y) {
rowData[x] |= 0x80 >> bit
}
}
}
ctrl.sendData(rowData)
}
}
func clearDisplay(ctrl controller, size image.Point, color image1bit.Bit) {
var colorValue byte
if color == image1bit.On {
colorValue = 0xff
}
spec := (&drawOpts{
devSize: size,
dstRect: image.Rectangle{Max: size},
}).spec()
if spec.MemRect.Empty() {
return
}
setMemoryArea(ctrl, spec.MemRect)
ctrl.sendCommand(writeRAMBW)
data := bytes.Repeat([]byte{colorValue}, spec.MemRect.Dx())
for y := 0; y < spec.MemRect.Max.Y; y++ {
ctrl.sendData(data)
}
}