Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Вітаю всіх читачів розділу "DIY або Зроби сам" на Habr! Сьогоднішня стаття буде про сенсорний вимикач на мікросхемі TTP223 даташіт. Вимикач працює на мікроконтролері nRF52832 даташіт, використаний модуль YJ-17103 з друкованою антеною та роз'ємом для зовнішньої антени MHF4. Сенсорний вимикач працює від батарей CR2430 або CR2450. Споживання в режимі передачі не більше 8мА, в режимі сну не більше 6мкА.
Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Як і всі попередні проекти, цей теж є Arduino проектом, програма написана в Arduino IDE. В основі програмної реалізації пристрою використано протокол Mysensors | GitHub бібліотеки, GitHub підтримки плат nRF5 у Mysensors. Англомовний форум спільноти http://forum.mysensors.org, російськомовний форум спільноти http://mysensors.ru/forum/
(Для охочих вивчити Документація, Послідовний протокол, API, протокол, Парсер | для бажаючих сприяти (внески) у розвитку проекту - Документація)

Плата сенсорного вимикача розроблялася у програмі Діптрейс, з урахуванням подальшого виготовлення методом Лазерної Прасної Технології (ЛУТ). Плата розроблялася у розмірах 60х60мм (стандартна скляна панель має розміри 80х80мм). Схема була роздрукована на сторінках журналу Антена і перенесена праскою Bosсh з налаштуванням «Льон» (максимальна потужність) на двосторонню фольговану склотекстолітову плату 1.5мм, 35мкм (за відсутністю іншої).
Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Травлення вироблялося розчином хлорного заліза, попередньо приготованим у пропорціях 1.5ч.ложки на 250мл теплої води. Процес зайняв 15 хвилин.
Свердління отворів під міжшарові переходи та під кріплення батарейного тримача виконувалося міні-дрилем DREMEL 3000 встановленою на стійці для свердління DREMEL 220. Отвори під міжшарові переходи були просвердлені свердлом 0,4мм, отвори під тримач батарейок1,1 свердл. Обрізка за межами плати була виконана тим же міні-дрилем з насадкою DREMEL 540 (Коло відрізне d=32.0мм). Обрізання проводилося у респіраторі.
Лудіння витравленої плати було виконано за допомогою сплаву Розе, у водному розчині (1 ч.ложка кристалізованої лимонної кислоти на 300мл води).

Процес паяння зайняв близько години, основна частина часу була витрачена на пропаювання дроту (луджена, діаметром 0.4мм) в отворах для міжшарових переходів.

Відмивалася плата аерозольним очисником FLUX OFF.
Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Розробка корпусу пристрою була виконана у тривимірному редакторі автоматизованого проектування. Розміри корпусу 78,5mm X 78,5mm X 12mm.
Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Закінчена модель корпусу і кришки відсіку батареї була збережена у форматі STL, далі необхідно було виконати підготовку цих моделей для друку на SLA принтері (додавання підтримки, орієнтація). На цьому етапі виникла невелика проблема, оскільки область друку у побутових принтерів SLA невелика. Модель корпусу пристрою в оптимальній щодо часу друку положенні не вміщалася в розміри області друку. При розміщенні моделі під 45 градусів так само дало невтішний результат, вага підтримки виходила рівною вагою моделі корпусу. Було ухвалено рішення друкувати модель вертикально, зробивши підтримку на одній з лицьових сторін, яка заздалегідь погодилася з фактом пост обробки. Друк корпусу зайняв 5 годин з налаштуванням шару в 50 мікрон. Далі була виконана обробка за допомогою дуже дрібнозернистого наждакового паперу (номер не напишу, тому що не знаю:)). Кришка батарейного відсіку друкувалася 40 хвилин.
Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Скляні панелі з Аліекспресом продаються з вже приклеєною пластмасовою рамкою, проблем зі зняттям рамки не виникло. Знімав попередньо розігрів скляну панель звичайним феном.
Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Розсіювач для LED підсвічування був зроблений з двостороннього скотчу з акриловим адгезивом 3M 9088-200. Для флуоресцентного підсвічування на вибір було кілька матеріалів, китайська стрічка, що клеїться, і папір, що клеїться, нарізаний у стрічки вітчизняної компанії Люмінофор. Вибір був зроблений на користь вітчизняного виробника, за моїми відчуттями світило яскравіше та довше. Квадрат із паперу з флуоресцентним пігментом був наклеєний зверху на двосторонній скотч 3M 9088-200.

