Mini interruttore touch cun pannellu di vetru nantu à nRF52832

In l'articulu d'oghje vogliu sparte cun voi un novu prughjettu. Questa volta hè un interruttore toccu cù un pannellu di vetru. U dispusitivu hè compactu, misura 42x42mm (i pannelli di vetru standard anu dimensioni 80x80mm). A storia di stu dispusitivu principia assai tempu fà, circa un annu fà.

Mini interruttore touch cun pannellu di vetru nantu à nRF52832

I primi opzioni eranu nantu à u microcontroller atmega328, ma à a fine tuttu hè finitu cù u microcontroller nRF52832.

Mini interruttore touch cun pannellu di vetru nantu à nRF52832

A parte touch di u dispusitivu corre nantu à chips TTP223. I dui sensori sò serviti da una interruzzione. Impulsatu da una batteria CR2477, via un cunvertitore boost in u chip TPS610981 | Datasheet.

Mini interruttore touch cun pannellu di vetru nantu à nRF52832
Mini interruttore touch cun pannellu di vetru nantu à nRF52832

U dispusitivu implementa un circuitu di spegnimentu cù transistori à effettu di campu. Dopu appughjà u buttone, u microcontroller stessu intercepte u cuntrollu di l'energia è dopu u buttone pò esse usatu per i modi di serviziu (in u mo casu, questu hè l'accoppiamentu cù l'altri dispositi, spegne u putere è resetting à i paràmetri di fabbrica).

Ci sò 2 LED rgb per indicà stati è modi di serviziu. Un emettitore piezo hè statu ancu aghjuntu per simulà un clic quandu toccu i buttoni toccu è l'indicazione di u sonu di i modi di serviziu. I LED è l'emettitore piezo pò esse attivati ​​è spenti à a discrezione di l'utilizatori. Questu hè fattu per mezu di u cuntrollu di a casa intelligente mandendu cumandamenti à i sensori tecnichi; l'utilizatore pò ancu cambià l'intervalli per l'inviu di a carica di a batteria è u livellu di signale attraversu u cuntrollu di a casa intelligente. In u mo casu hè MAJORDOMO.

U cunsumu in u modu di trasmissione hè 7mA (250kbit, 10ms), u cunsumu in u sonnu hè 40µA, u cunsumu in u statu off hè menu di 1µA (=consumu di u cunvertitore boost in modalità "idle"). Rx, tx, connettore swd per a prugrammazione hè furnitu. Un connettore miniatura 2x3p cù un pitch di 1.27 hè utilizatu. Un adattatore speciale hè fattu per a prugrammazione.

Mini interruttore touch cun pannellu di vetru nantu à nRF52832

Comu sempre, u funziunamentu di u dispusitivu hè basatu nantu à u protocolu MySensors. Stu interruttore toccu hè previstu per esse usatu in u sistema di cuntrollu di persiane roller. Ma in generale, l'applicazione hè limitata solu da a vostra imaginazione. Per esempiu, u mo figliolu (7 anni) hà digià fattu 3 ordini per e versioni di switch: per accende è spegne a luce in un toilette cù una bagnera (serà muntatu pocu da u pianu), per accende a luce in un bagnu. corridore longu è scuru quandu viaghja à un toilette cù una bagnera, è un altru cum'è lettu, per accende rapidamente a luce in a vostra stanza per chì i mostri scappanu.

Mini interruttore touch cun pannellu di vetru nantu à nRF52832
Mini interruttore touch cun pannellu di vetru nantu à nRF52832
Mini interruttore touch cun pannellu di vetru nantu à nRF52832

U casu hè statu tradiziunale stampatu nantu à una stampante SLA, u dispusitivu hè in miniatura, u casu hè statu chjucu, l'usu di sta tecnulugia di stampa hè ghjustificatu.

