Приближаются новогодние праздники, и было решено сделать что-то новогоднее с использованием имеющегося Raspberry Pi B-модели. До этого я использовал его в качестве веб-сервера. Затем немного игрался с GPIO на Python заставляя светиться светодиоды.
Что ж, небольшая искусственная елка подсказала мне как можно совместить мое желание с моими навыками. Гирлянда на светодиодах!
Гирлянда? Хм… Я же смогу сам заставить ее мигать как захочу! Я сразу решил заложить возможность легко добавлять различные эффекты ее «мигания». Сделать расширяемое приложение.
Что нужно
Не буду рассказывать как включать и устанавливать ОС на Raspberry Pi. По это написано немало статей. Скажу, что имелся Raspberry Pi Model B с установленной ОС Arch Linux. Думаю, большинство тут описанного будет справедливо и для других GNU/Linux дистрибутивов. Например, Raspbian.
Итак, из харда понадобится: Raspberry Pi, разноцветные светодиоды, резисторы номиналом от 100 Ом до 1 КОм, кнопка, клеммы для GPIO, провода.
Количество светодиодов может быть различным. В моем случае их задействовано было 6. Подключать к GPIO выходам светодиоды необходимо через резисторы. Из-за того, что светодиоды были совершенно разного происхождения, пришлось индивидуально подбирать номинал резисторов чтобы добиться оптимальной яркости. В целом можно обойтись номиналом 330 Ом.
Кнопка нужна чтобы можно было с её помощью переключать эффекты гирлянды.
Подключаем
Схема подключения:
Обратите внимание на порты с пометкой GND. Это земля, их нельзя использовать как вход или выход. Для кнопки резистор не нужен. Защита от замыкания уже предусмотрена в Raspberry Pi.
Вот так это выглядело:
С подключением светодиодов можно экспериментировать. На некоторые порты у меня подключено парралено два светодиода для большего разнообразия.
Подготавливаем программное обеспечение
Нам понадобится Python 3. На Arch Linux я ставил так:
sudo pacman -S python
Возможно на других дистрибутивах пакет будет называться python3.
Также необходим Python-модуль RPi.GPIO. Я устанавливал с помощью easy_install:
sudo easy_install RPi.GPIO
Возможно на других дистрибутивах этот модуль есть в репозитории, так что можете поискать.
После подключения хорошо бы протестировать как работают наши светодиоды и кнопка. Вот небольшой пример как это можно сделать:
#gpio_test.py
import RPi.GPIO as GPIO
import sys
GPIO.setmode(GPIO.BOARD)
led = 8
button = 5
# Устанавливаем порты светодиода и кнопки
GPIO.setup(led, GPIO.OUT)
GPIO.setup(button, GPIO.IN)
# Зажигаем светодиод
GPIO.output(led, 1)
# Ждём пока кнопка не нажата
while GPIO.input(button):
pass
# Как нажали -- тушим светодиод
if GPIO.input(button) == False:
GPIO.output(led, 0)
print("Button pressed")
GPIO.cleanup()
Здесь устанавливаем порт с номером из переменной led как выход, а порт с button как вход. Затем подаём на порт led сигнал 1 (зажгись!) и ждём пока кнопка не нажата. Как только кнопку нажали, тушим светодиод и зачищаемя.
В примере выше тестируем светодиод, подключённый к 8-му порту и кнопку, подключённую к 5-му. Запустить скрипт можно так:
sudo python3 gpio_test.py
Обратите внимание, что выполнять скрипт нужно с root-правами. Например, используя sudo. После запуска светодиод, подключённый к 8-му порту должен зажечься. После нажатия кнопки — потухнуть, и скрипт завершит свою работу.
Меняя значение переменной led в скрипте на номер нужного порта можно протестировать работу всех светодиодов.
Моё ПО
Не буду подробно останавливаться на коде, который я написал далее для нужного мне функционала. Дам ссылку на github: https://github.com/jhekasoft/raspberry-led-garland.
Расскажу как настраивать и писать свои эффекты.
Установка
Поместите приложение в /opt/raspberry-led-garland или создайте симлинк. Например:
sudo ln -s /home/jheka/raspberry-led-garland /opt/raspberry-led-garland
Конечно, размещатся оно может в любом месте, но дальше я буду рассчитывать, что оно лежит именно там.
Настройка
Конфиг находится в JSON-формате в файле settings.json. Если Вы подключили светодиоды и кнопку по схеме, котороя приводилась ранее, то в конфиге можно ничего не менять. Пример файла settings.json:
{
"leds": [
{"num": 8, "state": 0},
{"num": 11, "state": 0},
{"num": 12, "state": 0},
{"num": 13, "state": 0},
{"num": 15, "state": 0},
{"num": 16, "state": 0}
],
"button": {"num": 5},
"effects": ["static", "blink", "slow_blink", "fast_blink", "run", "fast_run", "off"],
"logfile": "/var/log/raspberry-led-garland.log"
}
Здесь в leds перечислены порты подключённых светодиодов (num) и их первоначально состояние (state, 0 — выключен). Количество светодиодов может быть произвольным, главное чтобы портов хватило у GPIO.
Далее в button установлен номер порта кнопки (num).
В effects перечислены имена эффектов в том порядке, в котором они будут переключаться кнопкой. Все эффекты лежат в дирректории effects в виде файла с Python-классом.
Логируется всё в файл, указанный в logfile.
Эффекты
По сути сейчас есть три основных эффекта: статика, мигание и бегущий светодиод.
Как упомянул выше, эффекты расположены в директории effects. Вы может добавлять туда свои эффекты. Также не забудьте добавить их в конфиг. Давайте разберём эффект blink. Вот его содержимое:
#blink.py
import RPi.GPIO as GPIO
class GarlandEffect(object):
delay = 1.0
ledstate = 0
def __init__(self, garland):
self.garland = garland
self.ledstate = 1
def iterate(self):
if not self.garland.checkIterationDelay(self.delay):
return False
self.garland.setLedsState(self.ledstate)
self.garland.gpioOutSetState()
self.ledstate = 1 - self.ledstate
return True
При инициализации (функция __init__) устанавливаем ему объект гирлянды (garland). Это должно происходить практически в каждом эффекте. Далее указываем начальное состояние для всех светодиодов 1 (включены).
При каждой итерации (функция iterate) проеряем, а не прошло ли время, равное установленной задержки в 1 секунду (delay = 1.0). Если нет, то ничего не делаем (return False). Если прошло, то устанавливаем состояние ledstate всем светодиодам и меняем значение переменной состояния на обратное чтобы на следующей итерации оно было другим.
Взглянем ещё на эффект fast_blink:
#fast_blink.py
import effects.blink
class GarlandEffect(effects.blink.GarlandEffect):
delay = 0.1
Как видим, он наследуется от blink и у него установлена задержка в 0.1 секунду. То есть это всё тот же blink, просто с меньшей задержкой между итерацией — светодиоды быстрее мигают.
Обратите внимание на эффект с именем off. В нём при инициализации тушаться все светодиоды и ничего не происходит при итерации. Таким образом реализовано затухание гирлянды.
systemd сервис
Чтобы гирлянда была автономной и работала после старта системы я написал небольшой конфиг для запуска нашего скрипка как сервиса. Файл systemd/garland.service необходимо скопировать в /etc/systemd/system/. Затем выполнить:
sudo systemctl enable garland.service
Скрипт запустится при старте системы. Для немедленного запуска можно выполнить это:
sudo systemctl start garland.service
Надеюсь, не обидетесь за использование systemd. Она на Arch Linux уже давно и я к ней привык. В своё время тоже холиварили по этому поводу.
Демо
Вот как это выгладит:
Надеюсь вы также подключитесь к этому делу и сделаете свои гирлянды с большим количеством светодиодов, а также напишите свои интересные эффекты! С наступающими!
Автор: jhekasoft