Приклеювання скла до корпусу вимикача було виконано за допомогою двостороннього скотчу з акриловим адгезивом 3M VHB 4910.
Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Кришка фіксувалася гвинтом M 1,4 Х 5мм.

Собівартість пристрою становила 890 руб.

Далі мала бути програмна частина. Без проблем не обійшлося. Виявляється сенсорні мікросхеми TTP223 добре працюють при стабілізованому живленні в.3.3в і не дуже добре при живленні безпосередньо від добре розрядженої батареї. При старті пристрою з живленням в районі 2.5v, плюс після додаткової «просідання» при відпрацюванні презентації Mysensors мікросхема TTP223 (відразу після калібрування) викликала переривання МК, оскільки знаходилася з активним тригером.

Було змінено схему подачі живлення на мікросхему (управлінням харчуванням TTP223 c gpio МК), підведено додаткову землю, на лініях rgb led (які проходять з іншого боку плати ємнісного сенсора) було замінено резистори з більш високим опором. Також у ПЗ було додано: активація живлення для ємнісної мікросхеми після старту фреймворку Mysensors та відпрацювання презентації. Збільшено вдвічі затримку для автокалібрування мікросхеми TTP223 при подачі на неї живлення. Всі ці зміни повністю усунули цю проблему.

Перед переглядом коду програми рекомендую познайомитись із базовою структурою скетчів у Mysensors.void before()
{
// Дополнительная функция, если сравнивать со стандартной структурой Ардуино скетчей, то before() это подобие setup(), отработка происходит до инициализации транспортного уровня Mysensors, рекомендуется например для инициализации устройств SPI
}

void setup()
{

}

void presentation()
{
//Тут происходит презентация ноды и ее сенсоров на контролере через маршрутизатор
sendSketchInfo("Name of my sensor node", "1.0"); // презентация названия ноды, версии ПО
present(CHILD_ID, S_WHATEVER, "Description"); // презентация сенсоров ноды, описания сенсоров
}

void loop()
{

}

Тестовий код програми сенсорного вимикача:test_sens.ino
/**
ТЕСТОВЫЙ СКЕТЧ СЕНСОРНОГО ВЫКЛЮЧАТЕЛЯ С ПРЕРЫВАНИЯМИ НА NRF_LPCOMP
*/
bool button_flag;
bool sens_flag;
bool send_flag;
bool detection;
bool nosleep;
byte timer;
unsigned long SLEEP_TIME = 21600000; //6 hours
unsigned long oldmillis;
unsigned long newmillis;
unsigned long interrupt_time;
unsigned long SLEEP_TIME_W;
uint16_t currentBatteryPercent;
uint16_t batteryVoltage = 0;
uint16_t battery_vcc_min = 2400;
uint16_t battery_vcc_max = 3000;

#define MY_RADIO_NRF5_ESB
//#define MY_PASSIVE_NODE
#define MY_NODE_ID 30
#define MY_PARENT_NODE_ID 0
#define MY_PARENT_NODE_IS_STATIC
#define MY_TRANSPORT_UPLINK_CHECK_DISABLED
#define IRT_PIN 3 //(PORT0, gpio 5)
#include <MySensors.h>
// see https://www.mysensors.org/download/serial_api_20
#define SENS_CHILD_ID 0
#define CHILD_ID_VOLT 254
MyMessage sensMsg(SENS_CHILD_ID, V_VAR1);
//MyMessage voltMsg(CHILD_ID_VOLT, V_VOLTAGE);

void preHwInit() {
sleep(2000);
pinMode(RED_LED, OUTPUT);
digitalWrite(RED_LED, HIGH);
pinMode(GREEN_LED, OUTPUT);
digitalWrite(GREEN_LED, HIGH);
pinMode(BLUE_LED, OUTPUT);
digitalWrite(BLUE_LED, HIGH);
pinMode(MODE_PIN, INPUT);
pinMode(SENS_PIN, INPUT);
}

void before()
{
NRF_POWER->DCDCEN = 1;
NRF_UART0->ENABLE = 0;
sleep(1000);
digitalWrite(BLUE_LED, LOW);
sleep(150);
digitalWrite(BLUE_LED, HIGH);
}