Vede u mudellu stampatuMini interruttore touch cun pannellu di vetru nantu à nRF52832
Mini interruttore touch cun pannellu di vetru nantu à nRF52832
Mini interruttore touch cun pannellu di vetru nantu à nRF52832

I magneti sò incollati in u casu è a tappa di u compartimentu di a batteria.

Video cù testi di stu dispusitivu:



Per quelli chì volenu ripetiri:

Pruvate u codice di u prugramma per un interruttore in un sistema di cuntrollu di persiane per Arduino IDE

Câblage Arduino

int8_t timer_status = 0;
boolean sens_flag1 = 0;
boolean sens_flag2 = 0;
boolean switch_a = 0;
boolean switch_b = 0;
uint16_t temp;
float vcc;
int battery;
int old_battery;
uint32_t oldmillis;
uint32_t newmillis;
uint32_t interrupt_time;
uint32_t SLEEP_TIME = 7000;
uint32_t SLEEP_TIME_W;
uint32_t SLEEP_TIME_W2;
int NrfRSSI;
uint16_t NrfRSSI2;
boolean wait_off;
//#define MY_DEBUG
#define MY_DISABLED_SERIAL
#define MY_RADIO_NRF5_ESB
#define MY_PASSIVE_NODE
#define MY_NODE_ID 120
#define MY_PARENT_NODE_ID 0
#define MY_PARENT_NODE_IS_STATIC
#define MY_TRANSPORT_UPLINK_CHECK_DISABLED
#define POWER_CHILD_ID 110
#define UP_POWER_SWITCH_ID 1
#define DOWN_POWER_SWITCH_ID 2
#define CHILD_ID_nRF52_RSSI_RX 3
#define BAT_COOF 0.0092957746478873
#define BAT_MIN 200
#define BAT_MAX 290
#include <MySensors.h>
MyMessage upMsg(UP_POWER_SWITCH_ID, V_STATUS);
MyMessage downMsg(DOWN_POWER_SWITCH_ID, V_STATUS);
MyMessage powerMsg(POWER_CHILD_ID, V_VAR1);
MyMessage msgRF52RssiReceiv(CHILD_ID_nRF52_RSSI_RX, V_VAR1);
void preHwInit() {
pinMode(31, OUTPUT); //power management pin
digitalWrite(31, HIGH);
delay(3000);
pinMode(3, INPUT); // on off mode button
pinMode(25, OUTPUT); // sens1 led
pinMode(26, OUTPUT); // sens1 led
pinMode(27, OUTPUT); // sens1 led
pinMode(6, OUTPUT); // sens21 led
pinMode(7, OUTPUT); // sens2 led
pinMode(8, OUTPUT); // sens2 led
pinMode(28, OUTPUT); // bizzer
pinMode(2, INPUT); // common interrupt for touch sensors
pinMode(9, INPUT); // touch sensors1
pinMode(10, INPUT); //touch sensors2
pinMode(29, INPUT); // battery
digitalWrite(28, LOW);
digitalWrite(27, HIGH);
digitalWrite(26, HIGH);
digitalWrite(25, HIGH);
digitalWrite(6, HIGH);
digitalWrite(7, HIGH);
digitalWrite(8, HIGH);
}
void before()
{
NRF_POWER->DCDCEN = 1;
analogReadResolution(12);
disableNfc();
turnOffAdc();
digitalWrite(25, LOW);
digitalWrite(6, LOW);
wait(200);
digitalWrite(25, HIGH);
digitalWrite(6, HIGH);
wait(100);
playSound0();
wait(100);
digitalWrite(25, LOW);
digitalWrite(6, LOW);
wait(200);
digitalWrite(25, HIGH);
digitalWrite(6, HIGH);
wait(3000);
digitalWrite(27, LOW);
digitalWrite(8, LOW);
wait(200);
digitalWrite(27, HIGH);
digitalWrite(8, HIGH);
wait(400);
digitalWrite(6, LOW);
digitalWrite(25, LOW);
wait(200);
digitalWrite(6, HIGH);
digitalWrite(25, HIGH);
wait(400);
digitalWrite(26, LOW);
digitalWrite(7, LOW);
wait(200);
digitalWrite(26, HIGH);
digitalWrite(7, HIGH);
wait(1000);
digitalWrite(26, LOW);
digitalWrite(7, LOW);
}
void setup()
{
digitalWrite(26, HIGH);
digitalWrite(7, HIGH);
wait(50);
playSound();
wait(2000);
readBatLev();
wait(200);
SLEEP_TIME_W = SLEEP_TIME;
}
void presentation()
{
sendSketchInfo("EFEKTA ON|OFF NODE 2CH", "1.0");
wait(100);
present(POWER_CHILD_ID, S_CUSTOM, "BATTERY DATA");
wait(100);
present(UP_POWER_SWITCH_ID, S_BINARY, "UP SWITCH");
wait(100);
present(DOWN_POWER_SWITCH_ID, S_BINARY, "DOWN SWITCH");
}
void loop()
{
if (sens_flag1 == 0 && sens_flag2 == 0) {
if (switch_a == 0 && switch_b == 0) {
timer_status = sleep(digitalPinToInterrupt(2), RISING, digitalPinToInterrupt(3), RISING, 3600000, false);
wait_off = 1;
} else {
//oldmillis = millis();
timer_status = sleep(digitalPinToInterrupt(2), RISING, digitalPinToInterrupt(3), RISING, SLEEP_TIME_W, false);
wait_off = 0;
}
}
if (timer_status == 3) {
wait(100);
digitalWrite(27, LOW);
digitalWrite(8, LOW);
wait(2000);
digitalWrite(27, HIGH);
digitalWrite(8, HIGH);
wait(100);
digitalWrite(31, LOW);
}
if (timer_status == 2) {
if (digitalRead(9) == HIGH && sens_flag1 == 0 && switch_b == 0) {
sens_flag1 = 1;
if (switch_a == 0) {
oldmillis = millis();
SLEEP_TIME_W = SLEEP_TIME;
switch_a = 1;
digitalWrite(6, LOW);
wait(10);
playSound1();
wait(20);
playSound2();
wait(50);
send(upMsg.set(switch_a));
wait(200);
} else {
switch_a = 0;
digitalWrite(6, HIGH);
wait(10);
playSound2();
wait(20);
playSound1();
wait(50);
send(upMsg.set(switch_a));
wait(200);
}
}
if (digitalRead(10) == HIGH && sens_flag2 == 0 && switch_a == 0) {
sens_flag2 = 1;
if (switch_b == 0) {
oldmillis = millis();
SLEEP_TIME_W = SLEEP_TIME;
switch_b = 1;
digitalWrite(25, LOW);
wait(10);
playSound1();
wait(20);
playSound2();
wait(50);
send(downMsg.set(switch_b));
wait(200);
} else {
switch_b = 0;
digitalWrite(25, HIGH);
wait(10);
playSound2();
wait(20);
playSound1();
wait(50);
send(downMsg.set(switch_b));
wait(200);
}
}
if (digitalRead(9) == LOW && sens_flag1 == 1) {
sens_flag1 = 0;
}
if (digitalRead(10) == LOW && sens_flag2 == 1) {
sens_flag2 = 0;
}
if (switch_a == 1 || switch_b == 1) {
if (wait_off == 0) {
newmillis = millis();
wait(10);
SLEEP_TIME_W2 = SLEEP_TIME_W;
wait(10);
interrupt_time = newmillis - oldmillis;
wait(10);
SLEEP_TIME_W = SLEEP_TIME_W2 - interrupt_time;
wait(10);
Serial.print("WAS IN A SLEEP: ");
Serial.print(newmillis - oldmillis);
Serial.println(" MILLISECONDS");
if (SLEEP_TIME_W < 1000) {
if (switch_a == 1) {
switch_a = 0;
digitalWrite(6, HIGH);
wait(10);
playSound2();
wait(20);
playSound1();
wait(50);
send(upMsg.set(switch_a));
wait(200);
}
if (switch_b == 1) {
switch_b = 0;
digitalWrite(25, HIGH);
wait(10);
playSound2();
wait(20);
playSound1();
wait(50);
send(downMsg.set(switch_b));
wait(200);
}
SLEEP_TIME_W = SLEEP_TIME;
wait(50);
}
Serial.println(SLEEP_TIME);
Serial.println(SLEEP_TIME_W);
Serial.println(SLEEP_TIME_W2);
Serial.print("GO TO SLEEP FOR: ");
Serial.print(SLEEP_TIME_W);
Serial.println(" MILLISECONDS");
}
oldmillis = millis();
}
}
if (timer_status == -1) {
if (switch_a == 1 || switch_b == 1) {
if (switch_a == 1) {
switch_a = 0;
digitalWrite(6, HIGH);
wait(10);
playSound2();
wait(20);
playSound1();
wait(50);
send(upMsg.set(switch_a));
wait(200);
}
if (switch_b == 1) {
switch_b = 0;
digitalWrite(25, HIGH);
wait(10);
playSound2();
wait(20);
playSound1();
wait(50);
send(downMsg.set(switch_b));
wait(200);
}
} else {
readBatLev();
}
}
}
void disableNfc() {
NRF_NFCT->TASKS_DISABLE = 1;
NRF_NVMC->CONFIG = 1;
NRF_UICR->NFCPINS = 0;
NRF_NVMC->CONFIG = 0;
}
void turnOffAdc() {
if (NRF_SAADC->ENABLE) {
NRF_SAADC->TASKS_STOP = 1;
while (NRF_SAADC->EVENTS_STOPPED) {}
NRF_SAADC->ENABLE = 0;
while (NRF_SAADC->ENABLE) {}
}
}
void myTone(uint32_t j, uint32_t k) {
j = 500000 / j;
k += millis();
while (k > millis()) {
digitalWrite(28, HIGH); delayMicroseconds(j);
digitalWrite(28, LOW ); delayMicroseconds(j);
}
}
void playSound0() {
myTone(1300, 50);
wait(20);
myTone(1300, 50);
wait(50);
}
void playSound() {
myTone(700, 30); 
wait(10);
myTone(700, 30);
wait(10);
myTone(700, 30);
wait(50);
}
void playSound1() {
myTone(200, 10);
wait(10);
myTone(400, 5);
wait(30);
}
void playSound2() {
myTone(400, 10);
wait(10);
myTone(200, 5);
wait(30);
}
void readBatLev() {
temp = analogRead(29);
vcc = temp * 0.0033 * 100;
battery = map((int)vcc, BAT_MIN, BAT_MAX, 0, 100);
if (battery < 0) {
battery = 0;
}
if (battery > 100) {
battery = 100;
}
sendBatteryLevel(battery, 1);
wait(2000, C_INTERNAL, I_BATTERY_LEVEL);
send(powerMsg.set(temp));
wait(200);
NrfRSSI = transportGetReceivingRSSI();
NrfRSSI2 = map(NrfRSSI, -85, -40, 0, 100);
if (NrfRSSI2 < 0) {
NrfRSSI2 = 0;
}
if (NrfRSSI2 > 100) {
NrfRSSI2 = 100;
}
send(msgRF52RssiReceiv.set(NrfRSSI2));
wait(200);
}

File di casu in stl - google drive

File PCB Gerber - google drive

Per e dumande nantu à stu sviluppu, nantu à e difficultà in i vostri sviluppi nantu à Arduinos è Mysensors venenu sempre in salvezza in u nostru chat di telegramma - https://t.me/mysensors_rus.

Source: www.habr.com

Add a comment