2022-09-06 03:19:34 +00:00
|
|
|
# NeoPixel driver for MicroPython
|
|
|
|
# MIT license; Copyright (c) 2016 Damien P. George, 2021 Jim Mussared
|
|
|
|
|
|
|
|
from machine import bitstream
|
|
|
|
|
|
|
|
|
|
|
|
class NeoPixel:
|
|
|
|
# G R B W
|
|
|
|
ORDER = (1, 0, 2, 3)
|
|
|
|
|
2023-10-04 14:40:43 +00:00
|
|
|
def _clamp(self, v, c_min, c_max):
|
|
|
|
return min(max(v, c_min), c_max)
|
|
|
|
|
2023-10-04 14:31:18 +00:00
|
|
|
def __init__(self, pin, n, bpp=3, timing=1, brightness=None):
|
2022-09-06 03:19:34 +00:00
|
|
|
self.pin = pin
|
|
|
|
self.n = n
|
|
|
|
self.bpp = bpp
|
2023-10-04 14:40:43 +00:00
|
|
|
self.brightness = None
|
|
|
|
if brightness is not None:
|
|
|
|
self.brightness = self._clamp(float(brightness), 0.0, 1.0)
|
2022-09-06 03:19:34 +00:00
|
|
|
self.buf = bytearray(n * bpp)
|
|
|
|
self.pin.init(pin.OUT)
|
|
|
|
# Timing arg can either be 1 for 800kHz or 0 for 400kHz,
|
|
|
|
# or a user-specified timing ns tuple (high_0, low_0, high_1, low_1).
|
|
|
|
self.timing = (
|
|
|
|
((400, 850, 800, 450) if timing else (800, 1700, 1600, 900))
|
|
|
|
if isinstance(timing, int)
|
|
|
|
else timing
|
|
|
|
)
|
|
|
|
|
2023-10-04 13:44:21 +00:00
|
|
|
def _apply_brightness(self, v):
|
|
|
|
if self.brightness is not None:
|
|
|
|
return tuple(round(c * self.brightness) for c in v)
|
|
|
|
return v
|
2023-09-28 21:02:06 +00:00
|
|
|
|
2022-09-06 03:19:34 +00:00
|
|
|
def __len__(self):
|
|
|
|
return self.n
|
|
|
|
|
|
|
|
def __setitem__(self, i, v):
|
|
|
|
offset = i * self.bpp
|
2023-10-04 13:44:21 +00:00
|
|
|
v = self._apply_brightness(v)
|
2022-09-06 03:19:34 +00:00
|
|
|
for i in range(self.bpp):
|
2023-10-04 13:44:21 +00:00
|
|
|
self.buf[offset + self.ORDER[i]] = v[i]
|
2022-09-06 03:19:34 +00:00
|
|
|
|
|
|
|
def __getitem__(self, i):
|
|
|
|
offset = i * self.bpp
|
|
|
|
return tuple(self.buf[offset + self.ORDER[i]] for i in range(self.bpp))
|
|
|
|
|
|
|
|
def fill(self, v):
|
2023-10-04 13:44:21 +00:00
|
|
|
v = self._apply_brightness(v)
|
|
|
|
b = self.buf
|
|
|
|
l = len(self.buf)
|
|
|
|
bpp = self.bpp
|
|
|
|
for i in range(bpp):
|
|
|
|
c = v[i]
|
|
|
|
j = self.ORDER[i]
|
|
|
|
while j < l:
|
|
|
|
b[j] = c
|
|
|
|
j += bpp
|
2022-09-06 03:19:34 +00:00
|
|
|
|
2023-09-29 13:03:43 +00:00
|
|
|
def set_brightness(self, b: float):
|
2023-10-04 14:40:43 +00:00
|
|
|
self.brightness = self._clamp(b, 0.0, 1.0)
|
2023-10-04 13:44:21 +00:00
|
|
|
# This may look odd but because __getitem__ and __setitem__ handle all the
|
|
|
|
# brightness logic already, we can offload the work to those methods.
|
2023-09-29 13:43:58 +00:00
|
|
|
for i in range(self.n):
|
2023-09-29 13:03:43 +00:00
|
|
|
self[i] = self[i]
|
|
|
|
|
2022-09-06 03:19:34 +00:00
|
|
|
def write(self):
|
|
|
|
# BITSTREAM_TYPE_HIGH_LOW = 0
|
|
|
|
bitstream(self.pin, 0, self.timing, self.buf)
|