void presentation() {
sendSketchInfo("EFEKTA Sens 1CH Sensor", "1.1");
present(SENS_CHILD_ID, S_CUSTOM, "SWITCH STATUS");
//present(CHILD_ID_VOLT, S_MULTIMETER, "Battery");
}

void setup() {
digitalWrite(BLUE_LED, LOW);
sleep(100);
digitalWrite(BLUE_LED, HIGH);
sleep(200);
digitalWrite(BLUE_LED, LOW);
sleep(100);
digitalWrite(BLUE_LED, HIGH);
lpComp();
detection = false;
SLEEP_TIME_W = SLEEP_TIME;
pinMode(31, OUTPUT);
digitalWrite(31, HIGH);
/*
while (timer < 10) {
timer++;
digitalWrite(GREEN_LED, LOW);
wait(5);
digitalWrite(GREEN_LED, HIGH);
wait(500);
}
timer = 0;
*/
sleep(7000);
while (timer < 3) {
timer++;
digitalWrite(GREEN_LED, LOW);
sleep(15);
digitalWrite(GREEN_LED, HIGH);
sleep(85);
}
timer = 0;
sleep(1000);
}

void loop() {

if (detection) {
if (digitalRead(MODE_PIN) == 1 && button_flag == 0 && digitalRead(SENS_PIN) == 0) {
//back side button detection
button_flag = 1;
nosleep = 1;
}
if (digitalRead(MODE_PIN) == 1 && button_flag == 1 && digitalRead(SENS_PIN) == 0) {
digitalWrite(RED_LED, LOW);
wait(10);
digitalWrite(RED_LED, HIGH);
wait(50);
}
if (digitalRead(MODE_PIN) == 0 && button_flag == 1 && digitalRead(SENS_PIN) == 0) {
nosleep = 0;
button_flag = 0;
digitalWrite(RED_LED, HIGH);
lpComp_reset();
}

if (digitalRead(SENS_PIN) == 1 && sens_flag == 0 && digitalRead(MODE_PIN) == 0) {
//sens detection
sens_flag = 1;
nosleep = 1;
newmillis = millis();
interrupt_time = newmillis - oldmillis;
SLEEP_TIME_W = SLEEP_TIME_W - interrupt_time;
if (send(sensMsg.set(detection))) {
send_flag = 1;
}
}
if (digitalRead(SENS_PIN) == 1 && sens_flag == 1 && digitalRead(MODE_PIN) == 0) {
if (send_flag == 1) {
while (timer < 10) {
timer++;
digitalWrite(GREEN_LED, LOW);
wait(20);
digitalWrite(GREEN_LED, HIGH);
wait(30);
}
timer = 0;
} else {
while (timer < 10) {
timer++;
digitalWrite(RED_LED, LOW);
wait(20);
digitalWrite(RED_LED, HIGH);
wait(30);
}
timer = 0;
}
}
if (digitalRead(SENS_PIN) == 0 && sens_flag == 1 && digitalRead(MODE_PIN) == 0) {
sens_flag = 0;
nosleep = 0;
send_flag = 0;
digitalWrite(GREEN_LED, HIGH);
sleep(500);
lpComp_reset();
}
if (SLEEP_TIME_W < 60000) {
SLEEP_TIME_W = SLEEP_TIME;
sendBatteryStatus();
}
}
else {
//if (detection == -1) {
SLEEP_TIME_W = SLEEP_TIME;
sendBatteryStatus();
}
if (nosleep == 0) {
oldmillis = millis();
sleep(SLEEP_TIME_W);
}
}

void sendBatteryStatus() {
wait(20);
batteryVoltage = hwCPUVoltage();
wait(2);

if (batteryVoltage > battery_vcc_max) {
currentBatteryPercent = 100;
}
else if (batteryVoltage < battery_vcc_min) {
currentBatteryPercent = 0;
} else {
currentBatteryPercent = (100 * (batteryVoltage - battery_vcc_min)) / (battery_vcc_max - battery_vcc_min);
}

sendBatteryLevel(currentBatteryPercent, 1);
wait(2000, C_INTERNAL, I_BATTERY_LEVEL);
//send(powerMsg.set(batteryVoltage), 1);
//wait(2000, 1, V_VAR1);
}

