2020-03-22 15:40:18 +00:00
|
|
|
# SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
# Copyright (C) 2020 Daniel Thompson
|
|
|
|
|
2020-03-08 20:47:19 +00:00
|
|
|
"""Sitronix ST7789 display driver for MicroPython.
|
|
|
|
|
|
|
|
Note: Although the ST7789 supports a variety of communication protocols
|
|
|
|
currently this driver only has support for SPI interfaces. However it is
|
|
|
|
structured such that other serial protocols can easily be added.
|
|
|
|
"""
|
2020-01-23 18:59:43 +00:00
|
|
|
|
2020-01-31 19:36:55 +00:00
|
|
|
import micropython
|
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
from micropython import const
|
|
|
|
from time import sleep_ms
|
|
|
|
|
|
|
|
# register definitions
|
|
|
|
_SWRESET = const(0x01)
|
2020-02-01 20:20:30 +00:00
|
|
|
_SLPIN = const(0x10)
|
2020-01-23 18:59:43 +00:00
|
|
|
_SLPOUT = const(0x11)
|
|
|
|
_NORON = const(0x13)
|
|
|
|
_INVOFF = const(0x20)
|
|
|
|
_INVON = const(0x21)
|
2020-03-08 20:47:19 +00:00
|
|
|
_DISPOFF = const(0x28)
|
2020-01-23 18:59:43 +00:00
|
|
|
_DISPON = const(0x29)
|
|
|
|
_CASET = const(0x2a)
|
|
|
|
_RASET = const(0x2b)
|
|
|
|
_RAMWR = const(0x2c)
|
|
|
|
_COLMOD = const(0x3a)
|
|
|
|
_MADCTL = const(0x36)
|
2020-01-21 22:10:50 +00:00
|
|
|
|
|
|
|
class ST7789(object):
|
2020-01-23 18:59:43 +00:00
|
|
|
def __init__(self, width, height):
|
|
|
|
self.width = width
|
|
|
|
self.height = height
|
|
|
|
self.linebuffer = bytearray(2 * width)
|
|
|
|
self.init_display()
|
2020-01-21 22:10:50 +00:00
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
def init_display(self):
|
2020-01-21 22:10:50 +00:00
|
|
|
self.reset()
|
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
self.write_cmd(_SLPOUT)
|
|
|
|
sleep_ms(10)
|
|
|
|
|
|
|
|
for cmd in (
|
|
|
|
(_COLMOD, b'\x05'), # MCU will send 16-bit RGB565
|
|
|
|
(_MADCTL, b'\x00'), # Left to right, top to bottom
|
|
|
|
#(_INVOFF, None), # Results in odd palette
|
|
|
|
(_INVON, None),
|
|
|
|
(_NORON, None),
|
|
|
|
):
|
|
|
|
self.write_cmd(cmd[0])
|
|
|
|
if cmd[1]:
|
|
|
|
self.write_data(cmd[1])
|
|
|
|
self.fill(0)
|
|
|
|
self.write_cmd(_DISPON)
|
2020-01-28 21:19:36 +00:00
|
|
|
|
|
|
|
# From the point we sent the SLPOUT there must be a
|
|
|
|
# 120ms gap before any subsequent SLPIN. In most cases
|
|
|
|
# (i.e. wen the SPI baud rate is slower than 8M then
|
|
|
|
# that time already elapsed as we zeroed the RAM).
|
|
|
|
#sleep_ms(125)
|
2020-01-23 18:59:43 +00:00
|
|
|
|
|
|
|
def poweroff(self):
|
2020-02-01 20:20:30 +00:00
|
|
|
self.write_cmd(_SLPIN)
|
|
|
|
sleep_ms(125)
|
2020-01-23 18:59:43 +00:00
|
|
|
|
|
|
|
def poweron(self):
|
2020-02-01 20:20:30 +00:00
|
|
|
self.write_cmd(_SLPOUT)
|
|
|
|
sleep_ms(125)
|
2020-01-23 18:59:43 +00:00
|
|
|
|
|
|
|
def contrast(self, contrast):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def invert(self, invert):
|
|
|
|
if invert:
|
|
|
|
self.write_cmd(_INVON)
|
2020-01-21 22:10:50 +00:00
|
|
|
else:
|
2020-01-23 18:59:43 +00:00
|
|
|
self.write_cmd(_INVOFF)
|
2020-01-21 22:10:50 +00:00
|
|
|
|
2020-03-08 20:47:19 +00:00
|
|
|
def mute(self, mute):
|
|
|
|
if mute:
|
|
|
|
self.write_cmd(_DISPOFF)
|
|
|
|
else:
|
|
|
|
self.write_cmd(_DISPON)
|
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
def set_window(self, x=0, y=0, width=None, height=None):
|
|
|
|
if not width:
|
|
|
|
width = self.width
|
|
|
|
if not height:
|
|
|
|
height = self.height
|
2020-01-21 22:10:50 +00:00
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
xp = x + width - 1
|
|
|
|
yp = y + height - 1
|
2020-01-21 22:10:50 +00:00
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
self.write_cmd(_CASET)
|
|
|
|
self.write_data(bytearray([x >> 8, x & 0xff, xp >> 8, xp & 0xff]))
|
|
|
|
self.write_cmd(_RASET)
|
|
|
|
self.write_data(bytearray([y >> 8, y & 0xff, yp >> 8, yp & 0xff]))
|
|
|
|
self.write_cmd(_RAMWR)
|
2020-01-21 22:10:50 +00:00
|
|
|
|
2020-02-19 19:47:51 +00:00
|
|
|
def rawblit(self, buf, x, y, width, height):
|
|
|
|
self.set_window(x, y, width, height)
|
|
|
|
self.write_data(buf)
|
|
|
|
|
2020-02-03 22:34:54 +00:00
|
|
|
def fill(self, bg, x=0, y=0, w=None, h=None):
|
|
|
|
if not w:
|
|
|
|
w = self.width - x
|
|
|
|
if not h:
|
|
|
|
h = self.height - y
|
|
|
|
self.set_window(x, y, w, h)
|
2020-01-21 22:10:50 +00:00
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
# Populate the line buffer
|
2020-02-03 22:34:54 +00:00
|
|
|
buf = memoryview(self.linebuffer)[0:2*w]
|
|
|
|
for xi in range(0, 2*w, 2):
|
|
|
|
buf[xi] = bg >> 8
|
|
|
|
buf[xi+1] = bg & 0xff
|
|
|
|
|
|
|
|
# Do the fill
|
|
|
|
for yi in range(h):
|
|
|
|
self.write_data(buf)
|
2020-01-21 22:10:50 +00:00
|
|
|
|
2020-01-23 18:59:43 +00:00
|
|
|
class ST7789_SPI(ST7789):
|
|
|
|
def __init__(self, width, height, spi, cs, dc, res=None, rate=8000000):
|
|
|
|
self.spi = spi
|
|
|
|
self.dc = dc
|
|
|
|
self.res = res
|
|
|
|
self.cs = cs
|
|
|
|
self.rate = rate
|
|
|
|
|
|
|
|
#self.spi.init(baudrate=self.rate, polarity=1, phase=1)
|
|
|
|
cs.init(cs.OUT, value=1)
|
|
|
|
dc.init(dc.OUT, value=0)
|
|
|
|
if res:
|
|
|
|
res.init(res.OUT, value=0)
|
|
|
|
|
|
|
|
super().__init__(width, height)
|
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
if self.res:
|
|
|
|
self.res(0)
|
|
|
|
sleep_ms(10)
|
|
|
|
self.res(1)
|
|
|
|
else:
|
|
|
|
self.write_cmd(_SWRESET)
|
2020-02-01 20:20:30 +00:00
|
|
|
sleep_ms(125)
|
2020-01-23 18:59:43 +00:00
|
|
|
|
|
|
|
def write_cmd(self, cmd):
|
|
|
|
self.dc(0)
|
|
|
|
self.cs(0)
|
|
|
|
self.spi.write(bytearray([cmd]))
|
|
|
|
self.cs(1)
|
2020-03-09 21:32:34 +00:00
|
|
|
self.dc(1)
|
2020-01-23 18:59:43 +00:00
|
|
|
|
|
|
|
def write_data(self, buf):
|
|
|
|
self.cs(0)
|
|
|
|
self.spi.write(buf)
|
|
|
|
self.cs(1)
|