Compare commits
3 Commits
master
...
new_voices
Author | SHA1 | Date |
---|---|---|
jocavdh | 568e34c768 | 5 years ago |
jocavdh | a226c11707 | 6 years ago |
jocavdh | dc8e78fefd | 6 years ago |
@ -0,0 +1,26 @@
|
||||
Pixel Ring
|
||||
==========
|
||||
|
||||
[![Build Status](https://travis-ci.org/respeaker/pixel_ring.svg?branch=master)](https://travis-ci.org/respeaker/pixel_ring)
|
||||
[![Pypi](https://img.shields.io/pypi/v/pixel_ring.svg)](https://pypi.python.org/pypi/pixel_ring)
|
||||
|
||||
|
||||
The library is for pixel ring based on APA102, ReSpeaker series pixel ring.
|
||||
|
||||
## Hardware
|
||||
+ ReSpeaker 4 Mic Array or ReSpeaker V2
|
||||
+ ReSpeaker V2
|
||||
+ ReSpeaker USB 6+1 Mic Array
|
||||
+ ReSpeaker USB 4 Mic Array
|
||||
|
||||
## Get started
|
||||
```
|
||||
git clone --depth 1 https://github.com/respeaker/pixel_ring.git
|
||||
cd pixel_ring
|
||||
pip install -U -e .
|
||||
python examples/respeaker_4mic_array.py
|
||||
```
|
||||
|
||||
|
||||
## Credits
|
||||
+ [APA102_Pi](https://github.com/tinue/APA102_Pi)
|
@ -0,0 +1,37 @@
|
||||
"""
|
||||
Control pixel ring on ReSpeaker 4 Mic Array
|
||||
|
||||
pip install pixel_ring gpiozero
|
||||
"""
|
||||
|
||||
import time
|
||||
|
||||
from pixel_ring import pixel_ring
|
||||
from gpiozero import LED
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
power = LED(5)
|
||||
power.on()
|
||||
|
||||
pixel_ring.set_brightness(20)
|
||||
pixel_ring.change_pattern('echo')
|
||||
while True:
|
||||
|
||||
try:
|
||||
pixel_ring.wakeup()
|
||||
time.sleep(3)
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.speak()
|
||||
time.sleep(6)
|
||||
pixel_ring.off()
|
||||
time.sleep(3)
|
||||
except KeyboardInterrupt:
|
||||
break
|
||||
|
||||
|
||||
pixel_ring.off()
|
||||
power.off()
|
||||
time.sleep(1)
|
||||
|
@ -0,0 +1,36 @@
|
||||
"""
|
||||
Control pixel ring on ReSpeaker 4 Mic Array
|
||||
|
||||
pip install pixel_ring gpiozero
|
||||
"""
|
||||
|
||||
import time
|
||||
|
||||
from pixel_ring import pixel_ring
|
||||
from gpiozero import LED
|
||||
|
||||
power = LED(5)
|
||||
power.on()
|
||||
|
||||
pixel_ring.set_brightness(10)
|
||||
|
||||
if __name__ == '__main__':
|
||||
while True:
|
||||
|
||||
try:
|
||||
pixel_ring.wakeup()
|
||||
time.sleep(3)
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.speak()
|
||||
time.sleep(6)
|
||||
pixel_ring.off()
|
||||
time.sleep(3)
|
||||
except KeyboardInterrupt:
|
||||
break
|
||||
|
||||
|
||||
pixel_ring.off()
|
||||
time.sleep(1)
|
||||
|
||||
power.off()
|
@ -0,0 +1,43 @@
|
||||
"""
|
||||
Control pixel ring on ReSpeaker V2
|
||||
|
||||
sudo apt install python-mraa libmraa1
|
||||
pip install pixel-ring
|
||||
|
||||
"""
|
||||
|
||||
import time
|
||||
|
||||
from pixel_ring import pixel_ring
|
||||
import mraa
|
||||
import os
|
||||
|
||||
en = mraa.Gpio(12)
|
||||
if os.geteuid() != 0 :
|
||||
time.sleep(1)
|
||||
|
||||
en.dir(mraa.DIR_OUT)
|
||||
en.write(0)
|
||||
|
||||
pixel_ring.set_brightness(20)
|
||||
|
||||
if __name__ == '__main__':
|
||||
while True:
|
||||
|
||||
try:
|
||||
pixel_ring.wakeup()
|
||||
time.sleep(3)
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.speak()
|
||||
time.sleep(6)
|
||||
pixel_ring.off()
|
||||
time.sleep(3)
|
||||
except KeyboardInterrupt:
|
||||
break
|
||||
|
||||
|
||||
pixel_ring.off()
|
||||
time.sleep(1)
|
||||
|
||||
en.write(1)
|
@ -0,0 +1,29 @@
|
||||
"""
|
||||
Control pixel ring on ReSpeaker USB Mic Array
|
||||
"""
|
||||
|
||||
import time
|
||||
|
||||
from pixel_ring import pixel_ring
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
pixel_ring.change_pattern('echo')
|
||||
while True:
|
||||
|
||||
try:
|
||||
pixel_ring.wakeup()
|
||||
time.sleep(3)
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.speak()
|
||||
time.sleep(6)
|
||||
pixel_ring.off()
|
||||
time.sleep(3)
|
||||
except KeyboardInterrupt:
|
||||
break
|
||||
|
||||
|
||||
pixel_ring.off()
|
||||
time.sleep(1)
|
||||
|
@ -0,0 +1,52 @@
|
||||
|
||||
|
||||
from . import usb_pixel_ring_v1
|
||||
from . import usb_pixel_ring_v2
|
||||
from .apa102_pixel_ring import PixelRing
|
||||
|
||||
pixel_ring = usb_pixel_ring_v2.find()
|
||||
|
||||
if not pixel_ring:
|
||||
pixel_ring = usb_pixel_ring_v1.find()
|
||||
|
||||
if not pixel_ring:
|
||||
pixel_ring = PixelRing()
|
||||
|
||||
|
||||
USAGE = '''
|
||||
If the hardware is ReSpeaker 4 Mic Array for Pi or ReSpeaker V2,
|
||||
there is a power-enable pin which should be enabled at first.
|
||||
+ ReSpeaker 4 Mic Array for Pi:
|
||||
|
||||
import gpiozero
|
||||
power = LED(5)
|
||||
power.on()
|
||||
|
||||
+ ReSpeaker V2:
|
||||
|
||||
import mraa
|
||||
power = mraa.Gpio(12)
|
||||
power.dir(mraa.DIR_OUT)
|
||||
power.write(0)
|
||||
'''
|
||||
|
||||
def main():
|
||||
import time
|
||||
|
||||
if isinstance(pixel_ring, usb_pixel_ring_v2.PixelRing):
|
||||
print('Found ReSpeaker USB 4 Mic Array')
|
||||
elif isinstance(pixel_ring, usb_pixel_ring_v1.UsbPixelRing):
|
||||
print('Found ReSpeaker USB 6+1 Mic Array')
|
||||
else:
|
||||
print('Control APA102 RGB LEDs via SPI')
|
||||
print(USAGE)
|
||||
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.off()
|
||||
time.sleep(1)
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
||||
|
@ -0,0 +1,243 @@
|
||||
"""
|
||||
from https://github.com/tinue/APA102_Pi
|
||||
This is the main driver module for APA102 LEDs
|
||||
"""
|
||||
import spidev
|
||||
from math import ceil
|
||||
|
||||
RGB_MAP = { 'rgb': [3, 2, 1], 'rbg': [3, 1, 2], 'grb': [2, 3, 1],
|
||||
'gbr': [2, 1, 3], 'brg': [1, 3, 2], 'bgr': [1, 2, 3] }
|
||||
|
||||
class APA102:
|
||||
"""
|
||||
Driver for APA102 LEDS (aka "DotStar").
|
||||
|
||||
(c) Martin Erzberger 2016-2017
|
||||
|
||||
My very first Python code, so I am sure there is a lot to be optimized ;)
|
||||
|
||||
Public methods are:
|
||||
- set_pixel
|
||||
- set_pixel_rgb
|
||||
- show
|
||||
- clear_strip
|
||||
- cleanup
|
||||
|
||||
Helper methods for color manipulation are:
|
||||
- combine_color
|
||||
- wheel
|
||||
|
||||
The rest of the methods are used internally and should not be used by the
|
||||
user of the library.
|
||||
|
||||
Very brief overview of APA102: An APA102 LED is addressed with SPI. The bits
|
||||
are shifted in one by one, starting with the least significant bit.
|
||||
|
||||
An LED usually just forwards everything that is sent to its data-in to
|
||||
data-out. While doing this, it remembers its own color and keeps glowing
|
||||
with that color as long as there is power.
|
||||
|
||||
An LED can be switched to not forward the data, but instead use the data
|
||||
to change it's own color. This is done by sending (at least) 32 bits of
|
||||
zeroes to data-in. The LED then accepts the next correct 32 bit LED
|
||||
frame (with color information) as its new color setting.
|
||||
|
||||
After having received the 32 bit color frame, the LED changes color,
|
||||
and then resumes to just copying data-in to data-out.
|
||||
|
||||
The really clever bit is this: While receiving the 32 bit LED frame,
|
||||
the LED sends zeroes on its data-out line. Because a color frame is
|
||||
32 bits, the LED sends 32 bits of zeroes to the next LED.
|
||||
As we have seen above, this means that the next LED is now ready
|
||||
to accept a color frame and update its color.
|
||||
|
||||
So that's really the entire protocol:
|
||||
- Start by sending 32 bits of zeroes. This prepares LED 1 to update
|
||||
its color.
|
||||
- Send color information one by one, starting with the color for LED 1,
|
||||
then LED 2 etc.
|
||||
- Finish off by cycling the clock line a few times to get all data
|
||||
to the very last LED on the strip
|
||||
|
||||
The last step is necessary, because each LED delays forwarding the data
|
||||
a bit. Imagine ten people in a row. When you yell the last color
|
||||
information, i.e. the one for person ten, to the first person in
|
||||
the line, then you are not finished yet. Person one has to turn around
|
||||
and yell it to person 2, and so on. So it takes ten additional "dummy"
|
||||
cycles until person ten knows the color. When you look closer,
|
||||
you will see that not even person 9 knows its own color yet. This
|
||||
information is still with person 2. Essentially the driver sends additional
|
||||
zeroes to LED 1 as long as it takes for the last color frame to make it
|
||||
down the line to the last LED.
|
||||
"""
|
||||
# Constants
|
||||
MAX_BRIGHTNESS = 0b11111 # Safeguard: Set to a value appropriate for your setup
|
||||
LED_START = 0b11100000 # Three "1" bits, followed by 5 brightness bits
|
||||
|
||||
def __init__(self, num_led, global_brightness=MAX_BRIGHTNESS,
|
||||
order='rgb', bus=0, device=1, max_speed_hz=8000000):
|
||||
self.num_led = num_led # The number of LEDs in the Strip
|
||||
order = order.lower()
|
||||
self.rgb = RGB_MAP.get(order, RGB_MAP['rgb'])
|
||||
# Limit the brightness to the maximum if it's set higher
|
||||
if global_brightness > self.MAX_BRIGHTNESS:
|
||||
self.global_brightness = self.MAX_BRIGHTNESS
|
||||
else:
|
||||
self.global_brightness = global_brightness
|
||||
|
||||
self.leds = [self.LED_START,0,0,0] * self.num_led # Pixel buffer
|
||||
self.spi = spidev.SpiDev() # Init the SPI device
|
||||
self.spi.open(bus, device) # Open SPI port 0, slave device (CS) 1
|
||||
# Up the speed a bit, so that the LEDs are painted faster
|
||||
if max_speed_hz:
|
||||
self.spi.max_speed_hz = max_speed_hz
|
||||
|
||||
def clock_start_frame(self):
|
||||
"""Sends a start frame to the LED strip.
|
||||
|
||||
This method clocks out a start frame, telling the receiving LED
|
||||
that it must update its own color now.
|
||||
"""
|
||||
self.spi.xfer2([0] * 4) # Start frame, 32 zero bits
|
||||
|
||||
|
||||
def clock_end_frame(self):
|
||||
"""Sends an end frame to the LED strip.
|
||||
|
||||
As explained above, dummy data must be sent after the last real colour
|
||||
information so that all of the data can reach its destination down the line.
|
||||
The delay is not as bad as with the human example above.
|
||||
It is only 1/2 bit per LED. This is because the SPI clock line
|
||||
needs to be inverted.
|
||||
|
||||
Say a bit is ready on the SPI data line. The sender communicates
|
||||
this by toggling the clock line. The bit is read by the LED
|
||||
and immediately forwarded to the output data line. When the clock goes
|
||||
down again on the input side, the LED will toggle the clock up
|
||||
on the output to tell the next LED that the bit is ready.
|
||||
|
||||
After one LED the clock is inverted, and after two LEDs it is in sync
|
||||
again, but one cycle behind. Therefore, for every two LEDs, one bit
|
||||
of delay gets accumulated. For 300 LEDs, 150 additional bits must be fed to
|
||||
the input of LED one so that the data can reach the last LED.
|
||||
|
||||
Ultimately, we need to send additional numLEDs/2 arbitrary data bits,
|
||||
in order to trigger numLEDs/2 additional clock changes. This driver
|
||||
sends zeroes, which has the benefit of getting LED one partially or
|
||||
fully ready for the next update to the strip. An optimized version
|
||||
of the driver could omit the "clockStartFrame" method if enough zeroes have
|
||||
been sent as part of "clockEndFrame".
|
||||
"""
|
||||
|
||||
self.spi.xfer2([0xFF] * 4)
|
||||
|
||||
# Round up num_led/2 bits (or num_led/16 bytes)
|
||||
#for _ in range((self.num_led + 15) // 16):
|
||||
# self.spi.xfer2([0x00])
|
||||
|
||||
|
||||
def clear_strip(self):
|
||||
""" Turns off the strip and shows the result right away."""
|
||||
|
||||
for led in range(self.num_led):
|
||||
self.set_pixel(led, 0, 0, 0)
|
||||
self.show()
|
||||
|
||||
|
||||
def set_pixel(self, led_num, red, green, blue, bright_percent=100):
|
||||
"""Sets the color of one pixel in the LED stripe.
|
||||
|
||||
The changed pixel is not shown yet on the Stripe, it is only
|
||||
written to the pixel buffer. Colors are passed individually.
|
||||
If brightness is not set the global brightness setting is used.
|
||||
"""
|
||||
if led_num < 0:
|
||||
return # Pixel is invisible, so ignore
|
||||
if led_num >= self.num_led:
|
||||
return # again, invisible
|
||||
|
||||
# Calculate pixel brightness as a percentage of the
|
||||
# defined global_brightness. Round up to nearest integer
|
||||
# as we expect some brightness unless set to 0
|
||||
brightness = int(ceil(bright_percent*self.global_brightness/100.0))
|
||||
|
||||
# LED startframe is three "1" bits, followed by 5 brightness bits
|
||||
ledstart = (brightness & 0b00011111) | self.LED_START
|
||||
|
||||
start_index = 4 * led_num
|
||||
self.leds[start_index] = ledstart
|
||||
self.leds[start_index + self.rgb[0]] = red
|
||||
self.leds[start_index + self.rgb[1]] = green
|
||||
self.leds[start_index + self.rgb[2]] = blue
|
||||
|
||||
|
||||
def set_pixel_rgb(self, led_num, rgb_color, bright_percent=100):
|
||||
"""Sets the color of one pixel in the LED stripe.
|
||||
|
||||
The changed pixel is not shown yet on the Stripe, it is only
|
||||
written to the pixel buffer.
|
||||
Colors are passed combined (3 bytes concatenated)
|
||||
If brightness is not set the global brightness setting is used.
|
||||
"""
|
||||
self.set_pixel(led_num, (rgb_color & 0xFF0000) >> 16,
|
||||
(rgb_color & 0x00FF00) >> 8, rgb_color & 0x0000FF,
|
||||
bright_percent)
|
||||
|
||||
|
||||
def rotate(self, positions=1):
|
||||
""" Rotate the LEDs by the specified number of positions.
|
||||
|
||||
Treating the internal LED array as a circular buffer, rotate it by
|
||||
the specified number of positions. The number could be negative,
|
||||
which means rotating in the opposite direction.
|
||||
"""
|
||||
cutoff = 4 * (positions % self.num_led)
|
||||
self.leds = self.leds[cutoff:] + self.leds[:cutoff]
|
||||
|
||||
|
||||
def show(self):
|
||||
"""Sends the content of the pixel buffer to the strip.
|
||||
|
||||
Todo: More than 1024 LEDs requires more than one xfer operation.
|
||||
"""
|
||||
self.clock_start_frame()
|
||||
# xfer2 kills the list, unfortunately. So it must be copied first
|
||||
# SPI takes up to 4096 Integers. So we are fine for up to 1024 LEDs.
|
||||
data = list(self.leds)
|
||||
while data:
|
||||
self.spi.xfer2(data[:32])
|
||||
data = data[32:]
|
||||
self.clock_end_frame()
|
||||
|
||||
|
||||
def cleanup(self):
|
||||
"""Release the SPI device; Call this method at the end"""
|
||||
|
||||
self.spi.close() # Close SPI port
|
||||
|
||||
@staticmethod
|
||||
def combine_color(red, green, blue):
|
||||
"""Make one 3*8 byte color value."""
|
||||
|
||||
return (red << 16) + (green << 8) + blue
|
||||
|
||||
|
||||
def wheel(self, wheel_pos):
|
||||
"""Get a color from a color wheel; Green -> Red -> Blue -> Green"""
|
||||
|
||||
if wheel_pos > 255:
|
||||
wheel_pos = 255 # Safeguard
|
||||
if wheel_pos < 85: # Green -> Red
|
||||
return self.combine_color(wheel_pos * 3, 255 - wheel_pos * 3, 0)
|
||||
if wheel_pos < 170: # Red -> Blue
|
||||
wheel_pos -= 85
|
||||
return self.combine_color(255 - wheel_pos * 3, 0, wheel_pos * 3)
|
||||
# Blue -> Green
|
||||
wheel_pos -= 170
|
||||
return self.combine_color(0, wheel_pos * 3, 255 - wheel_pos * 3)
|
||||
|
||||
|
||||
def dump_array(self):
|
||||
"""For debug purposes: Dump the LED array onto the console."""
|
||||
|
||||
print(self.leds)
|
@ -0,0 +1,105 @@
|
||||
|
||||
import time
|
||||
import threading
|
||||
try:
|
||||
import queue as Queue
|
||||
except ImportError:
|
||||
import Queue as Queue
|
||||
|
||||
from .apa102 import APA102
|
||||
from .pattern import Echo, GoogleHome
|
||||
|
||||
|
||||
class PixelRing(object):
|
||||
PIXELS_N = 12
|
||||
|
||||
def __init__(self, pattern='google'):
|
||||
if pattern == 'echo':
|
||||
self.pattern = Echo(show=self.show)
|
||||
else:
|
||||
self.pattern = GoogleHome(show=self.show)
|
||||
|
||||
self.dev = APA102(num_led=self.PIXELS_N)
|
||||
|
||||
self.queue = Queue.Queue()
|
||||
self.thread = threading.Thread(target=self._run)
|
||||
self.thread.daemon = True
|
||||
self.thread.start()
|
||||
self.off()
|
||||
|
||||
def set_brightness(self, brightness):
|
||||
if brightness > 100:
|
||||
brightness = 100
|
||||
|
||||
if brightness > 0:
|
||||
self.dev.global_brightness = int(0b11111 * brightness / 100)
|
||||
|
||||
def change_pattern(self, pattern):
|
||||
if pattern == 'echo':
|
||||
self.pattern = Echo(show=self.show)
|
||||
else:
|
||||
self.pattern = GoogleHome(show=self.show)
|
||||
|
||||
def wakeup(self, direction=0):
|
||||
def f():
|
||||
self.pattern.wakeup(direction)
|
||||
|
||||
self.put(f)
|
||||
|
||||
def listen(self):
|
||||
self.put(self.pattern.listen)
|
||||
|
||||
def think(self):
|
||||
self.put(self.pattern.think)
|
||||
|
||||
wait = think
|
||||
|
||||
def speak(self):
|
||||
self.put(self.pattern.speak)
|
||||
|
||||
def off(self):
|
||||
self.put(self.pattern.off)
|
||||
|
||||
def put(self, func):
|
||||
self.pattern.stop = True
|
||||
self.queue.put(func)
|
||||
|
||||
def _run(self):
|
||||
while True:
|
||||
func = self.queue.get()
|
||||
self.pattern.stop = False
|
||||
func()
|
||||
|
||||
def show(self, data):
|
||||
for i in range(self.PIXELS_N):
|
||||
self.dev.set_pixel(i, int(data[4*i + 1]), int(data[4*i + 2]), int(data[4*i + 3]))
|
||||
|
||||
self.dev.show()
|
||||
|
||||
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||
if rgb:
|
||||
r, g, b = (rgb >> 16) & 0xFF, (rgb >> 8) & 0xFF, rgb & 0xFF
|
||||
for i in range(self.PIXELS_N):
|
||||
self.dev.set_pixel(i, r, g, b)
|
||||
|
||||
self.dev.show()
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
pixel_ring = PixelRing()
|
||||
while True:
|
||||
try:
|
||||
pixel_ring.wakeup()
|
||||
time.sleep(3)
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.speak()
|
||||
time.sleep(6)
|
||||
pixel_ring.off()
|
||||
time.sleep(3)
|
||||
except KeyboardInterrupt:
|
||||
break
|
||||
|
||||
|
||||
pixel_ring.off()
|
||||
time.sleep(1)
|
@ -0,0 +1,145 @@
|
||||
"""
|
||||
LED pattern like Echo
|
||||
"""
|
||||
|
||||
import time
|
||||
|
||||
|
||||
class Echo(object):
|
||||
brightness = 24 * 8
|
||||
|
||||
def __init__(self, show, number=12):
|
||||
self.pixels_number = number
|
||||
self.pixels = [0] * 4 * number
|
||||
|
||||
if not callable(show):
|
||||
raise ValueError('show parameter is not callable')
|
||||
|
||||
self.show = show
|
||||
self.stop = False
|
||||
|
||||
def wakeup(self, direction=0):
|
||||
position = int((direction + 15) / (360 / self.pixels_number)) % self.pixels_number
|
||||
|
||||
pixels = [0, 0, 0, self.brightness] * self.pixels_number
|
||||
pixels[position * 4 + 2] = self.brightness
|
||||
|
||||
self.show(pixels)
|
||||
|
||||
def listen(self):
|
||||
pixels = [0, 0, 0, self.brightness] * self.pixels_number
|
||||
|
||||
self.show(pixels)
|
||||
|
||||
def think(self):
|
||||
half_brightness = int(self.brightness / 2)
|
||||
pixels = [0, 0, half_brightness, half_brightness, 0, 0, 0, self.brightness] * self.pixels_number
|
||||
|
||||
while not self.stop:
|
||||
self.show(pixels)
|
||||
time.sleep(0.2)
|
||||
pixels = pixels[-4:] + pixels[:-4]
|
||||
|
||||
def speak(self):
|
||||
step = int(self.brightness / 12)
|
||||
position = int(self.brightness / 2)
|
||||
while not self.stop:
|
||||
pixels = [0, 0, position, self.brightness - position] * self.pixels_number
|
||||
self.show(pixels)
|
||||
time.sleep(0.01)
|
||||
if position <= 0:
|
||||
step = int(self.brightness / 12)
|
||||
time.sleep(0.4)
|
||||
elif position >= int(self.brightness / 2):
|
||||
step = - int(self.brightness / 12)
|
||||
time.sleep(0.4)
|
||||
|
||||
position += step
|
||||
|
||||
def off(self):
|
||||
self.show([0] * 4 * 12)
|
||||
|
||||
class GoogleHome(object):
|
||||
def __init__(self, show):
|
||||
self.basis = [0] * 4 * 12
|
||||
self.basis[0 * 4 + 1] = 8
|
||||
self.basis[3 * 4 + 1] = 4
|
||||
self.basis[3 * 4 + 2] = 4
|
||||
self.basis[6 * 4 + 2] = 8
|
||||
self.basis[9 * 4 + 3] = 8
|
||||
|
||||
self.pixels = self.basis
|
||||
|
||||
if not callable(show):
|
||||
raise ValueError('show parameter is not callable')
|
||||
|
||||
self.show = show
|
||||
self.stop = False
|
||||
|
||||
def wakeup(self, direction=0):
|
||||
position = int((direction + 90 + 15) / 30) % 12
|
||||
|
||||
basis = self.basis[position*-4:] + self.basis[:position*-4]
|
||||
|
||||
pixels = [v * 25 for v in basis]
|
||||
self.show(pixels)
|
||||
time.sleep(0.1)
|
||||
|
||||
pixels = pixels[-4:] + pixels[:-4]
|
||||
self.show(pixels)
|
||||
time.sleep(0.1)
|
||||
|
||||
for i in range(2):
|
||||
new_pixels = pixels[-4:] + pixels[:-4]
|
||||
|
||||
self.show([v/2+pixels[index] for index, v in enumerate(new_pixels)])
|
||||
pixels = new_pixels
|
||||
time.sleep(0.1)
|
||||
|
||||
self.show(pixels)
|
||||
self.pixels = pixels
|
||||
|
||||
def listen(self):
|
||||
pixels = self.pixels
|
||||
for i in range(1, 25):
|
||||
self.show([(v * i / 24) for v in pixels])
|
||||
time.sleep(0.01)
|
||||
|
||||
def think(self):
|
||||
pixels = self.pixels
|
||||
|
||||
while not self.stop:
|
||||
pixels = pixels[-4:] + pixels[:-4]
|
||||
self.show(pixels)
|
||||
time.sleep(0.2)
|
||||
|
||||
t = 0.1
|
||||
for i in range(0, 5):
|
||||
pixels = pixels[-4:] + pixels[:-4]
|
||||
self.show([(v * (4 - i) / 4) for v in pixels])
|
||||
time.sleep(t)
|
||||
t /= 2
|
||||
|
||||
self.pixels = pixels
|
||||
|
||||
def speak(self):
|
||||
pixels = self.pixels
|
||||
step = 1
|
||||
brightness = 5
|
||||
while not self.stop:
|
||||
self.show([(v * brightness / 24) for v in pixels])
|
||||
time.sleep(0.02)
|
||||
|
||||
if brightness <= 5:
|
||||
step = 1
|
||||
time.sleep(0.4)
|
||||
elif brightness >= 24:
|
||||
step = -1
|
||||
time.sleep(0.4)
|
||||
|
||||
brightness += step
|
||||
|
||||
def off(self):
|
||||
self.show([0] * 4 * 12)
|
||||
|
||||
|
@ -0,0 +1,26 @@
|
||||
|
||||
|
||||
class PixelRing(object):
|
||||
def __init__(self):
|
||||
pass
|
||||
|
||||
def show(self, data):
|
||||
pass
|
||||
|
||||
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||
pass
|
||||
|
||||
def wakeup(self, angle=None):
|
||||
pass
|
||||
|
||||
def listen(self):
|
||||
pass
|
||||
|
||||
def think(self):
|
||||
pass
|
||||
|
||||
def speak(self):
|
||||
pass
|
||||
|
||||
def off(self):
|
||||
pass
|
@ -0,0 +1,190 @@
|
||||
|
||||
import usb.core
|
||||
import usb.util
|
||||
|
||||
|
||||
class HidDevice:
|
||||
"""
|
||||
This class provides basic functions to access
|
||||
a USB HID device to write an endpoint
|
||||
"""
|
||||
|
||||
def __init__(self, dev, ep_in, ep_out):
|
||||
self.dev = dev
|
||||
self.ep_in = ep_in
|
||||
self.ep_out = ep_out
|
||||
|
||||
def write(self, data):
|
||||
"""
|
||||
write data on the OUT endpoint associated to the HID interface
|
||||
"""
|
||||
self.ep_out.write(data)
|
||||
|
||||
def read(self):
|
||||
return self.ep_in.read(self.ep_in.wMaxPacketSize, -1)
|
||||
|
||||
def close(self):
|
||||
"""
|
||||
close the interface
|
||||
"""
|
||||
usb.util.dispose_resources(self.dev)
|
||||
|
||||
@staticmethod
|
||||
def find(vid=0x2886, pid=0x0007):
|
||||
dev = usb.core.find(idVendor=vid, idProduct=pid)
|
||||
if not dev:
|
||||
return
|
||||
|
||||
config = dev.get_active_configuration()
|
||||
|
||||
# Iterate on all interfaces to find a HID interface
|
||||
ep_in, ep_out = None, None
|
||||
for interface in config:
|
||||
if interface.bInterfaceClass == 0x03:
|
||||
try:
|
||||
if dev.is_kernel_driver_active(interface.bInterfaceNumber):
|
||||
dev.detach_kernel_driver(interface.bInterfaceNumber)
|
||||
except Exception as e:
|
||||
print(e.message)
|
||||
|
||||
for ep in interface:
|
||||
if ep.bEndpointAddress & 0x80:
|
||||
ep_in = ep
|
||||
else:
|
||||
ep_out = ep
|
||||
break
|
||||
|
||||
|
||||
|
||||
if ep_in and ep_out:
|
||||
hid = HidDevice(dev, ep_in, ep_out)
|
||||
|
||||
return hid
|
||||
|
||||
|
||||
class UsbPixelRing:
|
||||
PIXELS_N = 12
|
||||
|
||||
MONO = 1
|
||||
THINK = 3
|
||||
VOLUME = 5
|
||||
CUSTOM = 6
|
||||
|
||||
def __init__(self, hid=None, pattern=None):
|
||||
self.hid = hid if hid else HidDevice.find()
|
||||
if not self.hid:
|
||||
print('No USB device found')
|
||||
|
||||
colors = [0] * 4 * self.PIXELS_N
|
||||
colors[0] = 0x4
|
||||
colors[1] = 0x40
|
||||
colors[2] = 0x4
|
||||
|
||||
colors[4 + 1] = 0x8
|
||||
colors[4 * 11 + 1] = 0x8
|
||||
|
||||
self.direction_template = colors
|
||||
|
||||
def set_brightness(self, brightness):
|
||||
print('Not support to change brightness')
|
||||
|
||||
def change_pattern(self, pattern=None):
|
||||
print('Not support to change pattern')
|
||||
|
||||
def off(self):
|
||||
self.set_color(rgb=0)
|
||||
|
||||
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||
if rgb:
|
||||
self.write(0, [self.MONO, rgb & 0xFF, (rgb >> 8) & 0xFF, (rgb >> 16) & 0xFF])
|
||||
else:
|
||||
self.write(0, [self.MONO, b, g, r])
|
||||
|
||||
def think(self):
|
||||
self.write(0, [self.THINK, 0, 0, 0])
|
||||
|
||||
wait = think
|
||||
|
||||
speak = think
|
||||
|
||||
def set_volume(self, pixels):
|
||||
self.write(0, [self.VOLUME, 0, 0, pixels])
|
||||
|
||||
def wakeup(self, angle=0):
|
||||
if angle < 0 or angle > 360:
|
||||
return
|
||||
|
||||
position = int((angle + 15) % 360 / 30) % self.PIXELS_N
|
||||
colors = self.direction_template[-position*4:] + self.direction_template[:-position*4]
|
||||
|
||||
self.write(0, [self.CUSTOM, 0, 0, 0])
|
||||
self.write(3, colors)
|
||||
|
||||
return position
|
||||
|
||||
def listen(self, angle=0):
|
||||
self.write(0, [self.MONO, 0, 0x10, 0])
|
||||
|
||||
def show(self, data):
|
||||
self.write(0, [self.CUSTOM, 0, 0, 0])
|
||||
self.write(3, data)
|
||||
|
||||
@staticmethod
|
||||
def to_bytearray(data):
|
||||
if type(data) is int:
|
||||
array = bytearray([data & 0xFF])
|
||||
elif type(data) is bytearray:
|
||||
array = data
|
||||
elif type(data) is str or type(data) is bytes:
|
||||
array = bytearray(data)
|
||||
elif type(data) is list:
|
||||
array = bytearray(data)
|
||||
else:
|
||||
raise TypeError('%s is not supported' % type(data))
|
||||
|
||||
return array
|
||||
|
||||
def write(self, address, data):
|
||||
data = self.to_bytearray(data)
|
||||
length = len(data)
|
||||
if self.hid:
|
||||
packet = bytearray([address & 0xFF, (address >> 8) & 0xFF, length & 0xFF, (length >> 8) & 0xFF]) + data
|
||||
self.hid.write(packet)
|
||||
|
||||
def close(self):
|
||||
if self.hid:
|
||||
self.hid.close()
|
||||
|
||||
def __call__(self, data):
|
||||
self.write(3, data)
|
||||
|
||||
|
||||
def find():
|
||||
hid = HidDevice.find()
|
||||
|
||||
if hid:
|
||||
pixel_ring = UsbPixelRing(hid)
|
||||
return pixel_ring
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
import time
|
||||
|
||||
pixel_ring = UsbPixelRing()
|
||||
while True:
|
||||
try:
|
||||
pixel_ring.wakeup(180)
|
||||
time.sleep(3)
|
||||
pixel_ring.listen()
|
||||
time.sleep(3)
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.set_volume(8)
|
||||
time.sleep(3)
|
||||
pixel_ring.off()
|
||||
time.sleep(3)
|
||||
except KeyboardInterrupt:
|
||||
break
|
||||
|
||||
pixel_ring.off()
|
||||
|
@ -0,0 +1,122 @@
|
||||
|
||||
import usb.core
|
||||
import usb.util
|
||||
|
||||
|
||||
class PixelRing:
|
||||
TIMEOUT = 8000
|
||||
|
||||
def __init__(self, dev):
|
||||
self.dev = dev
|
||||
|
||||
def trace(self):
|
||||
self.write(0)
|
||||
|
||||
def mono(self, color):
|
||||
self.write(1, [(color >> 16) & 0xFF, (color >> 8) & 0xFF, color & 0xFF, 0])
|
||||
|
||||
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||
if rgb:
|
||||
self.mono(rgb)
|
||||
else:
|
||||
self.write(1, [r, g, b, 0])
|
||||
|
||||
def off(self):
|
||||
self.mono(0)
|
||||
|
||||
def listen(self, direction=None):
|
||||
self.write(2)
|
||||
|
||||
wakeup = listen
|
||||
|
||||
def speak(self):
|
||||
self.write(3)
|
||||
|
||||
def think(self):
|
||||
self.write(4)
|
||||
|
||||
wait = think
|
||||
|
||||
def spin(self):
|
||||
self.write(5)
|
||||
|
||||
def show(self, data):
|
||||
self.write(6, data)
|
||||
|
||||
customize = show
|
||||
|
||||
def set_brightness(self, brightness):
|
||||
self.write(0x20, [brightness])
|
||||
|
||||
def set_color_palette(self, a, b):
|
||||
self.write(0x21, [(a >> 16) & 0xFF, (a >> 8) & 0xFF, a & 0xFF, 0, (b >> 16) & 0xFF, (b >> 8) & 0xFF, b & 0xFF, 0])
|
||||
|
||||
def set_vad_led(self, state):
|
||||
self.write(0x22, [state])
|
||||
|
||||
def set_volume(self, volume):
|
||||
self.write(0x23, [volume])
|
||||
|
||||
def change_pattern(self, pattern):
|
||||
if pattern == 'echo':
|
||||
self.write(0x24, [1])
|
||||
else:
|
||||
self.write(0x24, [0])
|
||||
|
||||
def write(self, cmd, data=[0]):
|
||||
self.dev.ctrl_transfer(
|
||||
usb.util.CTRL_OUT | usb.util.CTRL_TYPE_VENDOR | usb.util.CTRL_RECIPIENT_DEVICE,
|
||||
0, cmd, 0x1C, data, self.TIMEOUT)
|
||||
|
||||
@property
|
||||
def version(self):
|
||||
return self.dev.ctrl_transfer(
|
||||
usb.util.CTRL_IN | usb.util.CTRL_TYPE_VENDOR | usb.util.CTRL_RECIPIENT_DEVICE,
|
||||
0, 0x80 | 0x40, 0x1C, 24, self.TIMEOUT).tostring()
|
||||
|
||||
def close(self):
|
||||
"""
|
||||
close the interface
|
||||
"""
|
||||
usb.util.dispose_resources(self.dev)
|
||||
|
||||
|
||||
def find(vid=0x2886, pid=0x0018):
|
||||
dev = usb.core.find(idVendor=vid, idProduct=pid)
|
||||
if not dev:
|
||||
return
|
||||
|
||||
# configuration = dev.get_active_configuration()
|
||||
|
||||
# interface_number = None
|
||||
# for interface in configuration:
|
||||
# interface_number = interface.bInterfaceNumber
|
||||
|
||||
# if dev.is_kernel_driver_active(interface_number):
|
||||
# dev.detach_kernel_driver(interface_number)
|
||||
|
||||
return PixelRing(dev)
|
||||
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
import time
|
||||
|
||||
pixel_ring = find()
|
||||
print(pixel_ring.version)
|
||||
while True:
|
||||
try:
|
||||
pixel_ring.wakeup(180)
|
||||
time.sleep(3)
|
||||
pixel_ring.listen()
|
||||
time.sleep(3)
|
||||
pixel_ring.think()
|
||||
time.sleep(3)
|
||||
pixel_ring.set_volume(8)
|
||||
time.sleep(3)
|
||||
pixel_ring.off()
|
||||
time.sleep(3)
|
||||
except KeyboardInterrupt:
|
||||
break
|
||||
|
||||
pixel_ring.off()
|
@ -0,0 +1,2 @@
|
||||
spidev
|
||||
pyusb
|
@ -0,0 +1,3 @@
|
||||
|
||||
[bdist_wheel]
|
||||
universal = 1
|
@ -0,0 +1,61 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
"""The setup script."""
|
||||
|
||||
from setuptools import setup, find_packages
|
||||
|
||||
README = \
|
||||
'''
|
||||
RGB LED library for ReSpeaker USB 6+1 Microphone Array, 4 Mic Array for Raspberry Pi
|
||||
to control the pixel ring
|
||||
'''
|
||||
|
||||
|
||||
requirements = [
|
||||
'spidev',
|
||||
'pyusb'
|
||||
]
|
||||
|
||||
setup_requirements = [
|
||||
# TODO: put setup requirements (distutils extensions, etc.) here
|
||||
]
|
||||
|
||||
test_requirements = [
|
||||
'pytest'
|
||||
]
|
||||
|
||||
setup(
|
||||
name='pixel-ring',
|
||||
version='0.1.0',
|
||||
description="respeaker series pixel ring library",
|
||||
long_description=README,
|
||||
author="Yihui Xiong",
|
||||
author_email='yihui.xiong@hotmail.com',
|
||||
url='https://github.com/respeaker/pixel_ring',
|
||||
packages=find_packages(include=['pixel_ring']),
|
||||
include_package_data=True,
|
||||
install_requires=requirements,
|
||||
entry_points={
|
||||
'console_scripts': [
|
||||
'pixel_ring_check=pixel_ring.__init__:main'
|
||||
],
|
||||
},
|
||||
license="GNU General Public License v2",
|
||||
zip_safe=False,
|
||||
keywords='voice doa beamforming kws',
|
||||
classifiers=[
|
||||
'Development Status :: 2 - Pre-Alpha',
|
||||
'Intended Audience :: Developers',
|
||||
'License :: OSI Approved :: GNU General Public License v2 (GPLv2)',
|
||||
'Natural Language :: English',
|
||||
"Programming Language :: Python :: 2",
|
||||
'Programming Language :: Python :: 2.7',
|
||||
'Programming Language :: Python :: 3',
|
||||
'Programming Language :: Python :: 3.4',
|
||||
'Programming Language :: Python :: 3.5',
|
||||
],
|
||||
test_suite='tests',
|
||||
tests_require=test_requirements,
|
||||
setup_requires=setup_requirements,
|
||||
)
|
@ -0,0 +1,7 @@
|
||||
"""
|
||||
It is an empty test as a real test requires to access SPI bus
|
||||
"""
|
||||
|
||||
|
||||
def test_pixel_ring():
|
||||
pass
|
Loading…
Reference in New Issue