void lpComp() {
NRF_LPCOMP->PSEL = IRT_PIN;
NRF_LPCOMP->ANADETECT = 1;
NRF_LPCOMP->INTENSET = B0100;
NRF_LPCOMP->ENABLE = 1;
NRF_LPCOMP->TASKS_START = 1;
NVIC_SetPriority(LPCOMP_IRQn, 15);
NVIC_ClearPendingIRQ(LPCOMP_IRQn);
NVIC_EnableIRQ(LPCOMP_IRQn);
}

void s_lpComp() {
if ((NRF_LPCOMP->ENABLE) && (NRF_LPCOMP->EVENTS_READY)) {
NRF_LPCOMP->INTENCLR = B0100;
}
}

void r_lpComp() {
NRF_LPCOMP->INTENSET = B0100;
}

#if __CORTEX_M == 0x04
#define NRF5_RESET_EVENT(event)
event = 0;
(void)event
#else
#define NRF5_RESET_EVENT(event) event = 0
#endif

extern "C" {
void LPCOMP_IRQHandler(void) {
detection = true;
NRF5_RESET_EVENT(NRF_LPCOMP->EVENTS_UP);
NRF_LPCOMP->EVENTS_UP = 0;
MY_HW_RTC->CC[0] = (MY_HW_RTC->COUNTER + 2);
}
}

void lpComp_reset () {
s_lpComp();
detection = false;
NRF_LPCOMP->EVENTS_UP = 0;
r_lpComp();
}

MyBoardNRF5.cpp
#ifdef MYBOARDNRF5
#include <variant.h>

/*
* Pins descriptions. Attributes are ignored by arduino-nrf5 variant.
* Definition taken from Arduino Primo Core with ordered ports
*/
const PinDescription g_APinDescription[]=
{
{ NOT_A_PORT, 0, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // LFCLK
{ NOT_A_PORT, 1, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // LFCLK
{ PORT0, 2, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A0, PWM4, NOT_ON_TIMER},
{ PORT0, 3, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A1, PWM5, NOT_ON_TIMER},
{ PORT0, 4, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A2, PWM6, NOT_ON_TIMER},
{ PORT0, 5, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A3, PWM7, NOT_ON_TIMER},
{ PORT0, 6, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // INT3
{ PORT0, 7, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // INT4
{ PORT0, 8, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), No_ADC_Channel, PWM10, NOT_ON_TIMER}, //USER_LED
{ PORT0, 9, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // NFC1
{ PORT0, 10, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // NFC2
{ PORT0, 11, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // TX
{ PORT0, 12, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // RX
{ PORT0, 13, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // SDA
{ PORT0, 14, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // SCL
{ PORT0, 15, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // SDA1
{ PORT0, 16, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // SCL1
{ PORT0, 17, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // TP4
{ PORT0, 18, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // TP5
{ PORT0, 19, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // INT2
{ PORT0, 20, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // INT1
{ PORT0, 21, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // INT1
{ PORT0, 22, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), No_ADC_Channel, PWM9, NOT_ON_TIMER},
{ PORT0, 23, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), No_ADC_Channel, PWM8, NOT_ON_TIMER},
{ PORT0, 24, PIO_DIGITAL, PIN_ATTR_DIGITAL, No_ADC_Channel, NOT_ON_PWM, NOT_ON_TIMER}, // INT
{ PORT0, 25, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), No_ADC_Channel, PWM11, NOT_ON_TIMER}, //RED_LED
{ PORT0, 26, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), No_ADC_Channel, PWM11, NOT_ON_TIMER}, //GREEN_LED
{ PORT0, 27, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), No_ADC_Channel, PWM11, NOT_ON_TIMER}, //BLUE_LED
{ PORT0, 28, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A4, PWM3, NOT_ON_TIMER},
{ PORT0, 29, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A5, PWM2, NOT_ON_TIMER},
{ PORT0, 30, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A6, PWM1, NOT_ON_TIMER},
{ PORT0, 31, PIO_DIGITAL, (PIN_ATTR_DIGITAL|PIN_ATTR_PWM), ADC_A7, PWM0, NOT_ON_TIMER}
};

// Don't remove this line
#include <compat_pin_mapping.h>

#endif

MyBoardNRF5.h
#ifndef _MYBOARDNRF5_H_
#define _MYBOARDNRF5_H_

#ifdef __cplusplus
extern "C"
{
#endif // __cplusplus

// Number of pins defined in PinDescription array
#define PINS_COUNT (32u)
#define NUM_DIGITAL_PINS (32u)
#define NUM_ANALOG_INPUTS (8u)
#define NUM_ANALOG_OUTPUTS (8u)

/*
* LEDs
*
* This is optional
*
* With My Sensors, you can use
* hwPinMode() instead of pinMode()
* hwPinMode() allows to use advanced modes like OUTPUT_H0H1 to drive LEDs.
* https://github.com/mysensors/MySensors/blob/development/drivers/NRF5/nrf5_wiring_constants.h
*
*/
#define PIN_LED1 (16)
#define PIN_LED2 (15)
#define PIN_LED3 (17)
#define RED_LED (PIN_LED1)
#define GREEN_LED (PIN_LED2)
#define BLUE_LED (PIN_LED3)
#define INTERRUPT_PIN (5)
#define MODE_PIN (25)
#define SENS_PIN (27)

/*
* Analog ports
*
* If you change g_APinDescription, replace PIN_AIN0 with
* port numbers mapped by the g_APinDescription Array.
* You can add PIN_AIN0 to the g_APinDescription Array if
* you want provide analog ports MCU independed, you can add
* PIN_AIN0..PIN_AIN7 to your custom g_APinDescription Array
* defined in MyBoardNRF5.cpp
*/
static const uint8_t A0 = ADC_A0;
static const uint8_t A1 = ADC_A1;
static const uint8_t A2 = ADC_A2;
static const uint8_t A3 = ADC_A3;
static const uint8_t A4 = ADC_A4;
static const uint8_t A5 = ADC_A5;
static const uint8_t A6 = ADC_A6;
static const uint8_t A7 = ADC_A7;

/*
* Serial interfaces
*
* RX and TX are required.
* If you have no serial port, use unused pins
* CTS and RTS are optional.
*/
#define PIN_SERIAL_RX (11)
#define PIN_SERIAL_TX (12)

#ifdef __cplusplus
}
#endif

#endif

Вимикач має сенсорну кнопку та тактову кнопку на звороті пристрою. Дана тактова кнопка використовуватиметься для сервісних режимів, режим прив'язки повітрям, обнулення пристрою. На кнопці реалізований залізний анти брязкіт. Лінія ємнісного сенсора і лінія тактової кнопки через діоди Шотки з'єднані і підключені до аналогового піну p0.05, так само з ємнісного сенсора і кнопки тактової йдуть лінії на піни МК р0.25 і p0.27 для зчитування станів після активації переривання на пін. 0.05. На піні p0.05 активовано переривання через компаратор (NRF_LPCOMP) EVENTS_UP. Натхнення для вирішення задачі отримував тут и тут.

Вимикач був доданий до мережі Mysensors, керованої контролером розумного будинку Мажордомо (сайт проекту)

PHP код для додавання в метод statusUpdate вимикача

if (getGlobal("MysensorsButton01.status")==1) {
if (getGlobal('MysensorsRelay04.status') == 0) {
setGlobal('MysensorsRelay04.status', '1');
} else if (getGlobal('MysensorsRelay04.status') == 1) {
setGlobal('MysensorsRelay04.status', '0');
} 
}

Результат дивіться на відео

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Пізніше був зроблений варіант з підвищуючим перетворювачем, але це не пов'язано з роботою ємнісної мікросхеми TTP223, тут більше бажання в хорошому і рівномірному підсвічуванні при відпрацюванні натискань по всьому терміну роботи від батарейки.

ПодивитисяБездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Github проекту - github.com/smartboxchannel/EFEKTA_WIRELESS_TOUCH_SWITCH

російськомовний сайт спільноти Mysensors

Телеграм чат Mysensors - Швидке вирішення проблем з Mysensors, поради, підказки, установка плат, робота з мікроконтролерами atmega 328, stm32, nRF5 в середовищі Arduino IDE - @mysensors_rus

Небагато фотокБездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Бездротовий сенсорний вимикач з додатковим флуоресцентним підсвічуванням

Джерело: habr.com

Додати коментар або відгук