mirror of
https://github.com/IoTManagerProject/IoTManager.git
synced 2026-03-30 20:09:14 +03:00
3
.vscode/settings.json
vendored
3
.vscode/settings.json
vendored
@@ -8,6 +8,7 @@
|
|||||||
"vector": "cpp",
|
"vector": "cpp",
|
||||||
"string_view": "cpp",
|
"string_view": "cpp",
|
||||||
"initializer_list": "cpp",
|
"initializer_list": "cpp",
|
||||||
"ranges": "cpp"
|
"ranges": "cpp",
|
||||||
|
"thread": "cpp"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Binary file not shown.
Binary file not shown.
@@ -4,12 +4,12 @@
|
|||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
<meta name="viewport" content="width=device-width,initial-scale=1" />
|
<meta name="viewport" content="width=device-width,initial-scale=1" />
|
||||||
|
|
||||||
<title>IoT Manager 4.3.4</title>
|
<title>IoT Manager 4.3.5</title>
|
||||||
|
|
||||||
<link rel="icon" type="image/png" href="/favicon.ico" />
|
<link rel="icon" type="image/png" href="/favicon.ico" />
|
||||||
<link rel="stylesheet" href="/build/bundle.css?434" />
|
<link rel="stylesheet" href="/build/bundle.css?435" />
|
||||||
|
|
||||||
<script defer src="/build/bundle.js?434"></script>
|
<script defer src="/build/bundle.js?435"></script>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body></body>
|
<body></body>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
"header": "Виртуальные элементы"
|
"header": "Виртуальные элементы"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "1. Логирование в график",
|
"name": "1. График",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "Loging",
|
"subtype": "Loging",
|
||||||
"id": "log",
|
"id": "log",
|
||||||
@@ -20,7 +20,20 @@
|
|||||||
"points": 300
|
"points": 300
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "2. Таймер",
|
"name": "2. График дневного расхода",
|
||||||
|
"type": "Writing",
|
||||||
|
"subtype": "LogingDaily",
|
||||||
|
"id": "log",
|
||||||
|
"widget": "chart3",
|
||||||
|
"page": "Графики",
|
||||||
|
"descr": "Температура",
|
||||||
|
"num": 2,
|
||||||
|
"int": 1,
|
||||||
|
"logid": "t",
|
||||||
|
"points": 365
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "3. Таймер",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "Timer",
|
"subtype": "Timer",
|
||||||
"id": "timer",
|
"id": "timer",
|
||||||
@@ -32,10 +45,10 @@
|
|||||||
"ticker": 1,
|
"ticker": 1,
|
||||||
"repeat": 1,
|
"repeat": 1,
|
||||||
"needSave": 0,
|
"needSave": 0,
|
||||||
"num": 2
|
"num": 3
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "3. Окно ввода числа (переменная)",
|
"name": "4. Окно ввода числа (переменная)",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Variable",
|
"subtype": "Variable",
|
||||||
"id": "value",
|
"id": "value",
|
||||||
@@ -44,10 +57,10 @@
|
|||||||
"descr": "Введите число",
|
"descr": "Введите число",
|
||||||
"int": "0",
|
"int": "0",
|
||||||
"val": "0.0",
|
"val": "0.0",
|
||||||
"num": 3
|
"num": 4
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "4. Окно ввода времени",
|
"name": "5. Окно ввода времени",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Variable",
|
"subtype": "Variable",
|
||||||
"id": "time",
|
"id": "time",
|
||||||
@@ -56,10 +69,10 @@
|
|||||||
"descr": "Введите время",
|
"descr": "Введите время",
|
||||||
"int": "0",
|
"int": "0",
|
||||||
"val": "02:00",
|
"val": "02:00",
|
||||||
"num": 4
|
"num": 5
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "5. Окно ввода даты",
|
"name": "6. Окно ввода даты",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Variable",
|
"subtype": "Variable",
|
||||||
"id": "time",
|
"id": "time",
|
||||||
@@ -68,10 +81,10 @@
|
|||||||
"descr": "Введите дату",
|
"descr": "Введите дату",
|
||||||
"int": "0",
|
"int": "0",
|
||||||
"val": "24.05.2022",
|
"val": "24.05.2022",
|
||||||
"num": 5
|
"num": 6
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "6. Окно ввода текста",
|
"name": "7. Окно ввода текста",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Variable",
|
"subtype": "Variable",
|
||||||
"id": "txt",
|
"id": "txt",
|
||||||
@@ -80,10 +93,10 @@
|
|||||||
"descr": "Введите текст",
|
"descr": "Введите текст",
|
||||||
"int": "0",
|
"int": "0",
|
||||||
"val": "текст",
|
"val": "текст",
|
||||||
"num": 6
|
"num": 7
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "7. Виртуальная кнопка",
|
"name": "8. Виртуальная кнопка",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "VButton",
|
"subtype": "VButton",
|
||||||
"id": "vbtn",
|
"id": "vbtn",
|
||||||
@@ -92,13 +105,26 @@
|
|||||||
"descr": "Кнопка",
|
"descr": "Кнопка",
|
||||||
"int": "0",
|
"int": "0",
|
||||||
"val": "0",
|
"val": "0",
|
||||||
"num": 7
|
"num": 8
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"header": "Сенсоры"
|
"header": "Сенсоры"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "8. AHT20 Температура",
|
"name": "9. Acs712 Ток",
|
||||||
|
"type": "Reading",
|
||||||
|
"subtype": "Acs712",
|
||||||
|
"id": "amp",
|
||||||
|
"widget": "anydataAmp",
|
||||||
|
"page": "Сенсоры",
|
||||||
|
"descr": "Ток",
|
||||||
|
"round": 3,
|
||||||
|
"pin": 39,
|
||||||
|
"int": 5,
|
||||||
|
"num": 9
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "10. AHT20 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Aht20t",
|
"subtype": "Aht20t",
|
||||||
"id": "Temp20",
|
"id": "Temp20",
|
||||||
@@ -108,10 +134,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x38",
|
"addr": "0x38",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 8
|
"num": 10
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "9. AHT20 Влажность",
|
"name": "11. AHT20 Влажность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Aht20h",
|
"subtype": "Aht20h",
|
||||||
"id": "Hum20",
|
"id": "Hum20",
|
||||||
@@ -121,10 +147,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x38",
|
"addr": "0x38",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 9
|
"num": 11
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "10. Аналоговый сенсор",
|
"name": "12. Аналоговый сенсор",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "AnalogAdc",
|
"subtype": "AnalogAdc",
|
||||||
"id": "t",
|
"id": "t",
|
||||||
@@ -138,10 +164,10 @@
|
|||||||
"pin": 0,
|
"pin": 0,
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"avgSteps": 1,
|
"avgSteps": 1,
|
||||||
"num": 10
|
"num": 12
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "11. BME280 Температура",
|
"name": "13. BME280 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Bme280t",
|
"subtype": "Bme280t",
|
||||||
"id": "tmp3",
|
"id": "tmp3",
|
||||||
@@ -151,10 +177,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x77",
|
"addr": "0x77",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 11
|
"num": 13
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "12. BME280 Давление",
|
"name": "14. BME280 Давление",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Bme280p",
|
"subtype": "Bme280p",
|
||||||
"id": "Press3",
|
"id": "Press3",
|
||||||
@@ -164,10 +190,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x77",
|
"addr": "0x77",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 12
|
"num": 14
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "13. BME280 Влажность",
|
"name": "15. BME280 Влажность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Bme280h",
|
"subtype": "Bme280h",
|
||||||
"id": "Hum3",
|
"id": "Hum3",
|
||||||
@@ -177,10 +203,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x77",
|
"addr": "0x77",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 13
|
"num": 15
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "14. BMP280 Температура",
|
"name": "16. BMP280 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Bmp280t",
|
"subtype": "Bmp280t",
|
||||||
"id": "tmp3",
|
"id": "tmp3",
|
||||||
@@ -190,10 +216,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x77",
|
"addr": "0x77",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 14
|
"num": 16
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "15. BMP280 Давление",
|
"name": "17. BMP280 Давление",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Bmp280p",
|
"subtype": "Bmp280p",
|
||||||
"id": "Press3",
|
"id": "Press3",
|
||||||
@@ -203,10 +229,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x77",
|
"addr": "0x77",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 15
|
"num": 17
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "16. DHT11 Температура",
|
"name": "18. DHT11 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Dht1122t",
|
"subtype": "Dht1122t",
|
||||||
"id": "tmp3",
|
"id": "tmp3",
|
||||||
@@ -216,10 +242,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"pin": 0,
|
"pin": 0,
|
||||||
"senstype": "dht11",
|
"senstype": "dht11",
|
||||||
"num": 16
|
"num": 18
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "17. DHT11 Влажность",
|
"name": "19. DHT11 Влажность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Dht1122h",
|
"subtype": "Dht1122h",
|
||||||
"id": "Hum3",
|
"id": "Hum3",
|
||||||
@@ -229,10 +255,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"pin": 0,
|
"pin": 0,
|
||||||
"senstype": "dht11",
|
"senstype": "dht11",
|
||||||
"num": 17
|
"num": 19
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "18. DS18B20 Температура",
|
"name": "20. DS18B20 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Ds18b20",
|
"subtype": "Ds18b20",
|
||||||
"id": "dstmp",
|
"id": "dstmp",
|
||||||
@@ -244,10 +270,10 @@
|
|||||||
"index": 0,
|
"index": 0,
|
||||||
"addr": "",
|
"addr": "",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 18
|
"num": 20
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "19. GY21 Температура",
|
"name": "21. GY21 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "GY21t",
|
"subtype": "GY21t",
|
||||||
"id": "tmp4",
|
"id": "tmp4",
|
||||||
@@ -256,10 +282,10 @@
|
|||||||
"descr": "Температура",
|
"descr": "Температура",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"num": 19
|
"num": 21
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "20. GY21 Влажность",
|
"name": "22. GY21 Влажность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "GY21h",
|
"subtype": "GY21h",
|
||||||
"id": "Hum4",
|
"id": "Hum4",
|
||||||
@@ -268,10 +294,10 @@
|
|||||||
"descr": "Влажность",
|
"descr": "Влажность",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"num": 20
|
"num": 22
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "21. HDC1080 Температура",
|
"name": "23. HDC1080 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Hdc1080t",
|
"subtype": "Hdc1080t",
|
||||||
"id": "Temp1080",
|
"id": "Temp1080",
|
||||||
@@ -281,10 +307,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x40",
|
"addr": "0x40",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 21
|
"num": 23
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "22. HDC1080 Влажность",
|
"name": "24. HDC1080 Влажность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Hdc1080h",
|
"subtype": "Hdc1080h",
|
||||||
"id": "Hum1080",
|
"id": "Hum1080",
|
||||||
@@ -294,10 +320,10 @@
|
|||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x40",
|
"addr": "0x40",
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 22
|
"num": 24
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "23. MAX6675 Температура",
|
"name": "25. MAX6675 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Max6675t",
|
"subtype": "Max6675t",
|
||||||
"id": "maxtmp",
|
"id": "maxtmp",
|
||||||
@@ -308,71 +334,71 @@
|
|||||||
"DO": 12,
|
"DO": 12,
|
||||||
"CS": 13,
|
"CS": 13,
|
||||||
"CLK": 14,
|
"CLK": 14,
|
||||||
"num": 23
|
"num": 25
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "24. PZEM 004t Напряжение",
|
"name": "26. PZEM 004t Напряжение",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004v",
|
"subtype": "Pzem004v",
|
||||||
"id": "V",
|
"id": "v",
|
||||||
"widget": "anydataVlt",
|
"widget": "anydataVlt",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Напряжение",
|
"descr": "Напряжение",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0xF8",
|
"addr": "0xF8",
|
||||||
"num": 24
|
"num": 26
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "25. PZEM 004t Сила тока",
|
"name": "27. PZEM 004t Сила тока",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004a",
|
"subtype": "Pzem004a",
|
||||||
"id": "A",
|
"id": "a",
|
||||||
"widget": "anydataAmp",
|
"widget": "anydataAmp",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Сила тока",
|
"descr": "Сила тока",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0xF8",
|
"addr": "0xF8",
|
||||||
"num": 25
|
"num": 27
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "26. PZEM 004t Мощность",
|
"name": "28. PZEM 004t Мощность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004w",
|
"subtype": "Pzem004w",
|
||||||
"id": "W",
|
"id": "w",
|
||||||
"widget": "anydataWt",
|
"widget": "anydataWt",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Мощность",
|
"descr": "Мощность",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0xF8",
|
"addr": "0xF8",
|
||||||
"num": 26
|
"num": 28
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "27. PZEM 004t Энергия",
|
"name": "29. PZEM 004t Энергия",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004wh",
|
"subtype": "Pzem004wh",
|
||||||
"id": "Wh",
|
"id": "wh",
|
||||||
"widget": "anydataWth",
|
"widget": "anydataWth",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Энергия",
|
"descr": "Энергия",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0xF8",
|
"addr": "0xF8",
|
||||||
"num": 27
|
"num": 29
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "28. PZEM 004t Частота",
|
"name": "30. PZEM 004t Частота",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004hz",
|
"subtype": "Pzem004hz",
|
||||||
"id": "Hz",
|
"id": "hz",
|
||||||
"widget": "anydataHtz",
|
"widget": "anydataHtz",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Энергия",
|
"descr": "Частота",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x77",
|
"addr": "0xF8",
|
||||||
"num": 28
|
"num": 30
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "29. Сканер кнопок 433 MHz",
|
"name": "31. Сканер кнопок 433 MHz",
|
||||||
"num": 29,
|
"num": 31,
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "RCswitch",
|
"subtype": "RCswitch",
|
||||||
"id": "rsw",
|
"id": "rsw",
|
||||||
@@ -381,7 +407,7 @@
|
|||||||
"pinTx": 12
|
"pinTx": 12
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "30. Sht20 Температура",
|
"name": "32. Sht20 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Sht20t",
|
"subtype": "Sht20t",
|
||||||
"id": "tmp2",
|
"id": "tmp2",
|
||||||
@@ -390,10 +416,10 @@
|
|||||||
"descr": "Температура",
|
"descr": "Температура",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 30
|
"num": 32
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "31. Sht20 Влажность",
|
"name": "33. Sht20 Влажность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Sht20h",
|
"subtype": "Sht20h",
|
||||||
"id": "Hum2",
|
"id": "Hum2",
|
||||||
@@ -402,10 +428,10 @@
|
|||||||
"descr": "Влажность",
|
"descr": "Влажность",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 31
|
"num": 33
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "32. Sht30 Температура",
|
"name": "34. Sht30 Температура",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Sht30t",
|
"subtype": "Sht30t",
|
||||||
"id": "tmp30",
|
"id": "tmp30",
|
||||||
@@ -414,10 +440,10 @@
|
|||||||
"descr": "SHT30 Температура",
|
"descr": "SHT30 Температура",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 32
|
"num": 34
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "33. Sht30 Влажность",
|
"name": "35. Sht30 Влажность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Sht30h",
|
"subtype": "Sht30h",
|
||||||
"id": "Hum30",
|
"id": "Hum30",
|
||||||
@@ -426,11 +452,11 @@
|
|||||||
"descr": "SHT30 Влажность",
|
"descr": "SHT30 Влажность",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"num": 33
|
"num": 35
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "34. HC-SR04 Ультразвуковой дальномер",
|
"name": "36. HC-SR04 Ультразвуковой дальномер",
|
||||||
"num": 34,
|
"num": 36,
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Sonar",
|
"subtype": "Sonar",
|
||||||
"id": "sonar",
|
"id": "sonar",
|
||||||
@@ -442,7 +468,7 @@
|
|||||||
"int": 5
|
"int": 5
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "35. UART",
|
"name": "37. UART",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "UART",
|
"subtype": "UART",
|
||||||
"page": "",
|
"page": "",
|
||||||
@@ -452,13 +478,13 @@
|
|||||||
"tx": 12,
|
"tx": 12,
|
||||||
"rx": 13,
|
"rx": 13,
|
||||||
"speed": 9600,
|
"speed": 9600,
|
||||||
"num": 35
|
"num": 37
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"header": "Исполнительные устройства"
|
"header": "Исполнительные устройства"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "36. Кнопка подключенная к пину",
|
"name": "38. Кнопка подключенная к пину",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "ButtonIn",
|
"subtype": "ButtonIn",
|
||||||
"id": "btn",
|
"id": "btn",
|
||||||
@@ -471,10 +497,10 @@
|
|||||||
"pinMode": "INPUT",
|
"pinMode": "INPUT",
|
||||||
"debounceDelay": 50,
|
"debounceDelay": 50,
|
||||||
"fixState": 0,
|
"fixState": 0,
|
||||||
"num": 36
|
"num": 38
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "37. Управление пином",
|
"name": "39. Управление пином",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "ButtonOut",
|
"subtype": "ButtonOut",
|
||||||
"id": "btn",
|
"id": "btn",
|
||||||
@@ -484,10 +510,10 @@
|
|||||||
"int": 0,
|
"int": 0,
|
||||||
"inv": 0,
|
"inv": 0,
|
||||||
"pin": 2,
|
"pin": 2,
|
||||||
"num": 37
|
"num": 39
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "38. Сервопривод",
|
"name": "40. Сервопривод",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "IoTServo",
|
"subtype": "IoTServo",
|
||||||
"id": "servo",
|
"id": "servo",
|
||||||
@@ -498,10 +524,10 @@
|
|||||||
"pin": 12,
|
"pin": 12,
|
||||||
"apin": -1,
|
"apin": -1,
|
||||||
"amap": "0, 4096, 0, 180",
|
"amap": "0, 4096, 0, 180",
|
||||||
"num": 38
|
"num": 40
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "39. Расширитель портов Mcp23017",
|
"name": "41. Расширитель портов Mcp23017",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Mcp23017",
|
"subtype": "Mcp23017",
|
||||||
"id": "Mcp",
|
"id": "Mcp",
|
||||||
@@ -511,10 +537,10 @@
|
|||||||
"int": "0",
|
"int": "0",
|
||||||
"addr": "0x20",
|
"addr": "0x20",
|
||||||
"index": 1,
|
"index": 1,
|
||||||
"num": 39
|
"num": 41
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "40. MP3 плеер",
|
"name": "42. MP3 плеер",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Mp3",
|
"subtype": "Mp3",
|
||||||
"id": "mp3",
|
"id": "mp3",
|
||||||
@@ -524,10 +550,10 @@
|
|||||||
"int": 1,
|
"int": 1,
|
||||||
"pins": "14,12",
|
"pins": "14,12",
|
||||||
"volume": 20,
|
"volume": 20,
|
||||||
"num": 40
|
"num": 42
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "41. PWM ESP8266",
|
"name": "43. PWM ESP8266",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "Pwm8266",
|
"subtype": "Pwm8266",
|
||||||
"id": "pwm",
|
"id": "pwm",
|
||||||
@@ -539,10 +565,10 @@
|
|||||||
"freq": 5000,
|
"freq": 5000,
|
||||||
"val": 0,
|
"val": 0,
|
||||||
"apin": -1,
|
"apin": -1,
|
||||||
"num": 41
|
"num": 43
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "42. Телеграм-Лайт",
|
"name": "44. Телеграм-Лайт",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "TelegramLT",
|
"subtype": "TelegramLT",
|
||||||
"id": "tg",
|
"id": "tg",
|
||||||
@@ -551,13 +577,13 @@
|
|||||||
"descr": "",
|
"descr": "",
|
||||||
"token": "",
|
"token": "",
|
||||||
"chatID": "",
|
"chatID": "",
|
||||||
"num": 42
|
"num": 44
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"header": "Экраны"
|
"header": "Экраны"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "43. LCD экран 2004",
|
"name": "45. LCD экран 2004",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Lcd2004",
|
"subtype": "Lcd2004",
|
||||||
"id": "Lcd",
|
"id": "Lcd",
|
||||||
@@ -569,10 +595,10 @@
|
|||||||
"size": "20,4",
|
"size": "20,4",
|
||||||
"coord": "0,0",
|
"coord": "0,0",
|
||||||
"id2show": "id датчика",
|
"id2show": "id датчика",
|
||||||
"num": 43
|
"num": 45
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "44. LCD экран 1602",
|
"name": "46. LCD экран 1602",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Lcd2004",
|
"subtype": "Lcd2004",
|
||||||
"id": "Lcd",
|
"id": "Lcd",
|
||||||
@@ -584,6 +610,6 @@
|
|||||||
"size": "16,2",
|
"size": "16,2",
|
||||||
"coord": "0,0",
|
"coord": "0,0",
|
||||||
"id2show": "id датчика",
|
"id2show": "id датчика",
|
||||||
"num": 44
|
"num": 46
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
@@ -47,7 +47,7 @@
|
|||||||
"name": "anydataWth",
|
"name": "anydataWth",
|
||||||
"label": "Энергия",
|
"label": "Энергия",
|
||||||
"widget": "anydata",
|
"widget": "anydata",
|
||||||
"after": "Wt/hr",
|
"after": "kWt/Hr",
|
||||||
"icon": "speedometer"
|
"icon": "speedometer"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -103,34 +103,27 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "chart1",
|
"name": "chart1",
|
||||||
"label": "График1",
|
"label": "График без точек",
|
||||||
"widget": "chart",
|
"widget": "chart",
|
||||||
"dateFormat": "HH:mm",
|
"dateFormat": "HH:mm",
|
||||||
"maxCount": 255,
|
"maxCount": 86400,
|
||||||
"pointRadius": 0
|
"pointRadius": 0
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "chart2",
|
"name": "chart2",
|
||||||
"label": "График2",
|
"label": "График с точками",
|
||||||
"widget": "chart",
|
"widget": "chart",
|
||||||
"maxCount": 255,
|
"maxCount": 86400,
|
||||||
"dateFormat": "HH:mm"
|
"dateFormat": "HH:mm"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "chart3",
|
"name": "chart3",
|
||||||
"label": "График3",
|
"label": "График Дневной",
|
||||||
"widget": "chart",
|
"widget": "chart",
|
||||||
"dateFormat": "DD.MM.YYYY",
|
"dateFormat": "DD.MM.YYYY",
|
||||||
"maxCount": 255,
|
"maxCount": 86400,
|
||||||
"type": "bar"
|
"type": "bar"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"name": "chart4",
|
|
||||||
"label": "График4",
|
|
||||||
"widget": "chart",
|
|
||||||
"maxCount": 255,
|
|
||||||
"dateFormat": "DD.MM.YYYY"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"name": "fillgauge",
|
"name": "fillgauge",
|
||||||
"label": "Бочка",
|
"label": "Бочка",
|
||||||
@@ -210,10 +203,10 @@
|
|||||||
"status": 0
|
"status": 0
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "anydataVlt",
|
"name": "anydataPpm",
|
||||||
"label": "Вольты",
|
"label": "PPM",
|
||||||
"widget": "anydata",
|
"widget": "anydata",
|
||||||
"after": "Vlt",
|
"after": "ppm",
|
||||||
"icon": "speedometer"
|
"icon": "speedometer"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
//Версия прошивки
|
//Версия прошивки
|
||||||
#define FIRMWARE_VERSION 426
|
#define FIRMWARE_VERSION 427
|
||||||
|
|
||||||
#ifdef esp8266_4mb
|
#ifdef esp8266_4mb
|
||||||
#define FIRMWARE_NAME "esp8266_4mb"
|
#define FIRMWARE_NAME "esp8266_4mb"
|
||||||
|
|||||||
@@ -10,5 +10,6 @@ extern void handleFileUpload();
|
|||||||
extern void handleFileDelete();
|
extern void handleFileDelete();
|
||||||
extern void handleFileCreate();
|
extern void handleFileCreate();
|
||||||
extern void handleFileList();
|
extern void handleFileList();
|
||||||
|
void printDirectory(File dir, String& out);
|
||||||
//#endif
|
//#endif
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <Arduino.h>
|
#include <Arduino.h>
|
||||||
|
|
||||||
#include <SoftwareSerial.h>
|
#include <SoftwareSerial.h>
|
||||||
|
|
||||||
#define PZEM_DEFAULT_ADDR 0xF8
|
#define PZEM_DEFAULT_ADDR 0xF8
|
||||||
@@ -22,13 +21,13 @@ class PZEMSensor {
|
|||||||
PZEMSensor(Stream *serial, uint16_t addr = PZEM_DEFAULT_ADDR);
|
PZEMSensor(Stream *serial, uint16_t addr = PZEM_DEFAULT_ADDR);
|
||||||
|
|
||||||
~PZEMSensor();
|
~PZEMSensor();
|
||||||
PZEM_Info* values();
|
PZEM_Info *values(bool &online);
|
||||||
bool setAddress(uint8_t addr);
|
bool setAddress(uint8_t addr);
|
||||||
uint8_t getAddress();
|
uint8_t getAddress();
|
||||||
bool setPowerAlarm(uint16_t watts);
|
bool setPowerAlarm(uint16_t watts);
|
||||||
bool getPowerAlarm();
|
bool getPowerAlarm();
|
||||||
bool reset();
|
bool reset();
|
||||||
void search();
|
bool search();
|
||||||
// Get most up to date values from device registers and cache them
|
// Get most up to date values from device registers and cache them
|
||||||
bool refresh();
|
bool refresh();
|
||||||
|
|
||||||
|
|||||||
@@ -43,10 +43,13 @@ PZEMSensor::PZEMSensor(Stream *port, uint16_t addr) {
|
|||||||
init();
|
init();
|
||||||
}
|
}
|
||||||
|
|
||||||
PZEM_Info *PZEMSensor::values() {
|
PZEM_Info *PZEMSensor::values(bool &online) {
|
||||||
// Update vales if necessary
|
// Update vales if necessary
|
||||||
if (!refresh()) {
|
if (!refresh()) {
|
||||||
_values = PZEM_Info();
|
_values = PZEM_Info();
|
||||||
|
online = false;
|
||||||
|
} else {
|
||||||
|
online = true;
|
||||||
}
|
}
|
||||||
return &_values;
|
return &_values;
|
||||||
}
|
}
|
||||||
@@ -295,7 +298,8 @@ uint16_t PZEMSensor::CRC16(const uint8_t *data, uint16_t len) {
|
|||||||
return crc;
|
return crc;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PZEMSensor::search() {
|
bool PZEMSensor::search() {
|
||||||
|
bool ret = false;
|
||||||
static uint8_t response[7];
|
static uint8_t response[7];
|
||||||
for (uint16_t addr = 0x01; addr <= 0xF8; addr++) {
|
for (uint16_t addr = 0x01; addr <= 0xF8; addr++) {
|
||||||
sendCmd8(CMD_RIR, 0x00, 0x01, false, addr);
|
sendCmd8(CMD_RIR, 0x00, 0x01, false, addr);
|
||||||
@@ -303,8 +307,9 @@ void PZEMSensor::search() {
|
|||||||
// Something went wrong
|
// Something went wrong
|
||||||
continue;
|
continue;
|
||||||
} else {
|
} else {
|
||||||
Serial.print("Device on addr: ");
|
Serial.println("Pzem " + String(addr));
|
||||||
Serial.print(addr);
|
ret = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,7 +28,11 @@
|
|||||||
"modules": {
|
"modules": {
|
||||||
"Виртуальные элементы": [
|
"Виртуальные элементы": [
|
||||||
{
|
{
|
||||||
"path": "src\\modules\\virtual\\Logging",
|
"path": "src\\modules\\virtual\\Loging",
|
||||||
|
"active": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "src\\modules\\virtual\\LogingDaily",
|
||||||
"active": true
|
"active": true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -45,6 +49,10 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"Сенсоры": [
|
"Сенсоры": [
|
||||||
|
{
|
||||||
|
"path": "src\\modules\\sensors\\Acs712",
|
||||||
|
"active": true
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"path": "src\\modules\\sensors\\Ads1115",
|
"path": "src\\modules\\sensors\\Ads1115",
|
||||||
"active": false
|
"active": false
|
||||||
|
|||||||
@@ -61,6 +61,7 @@ lib_deps =
|
|||||||
https://github.com/JonasGMorsch/GY-21.git
|
https://github.com/JonasGMorsch/GY-21.git
|
||||||
ClosedCube HDC1080
|
ClosedCube HDC1080
|
||||||
adafruit/MAX6675 library
|
adafruit/MAX6675 library
|
||||||
|
mandulaj/PZEM-004T-v30
|
||||||
rc-switch @ ^2.6.4
|
rc-switch @ ^2.6.4
|
||||||
robtillaart/SHT2x@^0.1.1
|
robtillaart/SHT2x@^0.1.1
|
||||||
WEMOS SHT3x@1.0.0
|
WEMOS SHT3x@1.0.0
|
||||||
@@ -70,10 +71,12 @@ lib_deps =
|
|||||||
dfrobot/DFRobotDFPlayerMini @ ^1.0.5
|
dfrobot/DFRobotDFPlayerMini @ ^1.0.5
|
||||||
marcoschwartz/LiquidCrystal_I2C@^1.1.4
|
marcoschwartz/LiquidCrystal_I2C@^1.1.4
|
||||||
build_src_filter =
|
build_src_filter =
|
||||||
+<modules\virtual\Logging>
|
+<modules\virtual\Loging>
|
||||||
|
+<modules\virtual\LogingDaily>
|
||||||
+<modules\virtual\Timer>
|
+<modules\virtual\Timer>
|
||||||
+<modules\virtual\Variable>
|
+<modules\virtual\Variable>
|
||||||
+<modules\virtual\VButton>
|
+<modules\virtual\VButton>
|
||||||
|
+<modules\sensors\Acs712>
|
||||||
+<modules\sensors\Aht20>
|
+<modules\sensors\Aht20>
|
||||||
+<modules\sensors\AnalogAdc>
|
+<modules\sensors\AnalogAdc>
|
||||||
+<modules\sensors\Bme280>
|
+<modules\sensors\Bme280>
|
||||||
|
|||||||
@@ -80,6 +80,12 @@ void setup() {
|
|||||||
|
|
||||||
// test
|
// test
|
||||||
Serial.println("-------test start--------");
|
Serial.println("-------test start--------");
|
||||||
|
|
||||||
|
// File dir = FileFS.open("/", "r");
|
||||||
|
// String out;
|
||||||
|
// printDirectory(dir, out);
|
||||||
|
// Serial.println(out);
|
||||||
|
|
||||||
//=======проверка очереди из структур=================
|
//=======проверка очереди из структур=================
|
||||||
|
|
||||||
// myDB = new IoTDB;
|
// myDB = new IoTDB;
|
||||||
|
|||||||
@@ -134,9 +134,10 @@ void mqttCallback(char* topic, uint8_t* payload, size_t length) {
|
|||||||
publishWidgets();
|
publishWidgets();
|
||||||
publishState();
|
publishState();
|
||||||
|
|
||||||
|
//обращение к логированию из ядра
|
||||||
//отправка данных графиков
|
//отправка данных графиков
|
||||||
for (std::list<IoTItem*>::iterator it = IoTItems.begin(); it != IoTItems.end(); ++it) {
|
for (std::list<IoTItem*>::iterator it = IoTItems.begin(); it != IoTItems.end(); ++it) {
|
||||||
if ((*it)->getSubtype() == "Loging") {
|
if ((*it)->getSubtype() == "Loging" || "LogingDaily") {
|
||||||
(*it)->setPublishDestination(TO_MQTT);
|
(*it)->setPublishDestination(TO_MQTT);
|
||||||
(*it)->publishValue();
|
(*it)->publishValue();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,42 +11,36 @@ void standWebServerInit() {
|
|||||||
HTTP.serveStatic("/bundle.css.gz", FileFS, "/", "max-age=31536000"); // кеширование на 1 год
|
HTTP.serveStatic("/bundle.css.gz", FileFS, "/", "max-age=31536000"); // кеширование на 1 год
|
||||||
HTTP.serveStatic("/favicon.png", FileFS, "/", "max-age=31536000"); // кеширование на 1 год
|
HTTP.serveStatic("/favicon.png", FileFS, "/", "max-age=31536000"); // кеширование на 1 год
|
||||||
|
|
||||||
HTTP.on("/devicelist.json", HTTP_GET, []() {
|
// HTTP.on("/devicelist.json", HTTP_GET, []() {
|
||||||
HTTP.send(200, "application/json", devListHeapJson);
|
// HTTP.send(200, "application/json", devListHeapJson);
|
||||||
});
|
// });
|
||||||
|
// HTTP.on("/settings.h.json", HTTP_GET, []() {
|
||||||
HTTP.on("/settings.h.json", HTTP_GET, []() {
|
// HTTP.send(200, "application/json", settingsFlashJson);
|
||||||
HTTP.send(200, "application/json", settingsFlashJson);
|
// });
|
||||||
});
|
// HTTP.on("/settings.f.json", HTTP_GET, []() {
|
||||||
|
// HTTP.send(200, "application/json", readFile(F("settings.json"), 20000));
|
||||||
HTTP.on("/settings.f.json", HTTP_GET, []() {
|
// });
|
||||||
HTTP.send(200, "application/json", readFile(F("settings.json"), 20000));
|
// HTTP.on("/params.json", HTTP_GET, []() {
|
||||||
});
|
// String json = getParamsJson();
|
||||||
|
// HTTP.send(200, "application/json", json);
|
||||||
HTTP.on("/params.json", HTTP_GET, []() {
|
// });
|
||||||
String json = getParamsJson();
|
// HTTP.on("/errors.json", HTTP_GET, []() {
|
||||||
HTTP.send(200, "application/json", json);
|
// HTTP.send(200, "application/json", errorsHeapJson);
|
||||||
});
|
// });
|
||||||
|
// HTTP.on("/config.json", HTTP_GET, []() {
|
||||||
HTTP.on("/errors.json", HTTP_GET, []() {
|
// HTTP.send(200, "application/json", readFile(F("config.json"), 20000));
|
||||||
HTTP.send(200, "application/json", errorsHeapJson);
|
// });
|
||||||
});
|
// HTTP.on("/layout.json", HTTP_GET, []() {
|
||||||
|
// HTTP.send(200, "application/json", readFile(F("layout.json"), 20000));
|
||||||
HTTP.on("/config.json", HTTP_GET, []() {
|
// });
|
||||||
HTTP.send(200, "application/json", readFile(F("config.json"), 20000));
|
// HTTP.on("/restart", HTTP_GET, []() {
|
||||||
});
|
// // ESP.restart();
|
||||||
|
// HTTP.send(200, "text/plain", "ok");
|
||||||
HTTP.on("/layout.json", HTTP_GET, []() {
|
// });
|
||||||
HTTP.send(200, "application/json", readFile(F("layout.json"), 20000));
|
|
||||||
});
|
|
||||||
|
|
||||||
HTTP.on("/restart", HTTP_GET, []() {
|
|
||||||
// ESP.restart();
|
|
||||||
HTTP.send(200, "text/plain", "ok");
|
|
||||||
});
|
|
||||||
|
|
||||||
// Добавляем функцию Update для перезаписи прошивки по WiFi при 1М(256K FileFS) и выше
|
// Добавляем функцию Update для перезаписи прошивки по WiFi при 1М(256K FileFS) и выше
|
||||||
// httpUpdater.setup(&HTTP);
|
// httpUpdater.setup(&HTTP);
|
||||||
|
|
||||||
// Запускаем HTTP сервер
|
// Запускаем HTTP сервер
|
||||||
HTTP.begin();
|
HTTP.begin();
|
||||||
|
|
||||||
@@ -95,7 +89,6 @@ void standWebServerInit() {
|
|||||||
HTTP.send(200, "text/plain", "");
|
HTTP.send(200, "text/plain", "");
|
||||||
},
|
},
|
||||||
handleFileUpload);
|
handleFileUpload);
|
||||||
//#endif
|
|
||||||
|
|
||||||
// called when the url is not defined here
|
// called when the url is not defined here
|
||||||
HTTP.onNotFound([]() {
|
HTTP.onNotFound([]() {
|
||||||
@@ -155,7 +148,6 @@ String getContentType(String filename) {
|
|||||||
return "text/plain";
|
return "text/plain";
|
||||||
}
|
}
|
||||||
|
|
||||||
//#ifdef REST_FILE_OPERATIONS
|
|
||||||
// Здесь функции для работы с файловой системой
|
// Здесь функции для работы с файловой системой
|
||||||
void handleFileUpload() {
|
void handleFileUpload() {
|
||||||
if (HTTP.uri() != "/edit") return;
|
if (HTTP.uri() != "/edit") return;
|
||||||
@@ -205,10 +197,6 @@ void handleFileCreate() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void handleFileList() {
|
void handleFileList() {
|
||||||
if (!HTTP.hasArg("list")) {
|
|
||||||
HTTP.send(500, "text/plain", "BAD ARGS");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
File dir = FileFS.open("/", "r");
|
File dir = FileFS.open("/", "r");
|
||||||
String output = "[";
|
String output = "[";
|
||||||
File entry;
|
File entry;
|
||||||
@@ -226,5 +214,22 @@ void handleFileList() {
|
|||||||
Serial.println(output);
|
Serial.println(output);
|
||||||
HTTP.send(200, "text/json", output);
|
HTTP.send(200, "text/json", output);
|
||||||
}
|
}
|
||||||
//#endif
|
|
||||||
|
void printDirectory(File dir, String& out) {
|
||||||
|
while (true) {
|
||||||
|
File entry = dir.openNextFile();
|
||||||
|
if (!entry) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (entry.isDirectory()) {
|
||||||
|
out += entry.name();
|
||||||
|
out += "/";
|
||||||
|
printDirectory(entry, out);
|
||||||
|
} else {
|
||||||
|
out += entry.name();
|
||||||
|
out += "\r\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -73,13 +73,14 @@ void webSocketEvent(uint8_t num, WStype_t type, uint8_t* payload, size_t length)
|
|||||||
|
|
||||||
//отвечаем на запрос графиков
|
//отвечаем на запрос графиков
|
||||||
if (headerStr == "/charts|") {
|
if (headerStr == "/charts|") {
|
||||||
|
//обращение к логированию из ядра
|
||||||
//отправка данных графиков только в выбранный сокет
|
//отправка данных графиков только в выбранный сокет
|
||||||
for (std::list<IoTItem*>::iterator it = IoTItems.begin(); it != IoTItems.end(); ++it) {
|
for (std::list<IoTItem*>::iterator it = IoTItems.begin(); it != IoTItems.end(); ++it) {
|
||||||
//сбрасываем даты графиков
|
//сбрасываем даты графиков
|
||||||
// if ((*it)->getID().endsWith("-date")) {
|
// if ((*it)->getID().endsWith("-date")) {
|
||||||
// (*it)->setTodayDate();
|
// (*it)->setTodayDate();
|
||||||
//}
|
//}
|
||||||
if ((*it)->getSubtype() == "Loging") {
|
if ((*it)->getSubtype() == "Loging" || "LogingDaily") {
|
||||||
(*it)->setPublishDestination(TO_WS, num);
|
(*it)->setPublishDestination(TO_WS, num);
|
||||||
(*it)->publishValue();
|
(*it)->publishValue();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -96,7 +96,7 @@ void IoTItem::regEvent(String value, String consoleInfo = "") {
|
|||||||
publishStatusMqtt(_id, value);
|
publishStatusMqtt(_id, value);
|
||||||
|
|
||||||
publishStatusWs(_id, value);
|
publishStatusWs(_id, value);
|
||||||
SerialPrint("i", "Sensor " + consoleInfo, "'" + _id + "' data: " + value + "'");
|
SerialPrint("i", "Sensor", consoleInfo + " '" + _id + "' data: " + value + "'");
|
||||||
|
|
||||||
// проверка если global установлен то шлем всем о событии
|
// проверка если global установлен то шлем всем о событии
|
||||||
// if (_global) {
|
// if (_global) {
|
||||||
@@ -150,8 +150,6 @@ bool IoTItem::isGpioDriver() {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//сетевое общение====================================================================================================================================
|
//сетевое общение====================================================================================================================================
|
||||||
|
|
||||||
externalVariable::externalVariable(String parameters) : IoTItem(parameters) {
|
externalVariable::externalVariable(String parameters) : IoTItem(parameters) {
|
||||||
|
|||||||
@@ -205,11 +205,11 @@ class BinaryExprAST : public ExprAST {
|
|||||||
|
|
||||||
if (!lhs->isDecimal || !rhs->isDecimal) {
|
if (!lhs->isDecimal || !rhs->isDecimal) {
|
||||||
if (lhs->isDecimal)
|
if (lhs->isDecimal)
|
||||||
lhsStr = lhs->valD;
|
lhsStr = (String)lhs->valD;
|
||||||
else
|
else
|
||||||
lhsStr = lhs->valS;
|
lhsStr = lhs->valS;
|
||||||
if (rhs->isDecimal)
|
if (rhs->isDecimal)
|
||||||
rhsStr = rhs->valD;
|
rhsStr = (String)rhs->valD;
|
||||||
else
|
else
|
||||||
rhsStr = rhs->valS;
|
rhsStr = rhs->valS;
|
||||||
switch (Op) {
|
switch (Op) {
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
#include "ESPConfiguration.h"
|
#include "ESPConfiguration.h"
|
||||||
|
|
||||||
void* getAPI_Loging(String subtype, String params);
|
void* getAPI_Loging(String subtype, String params);
|
||||||
|
void* getAPI_LogingDaily(String subtype, String params);
|
||||||
void* getAPI_Timer(String subtype, String params);
|
void* getAPI_Timer(String subtype, String params);
|
||||||
void* getAPI_Variable(String subtype, String params);
|
void* getAPI_Variable(String subtype, String params);
|
||||||
void* getAPI_VButton(String subtype, String params);
|
void* getAPI_VButton(String subtype, String params);
|
||||||
|
void* getAPI_Acs712(String subtype, String params);
|
||||||
void* getAPI_Aht20(String subtype, String params);
|
void* getAPI_Aht20(String subtype, String params);
|
||||||
void* getAPI_AnalogAdc(String subtype, String params);
|
void* getAPI_AnalogAdc(String subtype, String params);
|
||||||
void* getAPI_Bme280(String subtype, String params);
|
void* getAPI_Bme280(String subtype, String params);
|
||||||
@@ -31,9 +33,11 @@ void* getAPI_Lcd2004(String subtype, String params);
|
|||||||
void* getAPI(String subtype, String params) {
|
void* getAPI(String subtype, String params) {
|
||||||
void* tmpAPI;
|
void* tmpAPI;
|
||||||
if ((tmpAPI = getAPI_Loging(subtype, params)) != nullptr) return tmpAPI;
|
if ((tmpAPI = getAPI_Loging(subtype, params)) != nullptr) return tmpAPI;
|
||||||
|
if ((tmpAPI = getAPI_LogingDaily(subtype, params)) != nullptr) return tmpAPI;
|
||||||
if ((tmpAPI = getAPI_Timer(subtype, params)) != nullptr) return tmpAPI;
|
if ((tmpAPI = getAPI_Timer(subtype, params)) != nullptr) return tmpAPI;
|
||||||
if ((tmpAPI = getAPI_Variable(subtype, params)) != nullptr) return tmpAPI;
|
if ((tmpAPI = getAPI_Variable(subtype, params)) != nullptr) return tmpAPI;
|
||||||
if ((tmpAPI = getAPI_VButton(subtype, params)) != nullptr) return tmpAPI;
|
if ((tmpAPI = getAPI_VButton(subtype, params)) != nullptr) return tmpAPI;
|
||||||
|
if ((tmpAPI = getAPI_Acs712(subtype, params)) != nullptr) return tmpAPI;
|
||||||
if ((tmpAPI = getAPI_Aht20(subtype, params)) != nullptr) return tmpAPI;
|
if ((tmpAPI = getAPI_Aht20(subtype, params)) != nullptr) return tmpAPI;
|
||||||
if ((tmpAPI = getAPI_AnalogAdc(subtype, params)) != nullptr) return tmpAPI;
|
if ((tmpAPI = getAPI_AnalogAdc(subtype, params)) != nullptr) return tmpAPI;
|
||||||
if ((tmpAPI = getAPI_Bme280(subtype, params)) != nullptr) return tmpAPI;
|
if ((tmpAPI = getAPI_Bme280(subtype, params)) != nullptr) return tmpAPI;
|
||||||
|
|||||||
76
src/modules/sensors/Acs712/Acs712.cpp
Normal file
76
src/modules/sensors/Acs712/Acs712.cpp
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
#include "Global.h"
|
||||||
|
#include "classes/IoTItem.h"
|
||||||
|
|
||||||
|
extern IoTGpio IoTgpio;
|
||||||
|
|
||||||
|
class Acs712 : public IoTItem
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
unsigned int _pin;
|
||||||
|
const unsigned long _sampleTime = 100000UL; // sample over 100ms, it is an exact number of cycles for both 50Hz and 60Hz mains
|
||||||
|
const unsigned long _numSamples = 250UL; // choose the number of samples to divide sampleTime exactly, but low enough for the ADC to keep up
|
||||||
|
const unsigned long _sampleInterval = _sampleTime / _numSamples; // the sampling interval, must be longer than then ADC conversion time
|
||||||
|
int _adc_zero1; //Переменная автоматической калибровки
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
Acs712(String parameters) : IoTItem(parameters)
|
||||||
|
{
|
||||||
|
String tmp;
|
||||||
|
jsonRead(parameters, "pin", tmp);
|
||||||
|
_pin = tmp.toInt();
|
||||||
|
_adc_zero1 = determineVQ(_pin);
|
||||||
|
}
|
||||||
|
|
||||||
|
void doByInterval()
|
||||||
|
{
|
||||||
|
|
||||||
|
unsigned long currentAcc = 0;
|
||||||
|
unsigned int count = 0;
|
||||||
|
unsigned long prevMicros = micros() - _sampleInterval;
|
||||||
|
while (count < _numSamples)
|
||||||
|
{
|
||||||
|
if (micros() - prevMicros >= _sampleInterval)
|
||||||
|
{
|
||||||
|
int adc_raw = IoTgpio.analogRead(_pin) - _adc_zero1;
|
||||||
|
currentAcc += (unsigned long)(adc_raw * adc_raw);
|
||||||
|
++count;
|
||||||
|
prevMicros += _sampleInterval;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#ifdef ESP32
|
||||||
|
value.valD = int(sqrt((float)currentAcc / (float)_numSamples) * (75.7576 / 4095.0));
|
||||||
|
#else
|
||||||
|
value.valD = int(sqrt((float)currentAcc / (float)_numSamples) * (75.7576 / 1023.0));
|
||||||
|
#endif
|
||||||
|
regEvent(value.valD, "Acs712");
|
||||||
|
}
|
||||||
|
|
||||||
|
int determineVQ(int PIN)
|
||||||
|
{
|
||||||
|
long VQ = 0;
|
||||||
|
// read 5000 samples to stabilise value
|
||||||
|
for (int i = 0; i < 5000; i++)
|
||||||
|
{
|
||||||
|
VQ += IoTgpio.analogRead(PIN);
|
||||||
|
//delay(1); // depends on sampling (on filter capacitor), can be 1/80000 (80kHz) max.
|
||||||
|
}
|
||||||
|
VQ /= 5000;
|
||||||
|
return int(VQ);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
~Acs712(){};
|
||||||
|
};
|
||||||
|
|
||||||
|
void *getAPI_Acs712(String subtype, String param)
|
||||||
|
{
|
||||||
|
if (subtype == F("Acs712"))
|
||||||
|
{
|
||||||
|
return new Acs712(param);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
36
src/modules/sensors/Acs712/modinfo.json
Normal file
36
src/modules/sensors/Acs712/modinfo.json
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
{
|
||||||
|
"menuSection": "Сенсоры",
|
||||||
|
"configItem": [
|
||||||
|
{
|
||||||
|
"name": "Acs712 Ток",
|
||||||
|
"type": "Reading",
|
||||||
|
"subtype": "Acs712",
|
||||||
|
"id": "amp",
|
||||||
|
"widget": "anydataAmp",
|
||||||
|
"page": "Сенсоры",
|
||||||
|
"descr": "Ток",
|
||||||
|
"round": 3,
|
||||||
|
"pin": 39,
|
||||||
|
"int": 5
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"about": {
|
||||||
|
"authorName": "Yuriy Kuneev",
|
||||||
|
"authorContact": "https://t.me/Kuneev07",
|
||||||
|
"authorGit": "",
|
||||||
|
"exampleURL": "https://iotmanager.org/wiki",
|
||||||
|
"specialThanks": "",
|
||||||
|
"moduleName": "Acs712",
|
||||||
|
"moduleVersion": "1.0",
|
||||||
|
"moduleDesc": "Позволяет получить текущее значение тока на аналоговом пине с помощью модуля Acs712.",
|
||||||
|
"propInfo": {
|
||||||
|
"pin": "Аналоговый GPIO номер, к которому подключен датчик.",
|
||||||
|
"int": "Количество секунд между опросами датчика."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"defActive": true,
|
||||||
|
"devices": {
|
||||||
|
"esp32_4mb": [],
|
||||||
|
"esp8266_4mb": []
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -11,15 +11,25 @@
|
|||||||
|
|
||||||
extern IoTGpio IoTgpio;
|
extern IoTGpio IoTgpio;
|
||||||
|
|
||||||
|
// SoftwareSerial* swSerialCO2 = nullptr;
|
||||||
int rxPinCO2; // зеленый провод сенсора к прописаному по умолчанию D7 (GPIO13)
|
int rxPinCO2; // зеленый провод сенсора к прописаному по умолчанию D7 (GPIO13)
|
||||||
int txPinCO2; // синий провод сенсора к прописаному по умолчанию D6 (GPIO12)
|
int txPinCO2; // синий провод сенсора к прописаному по умолчанию D6 (GPIO12)
|
||||||
|
#ifdef ESP8266
|
||||||
|
|
||||||
SoftwareSerial *swSerialCO2 = nullptr;
|
SoftwareSerial *swSerialCO2 = nullptr;
|
||||||
|
#endif
|
||||||
|
#ifdef ESP32
|
||||||
|
#include <HardwareSerial.h>
|
||||||
|
HardwareSerial swSerialCO2(1);
|
||||||
|
#endif
|
||||||
|
|
||||||
int MHZ19_request(int request);
|
int MHZ19_request(int request);
|
||||||
|
|
||||||
void MHZ19uart_init();
|
void MHZ19uart_init();
|
||||||
|
|
||||||
bool MHZ19uart_flag = true;
|
bool MHZ19uartInit_flag = true;
|
||||||
int MHZ19C_PREHEATING_TIME = 2 * 30 * 1000; // покажет реальные данные после прогрева, через 2 мин.
|
bool MHZ19uartUpdateInputs_flag = true;
|
||||||
|
unsigned int preheating = 2 * 60; // покажет реальные данные после прогрева, через 2 мин.
|
||||||
|
|
||||||
int temperature = 0;
|
int temperature = 0;
|
||||||
bool temperatureUpdated = false;
|
bool temperatureUpdated = false;
|
||||||
@@ -32,7 +42,8 @@ bool ABCchanged = false;
|
|||||||
|
|
||||||
//Это файл сенсора, в нем осуществляется чтение сенсора.
|
//Это файл сенсора, в нем осуществляется чтение сенсора.
|
||||||
|
|
||||||
class Mhz19uart : public IoTItem {
|
class Mhz19uart : public IoTItem
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// Секция переменных.
|
// Секция переменных.
|
||||||
@@ -40,24 +51,43 @@ class Mhz19uart : public IoTItem {
|
|||||||
public:
|
public:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
|
|
||||||
Mhz19uart(String parameters) : IoTItem(parameters) {
|
Mhz19uart(String parameters) : IoTItem(parameters)
|
||||||
|
{
|
||||||
rxPinCO2 = jsonReadInt(parameters, "rxPin");
|
rxPinCO2 = jsonReadInt(parameters, "rxPin");
|
||||||
txPinCO2 = jsonReadInt(parameters, "txPin");
|
txPinCO2 = jsonReadInt(parameters, "txPin");
|
||||||
|
preheating = jsonReadInt(parameters, "warmUp");
|
||||||
range = jsonReadInt(parameters, "range");
|
range = jsonReadInt(parameters, "range");
|
||||||
ABC = jsonReadInt(parameters, "ABC");
|
ABC = jsonReadInt(parameters, "ABC");
|
||||||
if (!swSerialCO2) swSerialCO2 = new SoftwareSerial(rxPinCO2, txPinCO2);
|
MHZ19uartInit_flag = true;
|
||||||
|
MHZ19uartUpdateInputs_flag = true;
|
||||||
|
#ifdef ESP8266
|
||||||
|
if (!swSerialCO2)
|
||||||
|
swSerialCO2 = new SoftwareSerial(rxPinCO2, txPinCO2);
|
||||||
|
#endif
|
||||||
|
#ifdef ESP32
|
||||||
|
// HardwareSerial swSerialCO2(1);
|
||||||
|
// HardwareSerial swSerialCO2(2);
|
||||||
|
// swSerialCO2.begin(9600);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// doByInterval()
|
// doByInterval()
|
||||||
|
|
||||||
void doByInterval() {
|
void doByInterval()
|
||||||
|
{
|
||||||
MHZ19uart_init();
|
MHZ19uart_init();
|
||||||
if (millis() > MHZ19C_PREHEATING_TIME) {
|
|
||||||
Serial.println("Start checkUARTCO2");
|
if (millis() > preheating * 1000)
|
||||||
value.valD = MHZ19_request(1);
|
{
|
||||||
|
// Serial.println("Start checkUARTCO2");
|
||||||
|
int reply = MHZ19_request(1);
|
||||||
|
if (reply)
|
||||||
|
{
|
||||||
|
value.valD = reply;
|
||||||
|
regEvent(value.valD, "Mhz19uart");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
regEvent(value.valD, "Mhz19uart"); //обязательный вызов хотяб один
|
|
||||||
}
|
}
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
|
|
||||||
@@ -69,63 +99,110 @@ class Mhz19uart : public IoTItem {
|
|||||||
//создание и контроль соответствующих глобальных переменных
|
//создание и контроль соответствующих глобальных переменных
|
||||||
|
|
||||||
//замер по ШИМ создает задержку. вызываем его нечасто, по умолчанию раз в 5 минут
|
//замер по ШИМ создает задержку. вызываем его нечасто, по умолчанию раз в 5 минут
|
||||||
class Mhz19pwm : public IoTItem {
|
class Mhz19pwm : public IoTItem
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// Секция переменных.
|
// Секция переменных.
|
||||||
int pwmPin; // желтый провод сенсора к прописаному по умолчанию D8 (GPIO15)
|
int pwmPin = 15; // желтый провод сенсора к прописаному по умолчанию D8 (GPIO15)
|
||||||
|
int maxRetriesNotAvailable = 10;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
|
|
||||||
Mhz19pwm(String parameters) : IoTItem(parameters) {
|
Mhz19pwm(String parameters) : IoTItem(parameters)
|
||||||
pwmPin = jsonReadInt(parameters, "pin");
|
|
||||||
}
|
|
||||||
//=======================================================================================================
|
|
||||||
|
|
||||||
void doByInterval() {
|
|
||||||
MHZ19pwm_init();
|
|
||||||
if (millis() > MHZ19C_PREHEATING_TIME) //
|
|
||||||
{
|
{
|
||||||
Serial.println("Start checkPWM_CO2");
|
pwmPin = jsonReadInt(parameters, "pin");
|
||||||
value.valD = MHZ19pwm_request();
|
preheating = jsonReadInt(parameters, "warmUp");
|
||||||
}
|
maxRetriesNotAvailable = jsonReadInt(parameters, "maxRetriesNotAvailable");
|
||||||
regEvent(value.valD, "Mhz19pwm"); //обязательный вызов хотяб один
|
|
||||||
}
|
}
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
|
|
||||||
void MHZ19pwm_init() {
|
void doByInterval()
|
||||||
|
{
|
||||||
|
MHZ19pwm_init();
|
||||||
|
|
||||||
|
if (millis() > preheating * 1000) //
|
||||||
|
{
|
||||||
|
// Serial.println("Start checkPWM_CO2");
|
||||||
|
int reply = MHZ19pwm_request();
|
||||||
|
if (reply)
|
||||||
|
{
|
||||||
|
SerialPrint("E", "Sensor Mhz19pwm", "MHZ19pwm_init reply = " + String(reply));
|
||||||
|
value.valD = reply;
|
||||||
|
regEvent(value.valD, "Mhz19pwm");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//=======================================================================================================
|
||||||
|
|
||||||
|
void MHZ19pwm_init()
|
||||||
|
{
|
||||||
static bool MHZ19pwm_flag = true;
|
static bool MHZ19pwm_flag = true;
|
||||||
if (MHZ19pwm_flag) {
|
if (MHZ19pwm_flag)
|
||||||
|
{
|
||||||
pinMode(pwmPin, INPUT);
|
pinMode(pwmPin, INPUT);
|
||||||
MHZ19pwm_flag = false;
|
MHZ19pwm_flag = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int MHZ19pwm_request() {
|
int MHZ19pwm_request()
|
||||||
|
{
|
||||||
int reply;
|
int reply;
|
||||||
Serial.println("Запрос замера по PWM запущен");
|
// Serial.println("Запрос замера по PWM запущен");
|
||||||
unsigned long th, tl, ppm = 0, ppm2 = 0, ppm3 = 0;
|
unsigned long th = 0, tl, ppm = 0, ppm2 = 0, ppm3 = 0;
|
||||||
do {
|
// int pwmPinReading1 = digitalRead(pwmPin);
|
||||||
|
// SerialPrint("E", "Sensor Mhz19uart", "pwmPinReading1 = " + String(pwmPinReading1));
|
||||||
|
/*
|
||||||
|
do
|
||||||
|
{
|
||||||
th = pulseIn(pwmPin, HIGH, 1004000) / 1000;
|
th = pulseIn(pwmPin, HIGH, 1004000) / 1000;
|
||||||
tl = 1004 - th;
|
tl = 1004 - th;
|
||||||
ppm2 = 2000 * (th - 2) / (th + tl - 4); // расчёт для диапазона от 0 до 2000ppm
|
ppm2 = 2000 * (th - 2) / (th + tl - 4); // расчёт для диапазона от 0 до 2000ppm
|
||||||
ppm3 = 5000 * (th - 2) / (th + tl - 4); // расчёт для диапазона от 0 до 5000ppm
|
ppm3 = 5000 * (th - 2) / (th + tl - 4); // расчёт для диапазона от 0 до 5000ppm
|
||||||
} while (th == 0);
|
} while (th == 0);
|
||||||
|
*/
|
||||||
|
for (int i = 0; i < maxRetriesNotAvailable; i++)
|
||||||
|
{
|
||||||
|
th = pulseIn(pwmPin, HIGH, 1004000) / 1000;
|
||||||
|
|
||||||
|
if (th != 0)
|
||||||
|
{
|
||||||
|
tl = 1004 - th;
|
||||||
|
ppm2 = 2000 * (th - 2) / (th + tl - 4); // расчёт для диапазона от 0 до 2000ppm
|
||||||
|
ppm3 = 5000 * (th - 2) / (th + tl - 4); // расчёт для диапазона от 0 до 5000ppm
|
||||||
|
if (i > 1)
|
||||||
|
SerialPrint("E", "Sensor Mhz19pwm", "Got reading from PWM pin after " + String(i) + " tries");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// Serial.println(i);
|
||||||
|
}
|
||||||
|
if (!th)
|
||||||
|
{
|
||||||
|
SerialPrint("E", "Sensor Mhz19pwm", "No reading from PWM pin. Check wiring.");
|
||||||
|
reply = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
|
||||||
// Serial.print(th);
|
// Serial.print(th);
|
||||||
// Serial.println(" <- Milliseconds PWM is HIGH");
|
// Serial.println(" <- Milliseconds PWM is HIGH");
|
||||||
|
if (range == 2000)
|
||||||
if (range == 2000) {
|
{
|
||||||
reply = ppm2;
|
reply = ppm2;
|
||||||
Serial.print(ppm2);
|
// Serial.print(ppm2);
|
||||||
Serial.println(" <- ppm2 (PWM) with 2000ppm as limit");
|
// Serial.println(" <- ppm2 (PWM) with 2000ppm as limit");
|
||||||
} else {
|
|
||||||
reply = ppm3;
|
|
||||||
Serial.print(ppm3);
|
|
||||||
Serial.println(" <- ppm3 (PWM) with 5000ppm as limit");
|
|
||||||
}
|
}
|
||||||
Serial.println("Completed checkPwmCO2");
|
else
|
||||||
|
{
|
||||||
|
reply = ppm3;
|
||||||
|
// Serial.print(ppm3);
|
||||||
|
// Serial.println(" <- ppm3 (PWM) with 5000ppm as limit");
|
||||||
|
}
|
||||||
|
// Serial.println("Completed checkPwmCO2");
|
||||||
|
}
|
||||||
|
// int pwmPinReading2 = digitalRead(pwmPin);
|
||||||
|
// SerialPrint("E", "Sensor Mhz19uart", "pwmPinReading2 = " + String(pwmPinReading2));
|
||||||
return reply;
|
return reply;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -134,35 +211,51 @@ class Mhz19pwm : public IoTItem {
|
|||||||
|
|
||||||
//====================TEMP===================================================================================
|
//====================TEMP===================================================================================
|
||||||
|
|
||||||
class Mhz19temp : public IoTItem {
|
class Mhz19temp : public IoTItem
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// Секция переменных.
|
// Секция переменных.
|
||||||
public:
|
public:
|
||||||
//====================TEMP===================================================================================
|
//====================TEMP===================================================================================
|
||||||
|
|
||||||
Mhz19temp(String parameters) : IoTItem(parameters) {
|
Mhz19temp(String parameters) : IoTItem(parameters)
|
||||||
|
{
|
||||||
rxPinCO2 = jsonReadInt(parameters, "rxPin");
|
rxPinCO2 = jsonReadInt(parameters, "rxPin");
|
||||||
txPinCO2 = jsonReadInt(parameters, "txPin");
|
txPinCO2 = jsonReadInt(parameters, "txPin");
|
||||||
range = jsonReadInt(parameters, "range");
|
|
||||||
ABC = jsonReadInt(parameters, "ABC");
|
#ifdef ESP8266
|
||||||
if (!swSerialCO2) swSerialCO2 = new SoftwareSerial(rxPinCO2, txPinCO2);
|
if (!swSerialCO2)
|
||||||
|
swSerialCO2 = new SoftwareSerial(rxPinCO2, txPinCO2);
|
||||||
|
#endif
|
||||||
|
#ifdef ESP32
|
||||||
|
// HardwareSerial swSerialCO2(1);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
|
|
||||||
void doByInterval() {
|
void doByInterval()
|
||||||
|
{
|
||||||
|
int reply;
|
||||||
|
|
||||||
// Serial.println("Start Mhz19temp doByInterval");
|
// Serial.println("Start Mhz19temp doByInterval");
|
||||||
if (temperatureUpdated) {
|
if (temperatureUpdated)
|
||||||
value.valD = temperature;
|
{
|
||||||
|
reply = temperature;
|
||||||
temperatureUpdated = false;
|
temperatureUpdated = false;
|
||||||
} else {
|
|
||||||
MHZ19uart_init();
|
|
||||||
Serial.println("Start temperature request");
|
|
||||||
if (MHZ19_request(13)) {
|
|
||||||
value.valD = temperature;
|
|
||||||
}; // change
|
|
||||||
}
|
}
|
||||||
regEvent(value.valD, "Mhz19temp"); //обязательный вызов хотяб один
|
else
|
||||||
|
{
|
||||||
|
MHZ19uart_init();
|
||||||
|
// Serial.println("Start temperature request");
|
||||||
|
reply = MHZ19_request(13);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (reply)
|
||||||
|
{
|
||||||
|
value.valD = reply;
|
||||||
|
regEvent(value.valD, "Mhz19temp");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
|
|
||||||
@@ -171,156 +264,103 @@ class Mhz19temp : public IoTItem {
|
|||||||
~Mhz19temp(){};
|
~Mhz19temp(){};
|
||||||
};
|
};
|
||||||
|
|
||||||
//=======================Range================
|
void *getAPI_Mhz19(String subtype, String param)
|
||||||
class Mhz19range : public IoTItem {
|
{
|
||||||
private:
|
if (subtype == F("Mhz19uart"))
|
||||||
public:
|
{
|
||||||
Mhz19range(String parameters) : IoTItem(parameters) {
|
|
||||||
rxPinCO2 = jsonReadInt(parameters, "rxPin");
|
|
||||||
txPinCO2 = jsonReadInt(parameters, "txPin");
|
|
||||||
range = jsonReadInt(parameters, "range");
|
|
||||||
ABC = jsonReadInt(parameters, "ABC");
|
|
||||||
if (!swSerialCO2) swSerialCO2 = new SoftwareSerial(rxPinCO2, txPinCO2);
|
|
||||||
}
|
|
||||||
|
|
||||||
void doByInterval() {
|
|
||||||
if (range != prevRange) {
|
|
||||||
MHZ19uart_init();
|
|
||||||
Serial.println("Start change range");
|
|
||||||
|
|
||||||
if (range == 2000) {
|
|
||||||
if (MHZ19_request(9)) {
|
|
||||||
prevRange = 2000;
|
|
||||||
value.valD = 2000;
|
|
||||||
} // change range to 2000
|
|
||||||
} else {
|
|
||||||
if (MHZ19_request(10)) {
|
|
||||||
prevRange = 5000;
|
|
||||||
value.valD = 5000;
|
|
||||||
} // change range to 5000
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
value.valD = prevRange;
|
|
||||||
}
|
|
||||||
regEvent(value.valD, "Mhz19range"); //обязательный вызов хотяб один
|
|
||||||
}
|
|
||||||
|
|
||||||
~Mhz19range(){};
|
|
||||||
};
|
|
||||||
|
|
||||||
//===================ABC=================
|
|
||||||
|
|
||||||
class Mhz19ABC : public IoTItem {
|
|
||||||
private:
|
|
||||||
public:
|
|
||||||
Mhz19ABC(String parameters) : IoTItem(parameters) {
|
|
||||||
rxPinCO2 = jsonReadInt(parameters, "rxPin");
|
|
||||||
txPinCO2 = jsonReadInt(parameters, "txPin");
|
|
||||||
range = jsonReadInt(parameters, "range");
|
|
||||||
ABC = jsonReadInt(parameters, "ABC");
|
|
||||||
if (!swSerialCO2) swSerialCO2 = new SoftwareSerial(rxPinCO2, txPinCO2);
|
|
||||||
}
|
|
||||||
|
|
||||||
void doByInterval() {
|
|
||||||
if (ABC != prevABC) {
|
|
||||||
if (ABC == 1) {
|
|
||||||
if (MHZ19_request(7)) {
|
|
||||||
prevABC = 1;
|
|
||||||
value.valD = 1;
|
|
||||||
} // change ABC to 1
|
|
||||||
} else {
|
|
||||||
if (MHZ19_request(8)) {
|
|
||||||
prevABC = 0;
|
|
||||||
value.valD = 0;
|
|
||||||
} // change ABC to 0
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
value.valD = prevABC;
|
|
||||||
}
|
|
||||||
regEvent(value.valD, "Mhz19ABC"); //обязательный вызов хотяб один
|
|
||||||
}
|
|
||||||
|
|
||||||
~Mhz19ABC(){};
|
|
||||||
};
|
|
||||||
|
|
||||||
///============== end of classes==========================
|
|
||||||
|
|
||||||
void *getAPI_Mhz19(String subtype, String param) {
|
|
||||||
if (subtype == F("Mhz19uart")) {
|
|
||||||
return new Mhz19uart(param);
|
return new Mhz19uart(param);
|
||||||
} else if (subtype == F("Mhz19pwm")) {
|
}
|
||||||
|
else if (subtype == F("Mhz19pwm"))
|
||||||
|
{
|
||||||
return new Mhz19pwm(param);
|
return new Mhz19pwm(param);
|
||||||
} else if (subtype == F("Mhz19temp")) {
|
}
|
||||||
|
else if (subtype == F("Mhz19temp"))
|
||||||
|
{
|
||||||
return new Mhz19temp(param);
|
return new Mhz19temp(param);
|
||||||
} else if (subtype == F("Mhz19range")) {
|
}
|
||||||
return new Mhz19range(param);
|
else
|
||||||
} else if (subtype == F("Mhz19ABC")) {
|
{
|
||||||
return new Mhz19ABC(param);
|
|
||||||
} else {
|
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MHZ19uart_init() {
|
void MHZ19uart_init()
|
||||||
if (MHZ19uart_flag) {
|
{
|
||||||
int reply;
|
if (MHZ19uartInit_flag)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef ESP8266
|
||||||
swSerialCO2->begin(9600);
|
swSerialCO2->begin(9600);
|
||||||
|
#endif
|
||||||
|
#ifdef ESP32
|
||||||
|
// if(swSerialCO2.available()) { Serial.println("ok"); }
|
||||||
|
swSerialCO2.begin(9600, SERIAL_8N1, rxPinCO2, txPinCO2);
|
||||||
|
#endif
|
||||||
delay(50);
|
delay(50);
|
||||||
|
int reply;
|
||||||
reply = MHZ19_request(2); // show range, for test of uart only
|
reply = MHZ19_request(2); // show range, for test of uart only
|
||||||
Serial.print("show range reply = ");
|
// Serial.print("show range reply = ");
|
||||||
Serial.println(reply);
|
// Serial.println(reply);
|
||||||
|
|
||||||
if (reply) {
|
if (reply)
|
||||||
MHZ19uart_flag = false;
|
{
|
||||||
|
prevRange = reply;
|
||||||
|
MHZ19uartInit_flag = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!MHZ19uart_flag) {
|
if (!MHZ19uartInit_flag && ABC != prevABC)
|
||||||
static int prevABC;
|
{
|
||||||
int reply;
|
int reply;
|
||||||
|
|
||||||
if (ABC != prevABC) {
|
if (ABC != prevABC)
|
||||||
if (ABC) {
|
{
|
||||||
|
if (ABC)
|
||||||
|
{
|
||||||
reply = MHZ19_request(7); // ABC on
|
reply = MHZ19_request(7); // ABC on
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
reply = MHZ19_request(8); // ABC off
|
reply = MHZ19_request(8); // ABC off
|
||||||
}
|
}
|
||||||
Serial.print("ABC change reply = ");
|
// Serial.print("ABC change reply = ");
|
||||||
Serial.println(reply);
|
// Serial.println(reply);
|
||||||
}
|
}
|
||||||
if (reply) {
|
if (reply)
|
||||||
|
{
|
||||||
prevABC = ABC;
|
prevABC = ABC;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool MHZ19_range_flag = true;
|
if (range != prevRange && !MHZ19uartInit_flag)
|
||||||
if (MHZ19_range_flag && !MHZ19uart_flag && (millis() > 30 * 1000)) {
|
{
|
||||||
int reply;
|
int reply;
|
||||||
if (range == 2000) {
|
if (range == 2000)
|
||||||
reply = MHZ19_request(9); // Установка шкалы 0-2000
|
{
|
||||||
// 255 155 0 0 7 208 0 3 139
|
reply = MHZ19_request(9); // Установка шкалы 0-2000 - 255 155 0 0 7 208 0 3 139
|
||||||
} else {
|
|
||||||
reply = MHZ19_request(10); // Установка шкалы 0-5000
|
|
||||||
// 255 155 0 0 19 136 0 3 199
|
|
||||||
}
|
}
|
||||||
Serial.print("Scale change reply = ");
|
else
|
||||||
Serial.println(reply);
|
{
|
||||||
if (reply) {
|
reply = MHZ19_request(10); // Установка шкалы 0-5000 - 255 155 0 0 19 136 0 3 199
|
||||||
reply = MHZ19_request(2); // show range
|
}
|
||||||
Serial.print("show range reply = ");
|
// Serial.print("Scale change reply = ");
|
||||||
Serial.println(reply);
|
// Serial.println(reply);
|
||||||
|
if (reply)
|
||||||
MHZ19_range_flag = false;
|
{
|
||||||
|
// reply = MHZ19_request(2); // show range
|
||||||
|
// Serial.print("show range reply = ");
|
||||||
|
// Serial.println(reply);
|
||||||
|
prevRange = range;
|
||||||
|
// MHZ19_range_flag = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int MHZ19_request(int request) {
|
int MHZ19_request(int request)
|
||||||
|
{
|
||||||
int reply;
|
int reply;
|
||||||
Serial.print("prevRange = ");
|
// Serial.print("prevRange = ");
|
||||||
Serial.println(prevRange);
|
// Serial.println(prevRange);
|
||||||
|
|
||||||
byte uartReqSamplePpm[9] = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79}; // PPM
|
byte uartReqSamplePpm[9] = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79}; // PPM
|
||||||
// 255 1 134 0 0 0 0 0 121
|
// 255 1 134 0 0 0 0 0 121
|
||||||
@@ -331,11 +371,9 @@ int MHZ19_request(int request) {
|
|||||||
// Response 255 121 1 0 0 0 0 0 134
|
// Response 255 121 1 0 0 0 0 0 134
|
||||||
byte uartReqSampleABCoff[9] = {0xFF, 0x01, 0x79, 0x00, 0x00, 0x00, 0x00, 0x00, 0x86}; // ABC logic off
|
byte uartReqSampleABCoff[9] = {0xFF, 0x01, 0x79, 0x00, 0x00, 0x00, 0x00, 0x00, 0x86}; // ABC logic off
|
||||||
byte uartReqSampleABCstatus[9] = {0xFF, 0x01, 0x7D, 0x00, 0x00, 0x00, 0x00, 0x00, 0x82}; // ABC logic status
|
byte uartReqSampleABCstatus[9] = {0xFF, 0x01, 0x7D, 0x00, 0x00, 0x00, 0x00, 0x00, 0x82}; // ABC logic status
|
||||||
|
|
||||||
byte uartReqSample1000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x03, 0xE8, 0x7B};
|
byte uartReqSample1000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x03, 0xE8, 0x7B};
|
||||||
byte uartReqSample2000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x07, 0xD0, 0x8F}; // задаёт диапазон 0 - 2000ppm
|
byte uartReqSample2000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x07, 0xD0, 0x8F}; // задаёт диапазон 0 - 2000ppm
|
||||||
byte uartReqSample3000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x0B, 0xB8, 0xA3}; // задаёт диапазон 0 - 2000ppm
|
byte uartReqSample3000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x0B, 0xB8, 0xA3}; // задаёт диапазон 0 - 2000ppm
|
||||||
|
|
||||||
byte uartReqSample5000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x13, 0x88, 0xCB}; // задаёт диапазон 0 - 5000ppm
|
byte uartReqSample5000Range[9] = {0xFF, 0x01, 0x99, 0x00, 0x00, 0x00, 0x13, 0x88, 0xCB}; // задаёт диапазон 0 - 5000ppm
|
||||||
// 255 1 153 0 0 0 19 136 203
|
// 255 1 153 0 0 0 19 136 203
|
||||||
// Response 255 153 1 0 0 0 0 0 102
|
// Response 255 153 1 0 0 0 0 0 102
|
||||||
@@ -344,85 +382,120 @@ int MHZ19_request(int request) {
|
|||||||
// reply // 255 1 155 0 0 0 0 0 100
|
// reply // 255 1 155 0 0 0 0 0 100
|
||||||
byte uartReqSampleZeroPnt[9] = {0xFF, 0x01, 0x87, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78}; // !!ZERO POINT CALIBRATION
|
byte uartReqSampleZeroPnt[9] = {0xFF, 0x01, 0x87, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78}; // !!ZERO POINT CALIBRATION
|
||||||
byte uartReqSampleReset[9] = {0xFF, 0x01, 0x8D, 0x00, 0x00, 0x00, 0x00, 0x00, 0x72}; // reset
|
byte uartReqSampleReset[9] = {0xFF, 0x01, 0x8D, 0x00, 0x00, 0x00, 0x00, 0x00, 0x72}; // reset
|
||||||
|
|
||||||
byte uartReqSampleReset1[9] = {0xFF, 0x01, 0x78, 0x00, 0x00, 0x00, 0x00, 0x00, 0x87}; // reset - did not work out
|
byte uartReqSampleReset1[9] = {0xFF, 0x01, 0x78, 0x00, 0x00, 0x00, 0x00, 0x00, 0x87}; // reset - did not work out
|
||||||
|
|
||||||
byte request_cmd[9];
|
byte request_cmd[9];
|
||||||
// byte c;
|
// byte c;
|
||||||
switch (request) {
|
switch (request)
|
||||||
|
{
|
||||||
// случаи 3-7 для перезапуска сенсора
|
// случаи 3-7 для перезапуска сенсора
|
||||||
case 1: {
|
case 1:
|
||||||
Serial.println("Запрос No.1 - отправлен. Запрос замера по UART");
|
{
|
||||||
for (int i = 0; i < 9; i++) {
|
// Serial.println("Запрос No.1 - отправлен. Запрос замера по UART");
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSamplePpm[i];
|
request_cmd[i] = uartReqSamplePpm[i];
|
||||||
}
|
}
|
||||||
} break;
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case 2: {
|
case 2:
|
||||||
for (int i = 0; i < 9; i++) {
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSampleRequestRange[i];
|
request_cmd[i] = uartReqSampleRequestRange[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.2 - отправлен. Запрос шкалы");
|
// Serial.println("Запрос No.2 - отправлен. Запрос шкалы");
|
||||||
} break;
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case 3: {
|
case 3:
|
||||||
|
{
|
||||||
// код для запуска сенсора с работой по UART (запуск таймера)
|
// код для запуска сенсора с работой по UART (запуск таймера)
|
||||||
Serial.println("Запрос No.3 - отправлен. Сенсор по UART запущен");
|
// Serial.println("Запрос No.3 - отправлен. Сенсор по UART запущен");
|
||||||
} break;
|
}
|
||||||
case 4: {
|
break;
|
||||||
|
case 4:
|
||||||
|
{
|
||||||
// код для остановке сенсора с работой по UART (остановка таймера)
|
// код для остановке сенсора с работой по UART (остановка таймера)
|
||||||
Serial.println("Запрос No.4 - отправлен. Сенсор по UART остановлен");
|
// Serial.println("Запрос No.4 - отправлен. Сенсор по UART остановлен");
|
||||||
} break;
|
}
|
||||||
case 5: {
|
break;
|
||||||
|
case 5:
|
||||||
|
{
|
||||||
// код для запуска сенсора с работой по PWM (запуск таймера)
|
// код для запуска сенсора с работой по PWM (запуск таймера)
|
||||||
Serial.println("Запрос No.5 - отправлен. Сенсор по PWM запущен");
|
// Serial.println("Запрос No.5 - отправлен. Сенсор по PWM запущен");
|
||||||
} break;
|
}
|
||||||
case 6: {
|
break;
|
||||||
|
case 6:
|
||||||
|
{
|
||||||
// код для остановки сенсора с работой по PWM (остановка таймера)
|
// код для остановки сенсора с работой по PWM (остановка таймера)
|
||||||
Serial.println("Запрос No.6 - отправлен. Сенсор по PWM остановлен");
|
// Serial.println("Запрос No.6 - отправлен. Сенсор по PWM остановлен");
|
||||||
} break;
|
}
|
||||||
case 7: {
|
break;
|
||||||
for (int i = 0; i < 9; i++) {
|
case 7:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSampleABCon[i];
|
request_cmd[i] = uartReqSampleABCon[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.7 - отправлен. Включаем функцию атокалибровки");
|
// Serial.println("Запрос No.7 - отправлен. Включаем функцию атокалибровки");
|
||||||
} break;
|
}
|
||||||
case 8: {
|
break;
|
||||||
for (int i = 0; i < 9; i++) {
|
case 8:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSampleABCoff[i];
|
request_cmd[i] = uartReqSampleABCoff[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.8 - отправлен. Выключаем функцию атокалибровки");
|
// Serial.println("Запрос No.8 - отправлен. Выключаем функцию атокалибровки");
|
||||||
} break;
|
}
|
||||||
case 9: {
|
break;
|
||||||
for (int i = 0; i < 9; i++) {
|
case 9:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSample2000Range[i];
|
request_cmd[i] = uartReqSample2000Range[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.9 - отправлен. Установливаем шкалу 0-2000");
|
// Serial.println("Запрос No.9 - отправлен. Установливаем шкалу 0-2000");
|
||||||
} break;
|
}
|
||||||
case 10: {
|
break;
|
||||||
for (int i = 0; i < 9; i++) {
|
case 10:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSample5000Range[i];
|
request_cmd[i] = uartReqSample5000Range[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.10 - отправлен. Установливаем шкалу 0-5000");
|
// Serial.println("Запрос No.10 - отправлен. Установливаем шкалу 0-5000");
|
||||||
} break;
|
}
|
||||||
case 11: {
|
break;
|
||||||
for (int i = 0; i < 9; i++) {
|
case 11:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSampleZeroPnt[i];
|
request_cmd[i] = uartReqSampleZeroPnt[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.11 - отправлен. Калибровка. Установливаем нулевой уровень");
|
// Serial.println("Запрос No.11 - отправлен. Калибровка. Установливаем нулевой уровень");
|
||||||
} break;
|
}
|
||||||
case 12: {
|
break;
|
||||||
for (int i = 0; i < 9; i++) {
|
case 12:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSampleReset1[i];
|
request_cmd[i] = uartReqSampleReset1[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.11 - отправлен. Запрос на Сброс");
|
// Serial.println("Запрос No.11 - отправлен. Запрос на Сброс");
|
||||||
} break;
|
}
|
||||||
case 13: {
|
break;
|
||||||
for (int i = 0; i < 9; i++) {
|
case 13:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
request_cmd[i] = uartReqSamplePpm[i];
|
request_cmd[i] = uartReqSamplePpm[i];
|
||||||
}
|
}
|
||||||
Serial.println("Запрос No.13 - отправлен. Запрос по Температуре");
|
// Serial.println("Запрос No.13 - отправлен. Запрос по Температуре");
|
||||||
} break;
|
}
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
// byte c = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79};
|
// byte c = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79};
|
||||||
// if nothing else matches, do the default
|
// if nothing else matches, do the default
|
||||||
@@ -430,110 +503,159 @@ int MHZ19_request(int request) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef ESP8266
|
||||||
swSerialCO2->write(request_cmd, 9);
|
swSerialCO2->write(request_cmd, 9);
|
||||||
|
#endif
|
||||||
|
#ifdef ESP32
|
||||||
|
swSerialCO2.write(request_cmd, 9);
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
Serial.print("Request : ");
|
Serial.print("Request : ");
|
||||||
for (int i = 0; i < 9; i++) {
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
Serial.print(" ");
|
Serial.print(" ");
|
||||||
Serial.print(request_cmd[i]);
|
Serial.print(request_cmd[i]);
|
||||||
}
|
}
|
||||||
Serial.println(" ");
|
Serial.println(" ");
|
||||||
|
*/
|
||||||
delay(50);
|
delay(5);
|
||||||
|
|
||||||
unsigned char response[9];
|
unsigned char response[9];
|
||||||
|
|
||||||
Serial.print("Response :");
|
#ifdef ESP8266
|
||||||
swSerialCO2->readBytes(response, 9);
|
swSerialCO2->readBytes(response, 9);
|
||||||
|
#endif
|
||||||
|
|
||||||
for (int i = 0; i < 9; i++) {
|
#ifdef ESP32
|
||||||
|
swSerialCO2.readBytes(response, 9);
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
Serial.print("Response :");
|
||||||
|
for (int i = 0; i < 9; i++)
|
||||||
|
{
|
||||||
Serial.print(" ");
|
Serial.print(" ");
|
||||||
Serial.print(response[i]);
|
Serial.print(response[i]);
|
||||||
}
|
}
|
||||||
Serial.println(" ");
|
Serial.println(" ");
|
||||||
|
*/
|
||||||
byte crc = 0;
|
byte crc = 0;
|
||||||
for (int i = 1; i < 8; i++) crc += response[i];
|
for (int i = 1; i < 8; i++)
|
||||||
|
crc += response[i];
|
||||||
crc = 255 - crc;
|
crc = 255 - crc;
|
||||||
crc += 1;
|
crc += 1;
|
||||||
|
|
||||||
if (!(response[0] == 0xFF && response[8] == crc)) {
|
if (!(response[0] == 0xFF && response[8] == crc))
|
||||||
Serial.println("Range CRC error: " + String(crc) + " / " + String(response[8]));
|
{
|
||||||
|
String msg = "Range CRC error: " + String(crc) + " / " + String(response[8]) + " Check wiring";
|
||||||
|
SerialPrint("E", "Sensor Mhz19uart", msg);
|
||||||
reply = 0;
|
reply = 0;
|
||||||
} else {
|
MHZ19uartInit_flag = true; //
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
// Serial.println("No CRC errors");
|
// Serial.println("No CRC errors");
|
||||||
switch (request) {
|
switch (request)
|
||||||
case 1: {
|
{
|
||||||
|
case 1:
|
||||||
|
{
|
||||||
unsigned int responseHigh = (unsigned int)response[2];
|
unsigned int responseHigh = (unsigned int)response[2];
|
||||||
unsigned int responseLow = (unsigned int)response[3];
|
unsigned int responseLow = (unsigned int)response[3];
|
||||||
unsigned int ppm = (256 * responseHigh) + responseLow;
|
unsigned int ppm = (256 * responseHigh) + responseLow;
|
||||||
|
|
||||||
Serial.print("CO2 UART = ");
|
// Serial.print("CO2 UART = ");
|
||||||
Serial.println(ppm);
|
// Serial.println(ppm);
|
||||||
|
|
||||||
temperature = response[4] - 44; // - 40;
|
temperature = response[4] - 44; // - 40;
|
||||||
Serial.print("Temperature = ");
|
// Serial.print("Temperature = ");
|
||||||
Serial.println(temperature);
|
// Serial.println(temperature);
|
||||||
temperatureUpdated = true;
|
temperatureUpdated = true;
|
||||||
|
|
||||||
reply = ppm;
|
reply = ppm;
|
||||||
} break;
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case 2: {
|
case 2:
|
||||||
reply = 1;
|
{
|
||||||
Serial.println("Case 2 - OK. На запрос шкалы пришел ответ");
|
|
||||||
} break;
|
unsigned int responseHigh = (unsigned int)response[4];
|
||||||
case 3: {
|
unsigned int responseLow = (unsigned int)response[5];
|
||||||
|
unsigned int scale = (256 * responseHigh) + responseLow;
|
||||||
|
reply = scale;
|
||||||
|
SerialPrint("i", "Sensor Mhz19uart", "Запрос No.2 - сработал. Шкала получена - " + String(scale));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
{
|
||||||
// Serial.println("Case 3 - OK");
|
// Serial.println("Case 3 - OK");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
} break;
|
}
|
||||||
case 4: {
|
break;
|
||||||
|
case 4:
|
||||||
|
{
|
||||||
// Serial.println("Case 4 - OK");
|
// Serial.println("Case 4 - OK");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
} break;
|
}
|
||||||
case 5: {
|
break;
|
||||||
|
case 5:
|
||||||
|
{
|
||||||
// Serial.println("Case 5 - OK");
|
// Serial.println("Case 5 - OK");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
} break;
|
}
|
||||||
case 6: {
|
break;
|
||||||
Serial.println("Case 6 - OK");
|
case 6:
|
||||||
|
{
|
||||||
|
// Serial.println("Запрос No.6 - сработал.");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
} break;
|
}
|
||||||
case 7: {
|
break;
|
||||||
Serial.println("Case 7 - OK. ABC включен");
|
case 7:
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Mhz19uart", "Запрос No.7 - сработал. ABC включен");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
} break;
|
}
|
||||||
case 8: {
|
break;
|
||||||
Serial.println("Case 8 - OK. ABC выключен");
|
case 8:
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Mhz19uart", "Запрос No.8 - сработал. ABC выключен");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
} break;
|
}
|
||||||
case 9: {
|
break;
|
||||||
Serial.println("Case 9 - OK. Установлена шкала 0-2000");
|
case 9:
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Mhz19uart", "Запрос No.9 - сработал. Установлена шкала 0-2000");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
prevRange = 2000;
|
prevRange = 2000;
|
||||||
} break;
|
}
|
||||||
case 10: {
|
break;
|
||||||
Serial.println("Case 9 - OK. Установлена шкала 0-5000");
|
case 10:
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Mhz19uart", "Запрос No.10 - сработал. Установлена шкала 0-5000");
|
||||||
reply = 1;
|
reply = 1;
|
||||||
prevRange = 5000;
|
prevRange = 5000;
|
||||||
} break;
|
}
|
||||||
case 11: {
|
break;
|
||||||
|
case 11:
|
||||||
|
{
|
||||||
reply = 1;
|
reply = 1;
|
||||||
Serial.println("Запрос No.11 - сработал. Калибровка. Установлен нулевой уровень");
|
SerialPrint("i", "Sensor Mhz19uart", "Запрос No.11 - сработал. Калибровка. Установлен нулевой уровень");
|
||||||
} break;
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case 12: {
|
case 12:
|
||||||
|
{
|
||||||
reply = 1;
|
reply = 1;
|
||||||
Serial.println("Запрос No.12 - сработал. Сброс произошел");
|
SerialPrint("i", "Sensor Mhz19uart", "Запрос No.12 - сработал. Сброс произошел");
|
||||||
} break;
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case 13: {
|
case 13:
|
||||||
reply = 1;
|
{
|
||||||
temperature = response[4] - 44; // - 40;
|
temperature = response[4] - 44; // - 40;
|
||||||
|
reply = temperature;
|
||||||
Serial.println("Запрос No.12 - сработал. Температура получена");
|
// SerialPrint("i", "Sensor Mhz19uart", "Запрос No.13 - сработал. Температура получена - " + String(temperature));
|
||||||
Serial.println(temperature);
|
// Serial.print("Запрос No.13 - сработал. Температура получена - ");
|
||||||
} break;
|
// Serial.println(temperature);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
// byte c = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79};
|
// byte c = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79};
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"menuSection": "Сенсоры",
|
"menuSection": "Сенсоры",
|
||||||
"configItem": [
|
"configItem": [
|
||||||
{
|
{
|
||||||
"name": "MHZ-19 CO2 (UART)",
|
"name": "MHZ-19 CO2 UART",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Mhz19uart",
|
"subtype": "Mhz19uart",
|
||||||
"id": "co2uart",
|
"id": "co2uart",
|
||||||
@@ -12,7 +12,7 @@
|
|||||||
"plus": 0,
|
"plus": 0,
|
||||||
"multiply": 1,
|
"multiply": 1,
|
||||||
"round": 1,
|
"round": 1,
|
||||||
"pin": 0,
|
"warmUp": 120,
|
||||||
"rxPin": 13,
|
"rxPin": 13,
|
||||||
"txPin": 12,
|
"txPin": 12,
|
||||||
"int": 15,
|
"int": 15,
|
||||||
@@ -20,7 +20,7 @@
|
|||||||
"ABC": 1
|
"ABC": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "MHZ-19 CO2 (PWM)",
|
"name": "MHZ-19 CO2 PWM",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Mhz19pwm",
|
"subtype": "Mhz19pwm",
|
||||||
"id": "co2pwm",
|
"id": "co2pwm",
|
||||||
@@ -30,11 +30,13 @@
|
|||||||
"plus": 0,
|
"plus": 0,
|
||||||
"multiply": 1,
|
"multiply": 1,
|
||||||
"round": 1,
|
"round": 1,
|
||||||
|
"warmUp": 120,
|
||||||
"pin": 15,
|
"pin": 15,
|
||||||
|
"maxRetriesNotAvailable": 10,
|
||||||
"int": 300
|
"int": 300
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Cенсор температуры от MHZ-19 UART",
|
"name": "MHZ-19 Температура UART",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Mhz19temp",
|
"subtype": "Mhz19temp",
|
||||||
"id": "Mhz19temp",
|
"id": "Mhz19temp",
|
||||||
@@ -46,64 +48,28 @@
|
|||||||
"round": 1,
|
"round": 1,
|
||||||
"rxPin": 13,
|
"rxPin": 13,
|
||||||
"txPin": 12,
|
"txPin": 12,
|
||||||
"ABC": 1,
|
|
||||||
"int": 30
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "Рабочий диапазон от MHZ-19 UART",
|
|
||||||
"type": "Reading",
|
|
||||||
"subtype": "Mhz19range",
|
|
||||||
"id": "Mhz19range",
|
|
||||||
"widget": "anydataPpm",
|
|
||||||
"page": "Сенсоры",
|
|
||||||
"descr": "Диапазон",
|
|
||||||
"plus": 0,
|
|
||||||
"multiply": 1,
|
|
||||||
"round": 1,
|
|
||||||
"rxPin": 13,
|
|
||||||
"txPin": 12,
|
|
||||||
"range": 5000,
|
|
||||||
"ABC": 1,
|
|
||||||
"int": 30
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "Автокалибровка от MHZ-19 UART",
|
|
||||||
"type": "Reading",
|
|
||||||
"subtype": "Mhz19ABC",
|
|
||||||
"id": "Mhz19ABC",
|
|
||||||
"widget": "anydataDef",
|
|
||||||
"page": "Сенсоры",
|
|
||||||
"descr": "ABC",
|
|
||||||
"rxPin": 13,
|
|
||||||
"txPin": 12,
|
|
||||||
"range": 5000,
|
|
||||||
"ABC": 1,
|
|
||||||
"int": 30
|
"int": 30
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"about": {
|
"about": {
|
||||||
"authorName": "Alex K",
|
"authorName": "Alex K",
|
||||||
"authorContact": "https://t.me/cmche",
|
"authorContact": "https://t.me/cmche",
|
||||||
"authorGit": "",
|
"authorGit": "https://github.com/CHE77/Mhz19forIotManager",
|
||||||
"specialThanks": "",
|
"specialThanks": "",
|
||||||
"moduleName": "Mhz19",
|
"moduleName": "Mhz19",
|
||||||
"moduleVersion": "1.0",
|
"moduleVersion": "2.0 - можно переназначать пины",
|
||||||
"usedRam": 15,
|
"moduleDesc": "Позволяет получить значения уровня концетрации CO2 с Mhz19 по UART и/или ШИМ. Замер по ШИМ может производить 1-2 сек. задержки",
|
||||||
"subTypes": [
|
|
||||||
"Mhz19uart",
|
|
||||||
"Mhz19pwm",
|
|
||||||
"Mhz19temp",
|
|
||||||
"Mhz19range",
|
|
||||||
"Mhz19ABC"
|
|
||||||
],
|
|
||||||
"title": "Датчик температуры и CO2 с Mhz19",
|
|
||||||
"moduleDesc": "Позволяет получить значения температуры и CO2 с Mhz19.",
|
|
||||||
"propInfo": {
|
"propInfo": {
|
||||||
|
"plus": "поправочный коэффиент +c",
|
||||||
|
"multiply": "поправочный коэффиент k*",
|
||||||
|
"round": "округление",
|
||||||
"int": "Количество секунд между опросами датчика.",
|
"int": "Количество секунд между опросами датчика.",
|
||||||
"rxPin": "",
|
"rxPin": "Esp8266: GPIO 13 - D7, ESP32: GPIO 19 - RX1, > MHZ19: TXD зеленый провод",
|
||||||
"txPin": "",
|
"txPin": "Esp8266: GPIO 12 - D6, ESP32: GPIO 18 - TX1, > MHZ19: RXD синий провод",
|
||||||
"range": "",
|
"range": "Шкала по умолчанию 0-5000ppm. Также можно выбрать 2000",
|
||||||
"ABC": ""
|
"ABC": "Автокалибровка. По умолчанию включена. Раз в сутки на 20 мин. надо выставлять на свежий воздух.",
|
||||||
|
"pin": "пин получения значений по ШИМ. Esp8266: GPIO 15 - D8, ESP32: GPIO 21, > MHZ19: PWM желтый провод",
|
||||||
|
"maxRetriesNotAvailable": "Максимальное количество попыток опроса сенсора по ШИМ. (может задерживать контроллер)"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"defActive": false,
|
"defActive": false,
|
||||||
|
|||||||
@@ -5,11 +5,10 @@
|
|||||||
#include "PZEMSensor.h"
|
#include "PZEMSensor.h"
|
||||||
#include "modules/sensors/UART/Uart.h"
|
#include "modules/sensors/UART/Uart.h"
|
||||||
|
|
||||||
PZEMSensor* pzem;
|
|
||||||
|
|
||||||
class Pzem004v : public IoTItem {
|
class Pzem004v : public IoTItem {
|
||||||
private:
|
private:
|
||||||
String addr;
|
String addr;
|
||||||
|
PZEMSensor* pzem;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Pzem004v(String parameters) : IoTItem(parameters) {
|
Pzem004v(String parameters) : IoTItem(parameters) {
|
||||||
@@ -21,17 +20,147 @@ class Pzem004v : public IoTItem {
|
|||||||
|
|
||||||
void doByInterval() {
|
void doByInterval() {
|
||||||
if (pzem) {
|
if (pzem) {
|
||||||
value.valD = pzem->values()->voltage;
|
bool online = false;
|
||||||
regEvent(value.valD, "Pzem Voltage");
|
value.valD = pzem->values(online)->voltage;
|
||||||
|
if (online) {
|
||||||
|
regEvent(value.valD, "Pzem V");
|
||||||
|
} else {
|
||||||
|
regEvent(NAN, "Pzem V");
|
||||||
|
SerialPrint("E", "Pzem", "V error");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
~Pzem004v(){};
|
~Pzem004v(){};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class Pzem004a : public IoTItem {
|
||||||
|
private:
|
||||||
|
String addr;
|
||||||
|
PZEMSensor* pzem;
|
||||||
|
|
||||||
|
public:
|
||||||
|
Pzem004a(String parameters) : IoTItem(parameters) {
|
||||||
|
addr = jsonReadStr(parameters, "addr");
|
||||||
|
if (myUART) {
|
||||||
|
pzem = new PZEMSensor(myUART, hexStringToUint8(addr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void doByInterval() {
|
||||||
|
if (pzem) {
|
||||||
|
bool online = false;
|
||||||
|
value.valD = pzem->values(online)->current;
|
||||||
|
if (online) {
|
||||||
|
regEvent(value.valD, "Pzem A");
|
||||||
|
} else {
|
||||||
|
regEvent(NAN, "Pzem A");
|
||||||
|
SerialPrint("E", "Pzem", "A error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~Pzem004a(){};
|
||||||
|
};
|
||||||
|
|
||||||
|
class Pzem004w : public IoTItem {
|
||||||
|
private:
|
||||||
|
String addr;
|
||||||
|
PZEMSensor* pzem;
|
||||||
|
|
||||||
|
public:
|
||||||
|
Pzem004w(String parameters) : IoTItem(parameters) {
|
||||||
|
addr = jsonReadStr(parameters, "addr");
|
||||||
|
if (myUART) {
|
||||||
|
pzem = new PZEMSensor(myUART, hexStringToUint8(addr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void doByInterval() {
|
||||||
|
if (pzem) {
|
||||||
|
bool online = false;
|
||||||
|
value.valD = pzem->values(online)->power;
|
||||||
|
if (online) {
|
||||||
|
regEvent(value.valD, "Pzem W");
|
||||||
|
} else {
|
||||||
|
regEvent(NAN, "Pzem W");
|
||||||
|
SerialPrint("E", "Pzem", "W error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~Pzem004w(){};
|
||||||
|
};
|
||||||
|
|
||||||
|
class Pzem004wh : public IoTItem {
|
||||||
|
private:
|
||||||
|
String addr;
|
||||||
|
PZEMSensor* pzem;
|
||||||
|
|
||||||
|
public:
|
||||||
|
Pzem004wh(String parameters) : IoTItem(parameters) {
|
||||||
|
addr = jsonReadStr(parameters, "addr");
|
||||||
|
if (myUART) {
|
||||||
|
pzem = new PZEMSensor(myUART, hexStringToUint8(addr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void doByInterval() {
|
||||||
|
if (pzem) {
|
||||||
|
bool online = false;
|
||||||
|
value.valD = pzem->values(online)->energy;
|
||||||
|
if (online) {
|
||||||
|
regEvent(value.valD, "Pzem Wh");
|
||||||
|
} else {
|
||||||
|
regEvent(NAN, "Pzem Wh");
|
||||||
|
SerialPrint("E", "Pzem", "Wh error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~Pzem004wh(){};
|
||||||
|
};
|
||||||
|
|
||||||
|
class Pzem004hz : public IoTItem {
|
||||||
|
private:
|
||||||
|
String addr;
|
||||||
|
PZEMSensor* pzem;
|
||||||
|
|
||||||
|
public:
|
||||||
|
Pzem004hz(String parameters) : IoTItem(parameters) {
|
||||||
|
addr = jsonReadStr(parameters, "addr");
|
||||||
|
if (myUART) {
|
||||||
|
pzem = new PZEMSensor(myUART, hexStringToUint8(addr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void doByInterval() {
|
||||||
|
if (pzem) {
|
||||||
|
bool online = false;
|
||||||
|
value.valD = pzem->values(online)->freq;
|
||||||
|
if (online) {
|
||||||
|
regEvent(value.valD, "Pzem Hz");
|
||||||
|
} else {
|
||||||
|
regEvent(NAN, "Pzem Hz");
|
||||||
|
SerialPrint("E", "Pzem", "Hz error");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~Pzem004hz(){};
|
||||||
|
};
|
||||||
|
|
||||||
void* getAPI_Pzem004(String subtype, String param) {
|
void* getAPI_Pzem004(String subtype, String param) {
|
||||||
if (subtype == F("Pzem004v")) {
|
if (subtype == F("Pzem004v")) {
|
||||||
return new Pzem004v(param);
|
return new Pzem004v(param);
|
||||||
|
} else if (subtype == F("Pzem004a")) {
|
||||||
|
return new Pzem004a(param);
|
||||||
|
} else if (subtype == F("Pzem004w")) {
|
||||||
|
return new Pzem004w(param);
|
||||||
|
} else if (subtype == F("Pzem004wh")) {
|
||||||
|
return new Pzem004wh(param);
|
||||||
|
} else if (subtype == F("Pzem004hz")) {
|
||||||
|
return new Pzem004hz(param);
|
||||||
} else {
|
} else {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
"name": "PZEM 004t Напряжение",
|
"name": "PZEM 004t Напряжение",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004v",
|
"subtype": "Pzem004v",
|
||||||
"id": "V",
|
"id": "v",
|
||||||
"widget": "anydataVlt",
|
"widget": "anydataVlt",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Напряжение",
|
"descr": "Напряжение",
|
||||||
@@ -16,7 +16,7 @@
|
|||||||
"name": "PZEM 004t Сила тока",
|
"name": "PZEM 004t Сила тока",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004a",
|
"subtype": "Pzem004a",
|
||||||
"id": "A",
|
"id": "a",
|
||||||
"widget": "anydataAmp",
|
"widget": "anydataAmp",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Сила тока",
|
"descr": "Сила тока",
|
||||||
@@ -27,7 +27,7 @@
|
|||||||
"name": "PZEM 004t Мощность",
|
"name": "PZEM 004t Мощность",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004w",
|
"subtype": "Pzem004w",
|
||||||
"id": "W",
|
"id": "w",
|
||||||
"widget": "anydataWt",
|
"widget": "anydataWt",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Мощность",
|
"descr": "Мощность",
|
||||||
@@ -38,7 +38,7 @@
|
|||||||
"name": "PZEM 004t Энергия",
|
"name": "PZEM 004t Энергия",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004wh",
|
"subtype": "Pzem004wh",
|
||||||
"id": "Wh",
|
"id": "wh",
|
||||||
"widget": "anydataWth",
|
"widget": "anydataWth",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Энергия",
|
"descr": "Энергия",
|
||||||
@@ -49,12 +49,12 @@
|
|||||||
"name": "PZEM 004t Частота",
|
"name": "PZEM 004t Частота",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Pzem004hz",
|
"subtype": "Pzem004hz",
|
||||||
"id": "Hz",
|
"id": "hz",
|
||||||
"widget": "anydataHtz",
|
"widget": "anydataHtz",
|
||||||
"page": "Сенсоры",
|
"page": "Сенсоры",
|
||||||
"descr": "Энергия",
|
"descr": "Частота",
|
||||||
"int": 15,
|
"int": 15,
|
||||||
"addr": "0x77"
|
"addr": "0xF8"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"about": {
|
"about": {
|
||||||
@@ -72,16 +72,20 @@
|
|||||||
"Pzem004wh",
|
"Pzem004wh",
|
||||||
"Pzem004hz"
|
"Pzem004hz"
|
||||||
],
|
],
|
||||||
"title": "Счетчик электроэнергии PZEM 004t версии 3.0 (с модбасом)",
|
"title": "Счетчик электроэнергии PZEM 004 t версии 3.0 (с модбасом). Возможно подключение трех счетчиков к одной esp для трехфазных сетей. Для этого нужно настроить разные адреса modbus в платах pzem",
|
||||||
"moduleDesc": "Считает потраченную электроэнергию, измеряет напряжение, частоту, силу тока и прочие параметры",
|
"moduleDesc": "Считает потраченную электроэнергию, измеряет напряжение, частоту, силу тока и прочие параметры",
|
||||||
"propInfo": {
|
"propInfo": {
|
||||||
"addr": "Адрес modbus",
|
"addr": "Адрес modbus",
|
||||||
"int": "Количество секунд между опросами датчика"
|
"int": "Количество секунд между опросами датчика. Желателно устанавливать разные интервалы для параметров что бы опросы происходили в разное время."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"defActive": true,
|
"defActive": true,
|
||||||
"devices": {
|
"devices": {
|
||||||
"esp32_4mb": [],
|
"esp32_4mb": [
|
||||||
"esp8266_4mb": []
|
"mandulaj/PZEM-004T-v30"
|
||||||
|
],
|
||||||
|
"esp8266_4mb": [
|
||||||
|
"mandulaj/PZEM-004T-v30"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -14,33 +14,50 @@ extern IoTGpio IoTgpio;
|
|||||||
|
|
||||||
#include "SdsDustSensor.h"
|
#include "SdsDustSensor.h"
|
||||||
|
|
||||||
//встроена в ядро для 8266, для 32 по этому же имени обращаемся к другой библиотеке plerup/EspSoftwareSerial
|
//#define __DEBUG_SDS_DUST_SENSOR__
|
||||||
#include <SoftwareSerial.h>
|
|
||||||
|
int rxPinSDS; // Esp8266: 14/D5 – подключаем к Tx сенсора
|
||||||
|
int txPinSDS; // Esp8266: 16/D0 – подключаем к Rx сенсора
|
||||||
|
|
||||||
|
// SdsDustSensor sds(rxPinSDS, txPinSDS);
|
||||||
|
SdsDustSensor *sds = nullptr;
|
||||||
|
|
||||||
// to do убрать глобальный экземпляр
|
|
||||||
#ifdef ESP8266
|
|
||||||
int rxPinSDS = 13; // D7 – подключаем к Tx сенсора
|
|
||||||
int txPinSDS = 12; // D6 – подключаем к Rx сенсора
|
|
||||||
SdsDustSensor sds(rxPinSDS, txPinSDS);
|
|
||||||
#endif
|
|
||||||
#ifdef ESP32
|
#ifdef ESP32
|
||||||
#include <HardwareSerial.h>
|
#include <HardwareSerial.h>
|
||||||
HardwareSerial sdsSerial(2);
|
HardwareSerial sdsSerial(2);
|
||||||
SdsDustSensor sds(sdsSerial);
|
// SdsDustSensor sds(sdsSerial);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
unsigned int warmUp;
|
int retryDelayMs = 5;
|
||||||
unsigned int period;
|
int maxRetriesNotAvailable = 100;
|
||||||
|
unsigned int purge = 30;
|
||||||
|
unsigned int interval = 300;
|
||||||
|
unsigned int purgeDelay = 270;
|
||||||
|
unsigned int continuous = 0;
|
||||||
|
bool startUp = true;
|
||||||
|
|
||||||
|
TickerScheduler ts_sds(2);
|
||||||
|
|
||||||
|
enum TimerTask_t_sds
|
||||||
|
{
|
||||||
|
WAKEUP,
|
||||||
|
PRINT
|
||||||
|
};
|
||||||
|
|
||||||
|
int firstSensor = 0;
|
||||||
bool SDS011_init_flag = true;
|
bool SDS011_init_flag = true;
|
||||||
void SDS011_init();
|
void SDS011_init();
|
||||||
float Sds011request(int sensorID);
|
void Sds011request(int sensorID);
|
||||||
|
|
||||||
|
IoTItem *item_Sds011_25 = nullptr; // pointer
|
||||||
|
IoTItem *item_Sds011_10 = nullptr;
|
||||||
|
|
||||||
//Это файл сенсора, в нем осуществляется чтение сенсора.
|
//Это файл сенсора, в нем осуществляется чтение сенсора.
|
||||||
//для добавления сенсора вам нужно скопировать этот файл и заменить в нем текст AnalogAdc на название вашего сенсора
|
//для добавления сенсора вам нужно скопировать этот файл и заменить в нем текст AnalogAdc на название вашего сенсора
|
||||||
//Название должно быть уникальным, коротким и отражать суть сенсора.
|
//Название должно быть уникальным, коротким и отражать суть сенсора.
|
||||||
|
|
||||||
class Sds011_25 : public IoTItem {
|
class Sds011_25 : public IoTItem
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// Секция переменных.
|
// Секция переменных.
|
||||||
@@ -55,14 +72,28 @@ class Sds011_25 : public IoTItem {
|
|||||||
//Такие как ...begin и подставлять в них параметры полученные из web интерфейса.
|
//Такие как ...begin и подставлять в них параметры полученные из web интерфейса.
|
||||||
//Все параметры хранятся в перемененной parameters, вы можете прочитать любой параметр используя jsonRead функции:
|
//Все параметры хранятся в перемененной parameters, вы можете прочитать любой параметр используя jsonRead функции:
|
||||||
// jsonReadStr, jsonReadBool, jsonReadInt
|
// jsonReadStr, jsonReadBool, jsonReadInt
|
||||||
Sds011_25(String parameters) : IoTItem(parameters) {
|
Sds011_25(String parameters) : IoTItem(parameters)
|
||||||
// _pin = jsonReadInt(parameters, "pin");
|
{
|
||||||
#ifdef ESP8266
|
item_Sds011_25 = this;
|
||||||
|
|
||||||
rxPinSDS = jsonReadInt(parameters, "rxPin");
|
rxPinSDS = jsonReadInt(parameters, "rxPin");
|
||||||
txPinSDS = jsonReadInt(parameters, "txPin");
|
txPinSDS = jsonReadInt(parameters, "txPin");
|
||||||
#endif
|
|
||||||
warmUp = jsonReadInt(parameters, "warmUp"); // сек. пробужнение должен быть больше
|
purge = jsonReadInt(parameters, "purge"); // сек. пробужнение должен быть больше
|
||||||
period = jsonReadInt(parameters, "period"); // сек. время зарогрева/продувки, затем идут замеры
|
interval = jsonReadInt(parameters, "int"); // сек. время зарогрева/продувки, затем идут замеры
|
||||||
|
continuous = jsonReadInt(parameters, "continuousMode"); // сек. время зарогрева/продувки, затем идут замеры
|
||||||
|
maxRetriesNotAvailable = jsonReadInt(parameters, "maxRetriesNotAvailable"); // сек. время зарогрева/продувки, затем идут замеры
|
||||||
|
retryDelayMs = jsonReadInt(parameters, "retryDelayMs"); // сек. время зарогрева/продувки, затем идут замеры
|
||||||
|
if (continuous)
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Sds011", "Continuous mode");
|
||||||
|
ts_sds.remove(PRINT);
|
||||||
|
ts_sds.remove(WAKEUP);
|
||||||
|
}
|
||||||
|
purgeDelay = interval - purge;
|
||||||
|
|
||||||
|
SDS011_init();
|
||||||
|
firstSensor = 0;
|
||||||
}
|
}
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// doByInterval()
|
// doByInterval()
|
||||||
@@ -72,20 +103,21 @@ class Sds011_25 : public IoTItem {
|
|||||||
//если у сенсора несколько величин то делайте несколько regEvent
|
//если у сенсора несколько величин то делайте несколько regEvent
|
||||||
//не используйте delay - помните, что данный loop общий для всех модулей. Если у вас планируется длительная операция, постарайтесь разбить ее на порции
|
//не используйте delay - помните, что данный loop общий для всех модулей. Если у вас планируется длительная операция, постарайтесь разбить ее на порции
|
||||||
//и выполнить за несколько тактов
|
//и выполнить за несколько тактов
|
||||||
void doByInterval() {
|
void doByInterval()
|
||||||
|
{
|
||||||
SDS011_init();
|
SDS011_init();
|
||||||
Serial.println("request from 25");
|
// SerialPrint("i", "Sensor Sds011", "Request for 2.5");
|
||||||
value.valD = Sds011request(25);
|
Sds011request(25);
|
||||||
|
// value.valD = Sds011request(25);
|
||||||
regEvent(value.valD, "Sds011_25"); //обязательный вызов хотяб один
|
// regEvent(value.valD, "Sds011_25"); //обязательный вызов хотяб один
|
||||||
}
|
}
|
||||||
|
|
||||||
~Sds011_25(){};
|
~Sds011_25(){};
|
||||||
};
|
};
|
||||||
|
|
||||||
//////////////////////////////////// for PM 10//=
|
//////////////////////////////////// for PM 10//=
|
||||||
|
|
||||||
class Sds011_10 : public IoTItem {
|
class Sds011_10 : public IoTItem
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// Секция переменных.
|
// Секция переменных.
|
||||||
@@ -95,18 +127,46 @@ class Sds011_10 : public IoTItem {
|
|||||||
public:
|
public:
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// setup()
|
// setup()
|
||||||
//это аналог setup из arduino. Здесь вы можете выполнять методы инициализации сенсора.
|
|
||||||
//Такие как ...begin и подставлять в них параметры полученные из web интерфейса.
|
Sds011_10(String parameters) : IoTItem(parameters)
|
||||||
//Все параметры хранятся в перемененной parameters, вы можете прочитать любой параметр используя jsonRead функции:
|
{
|
||||||
// jsonReadStr, jsonReadBool, jsonReadInt
|
item_Sds011_10 = this;
|
||||||
Sds011_10(String parameters) : IoTItem(parameters) {
|
|
||||||
// _pin = jsonReadInt(parameters, "pin");
|
|
||||||
#ifdef ESP8266
|
|
||||||
rxPinSDS = jsonReadInt(parameters, "rxPin");
|
rxPinSDS = jsonReadInt(parameters, "rxPin");
|
||||||
txPinSDS = jsonReadInt(parameters, "txPin");
|
txPinSDS = jsonReadInt(parameters, "txPin");
|
||||||
#endif
|
|
||||||
warmUp = jsonReadInt(parameters, "warmUp"); // сек. пробужнение должен быть больше
|
purge = jsonReadInt(parameters, "purge"); // сек. пробужнение должен быть больше
|
||||||
period = jsonReadInt(parameters, "period"); // сек. время зарогрева/продувки, затем идут замеры
|
interval = jsonReadInt(parameters, "int"); // сек. время зарогрева/продувки, затем идут замеры
|
||||||
|
continuous = jsonReadInt(parameters, "continuousMode"); // сек. время зарогрева/продувки, затем идут замеры
|
||||||
|
if (continuous)
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Sds011", "Continuous mode");
|
||||||
|
ts_sds.remove(PRINT);
|
||||||
|
ts_sds.remove(WAKEUP);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Sds011", "WakeUp mode");
|
||||||
|
}
|
||||||
|
purgeDelay = interval - purge;
|
||||||
|
SDS011_init();
|
||||||
|
firstSensor = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
//луп выполняющий переодическое дерганье
|
||||||
|
void loop()
|
||||||
|
{
|
||||||
|
ts_sds.update();
|
||||||
|
if (enableDoByInt)
|
||||||
|
{
|
||||||
|
currentMillis = millis();
|
||||||
|
difference = currentMillis - prevMillis;
|
||||||
|
if (difference >= _interval)
|
||||||
|
{
|
||||||
|
prevMillis = millis();
|
||||||
|
this->doByInterval();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
//=======================================================================================================
|
//=======================================================================================================
|
||||||
// doByInterval()
|
// doByInterval()
|
||||||
@@ -117,12 +177,13 @@ class Sds011_10 : public IoTItem {
|
|||||||
//не используйте delay - помните, что данный loop общий для всех модулей. Если у вас планируется длительная операция, постарайтесь разбить ее на порции
|
//не используйте delay - помните, что данный loop общий для всех модулей. Если у вас планируется длительная операция, постарайтесь разбить ее на порции
|
||||||
//и выполнить за несколько тактов
|
//и выполнить за несколько тактов
|
||||||
|
|
||||||
void doByInterval() {
|
void doByInterval()
|
||||||
|
{
|
||||||
SDS011_init();
|
SDS011_init();
|
||||||
Serial.println("request from 10");
|
// SerialPrint("i", "Sensor Sds011", "Request for 10ppm");
|
||||||
value.valD = Sds011request(10);
|
Sds011request(10);
|
||||||
|
// value.valD = Sds011request(10);
|
||||||
regEvent(value.valD, "Sds011_10"); //обязательный вызов хотяб один
|
// regEvent(value.valD, "Sds011_10"); //обязательный вызов хотяб один
|
||||||
}
|
}
|
||||||
~Sds011_10(){};
|
~Sds011_10(){};
|
||||||
};
|
};
|
||||||
@@ -130,71 +191,148 @@ class Sds011_10 : public IoTItem {
|
|||||||
//после замены названия сенсора, на функцию можно не обращать внимания
|
//после замены названия сенсора, на функцию можно не обращать внимания
|
||||||
//если сенсор предполагает использование общего объекта библиотеки для нескольких экземпляров сенсора, то в данной функции необходимо предусмотреть
|
//если сенсор предполагает использование общего объекта библиотеки для нескольких экземпляров сенсора, то в данной функции необходимо предусмотреть
|
||||||
//создание и контроль соответствующих глобальных переменных
|
//создание и контроль соответствующих глобальных переменных
|
||||||
void* getAPI_Sds011(String subtype, String param) {
|
void *getAPI_Sds011(String subtype, String param)
|
||||||
if (subtype == F("Sds011_25")) {
|
{
|
||||||
|
if (subtype == F("Sds011_25"))
|
||||||
|
{
|
||||||
return new Sds011_25(param);
|
return new Sds011_25(param);
|
||||||
} else if (subtype == F("Sds011_10")) {
|
}
|
||||||
|
else if (subtype == F("Sds011_10"))
|
||||||
|
{
|
||||||
return new Sds011_10(param);
|
return new Sds011_10(param);
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
float Sds011request(int sensorID) {
|
void Sds011request(int sensorID)
|
||||||
float reply = 0;
|
{
|
||||||
static int a = 0;
|
float pm25 = 0;
|
||||||
static float pm25 = 0;
|
float pm10 = 0;
|
||||||
static float pm10 = 0;
|
|
||||||
static int startMillis = millis();
|
|
||||||
|
|
||||||
if (a == 0) {
|
if (firstSensor == 0)
|
||||||
Serial.print("SDS011 ... warmUp = ");
|
firstSensor = sensorID;
|
||||||
Serial.print(warmUp);
|
|
||||||
Serial.print(" period = ");
|
if (firstSensor == sensorID)
|
||||||
Serial.println(period);
|
{
|
||||||
sds.wakeup();
|
if (!continuous)
|
||||||
startMillis = millis();
|
{
|
||||||
a = a + 1;
|
ts_sds.remove(PRINT);
|
||||||
|
ts_sds.remove(WAKEUP);
|
||||||
}
|
}
|
||||||
if (a == 1 && millis() >= (startMillis + warmUp * 1000)) {
|
|
||||||
PmResult pm = sds.readPm();
|
PmResult pm = sds->readPm();
|
||||||
if (pm.isOk()) {
|
// SerialPrint("i", "Sensor Sds011", "sds.readPm()");
|
||||||
|
if (pm.isOk())
|
||||||
|
{
|
||||||
pm25 = pm.pm25;
|
pm25 = pm.pm25;
|
||||||
pm10 = pm.pm10;
|
pm10 = pm.pm10;
|
||||||
Serial.print("PM2.5 = ");
|
// SerialPrint("i", "Sensor Sds011", pm.toString());
|
||||||
Serial.print(pm25);
|
|
||||||
Serial.print(" PM10 = ");
|
if (item_Sds011_25 && pm25)
|
||||||
Serial.println(pm10);
|
{
|
||||||
a = a + 1;
|
item_Sds011_25->value.valD = pm25;
|
||||||
sds.sleep();
|
item_Sds011_25->regEvent(item_Sds011_25->value.valD, "Sds011_25");
|
||||||
} else {
|
|
||||||
Serial.print("Could not read values from sensor 25, reason: ");
|
|
||||||
Serial.println(pm.statusToString());
|
|
||||||
a = a + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (a > 1 && millis() >= (startMillis + period * 1000)) {
|
|
||||||
Serial.println("end of period for pm25");
|
|
||||||
a = 0;
|
|
||||||
}
|
|
||||||
if (sensorID == 25) {
|
|
||||||
reply = pm25;
|
|
||||||
}
|
|
||||||
if (sensorID == 10) {
|
|
||||||
reply = pm10;
|
|
||||||
}
|
|
||||||
return reply;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SDS011_init() {
|
if (item_Sds011_10 && pm10)
|
||||||
if (SDS011_init_flag) {
|
{
|
||||||
sds.begin();
|
item_Sds011_10->value.valD = pm10;
|
||||||
Serial.println(sds.queryFirmwareVersion().toString()); // prints firmware version
|
item_Sds011_10->regEvent(item_Sds011_10->value.valD, "Sds011_10");
|
||||||
// Serial.println(sds.setActiveReportingMode().toString()); //
|
}
|
||||||
String ReportingMode = sds.setActiveReportingMode().toString();
|
}
|
||||||
Serial.println(ReportingMode);
|
else
|
||||||
if (ReportingMode == "Mode: active") {
|
{
|
||||||
|
String msg = "Could not read values from sensor. Reason: " + pm.statusToString();
|
||||||
|
SerialPrint("E", "Sensor Sds011", msg);
|
||||||
|
SDS011_init_flag = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!continuous)
|
||||||
|
{
|
||||||
|
sds->sleep();
|
||||||
|
SerialPrint("i", "Sensor Sds011", "sleep");
|
||||||
|
String msg = "wakeUp planned in " + String(purgeDelay) + " seconds";
|
||||||
|
SerialPrint("i", "Sensor Sds011", msg);
|
||||||
|
ts_sds.add(
|
||||||
|
PRINT, purgeDelay * 1000, [](void *)
|
||||||
|
{ SerialPrint("i", "Sensor Sds011", "delayed wakeUp"); },
|
||||||
|
nullptr, false);
|
||||||
|
ts_sds.add(
|
||||||
|
WAKEUP, purgeDelay * 1000, [](void *)
|
||||||
|
{ sds->wakeup(); },
|
||||||
|
nullptr, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
//пропускаем вызов от второго сенсора
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SDS011_init()
|
||||||
|
{
|
||||||
|
if (SDS011_init_flag)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef ESP8266
|
||||||
|
|
||||||
|
if (!sds)
|
||||||
|
{
|
||||||
|
Serial.println("no sds, creating");
|
||||||
|
sds = new SdsDustSensor(rxPinSDS, txPinSDS, retryDelayMs, maxRetriesNotAvailable);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial.println("sds already created");
|
||||||
|
}
|
||||||
|
|
||||||
|
sds->begin(9600);
|
||||||
|
delay(200); //
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef ESP32
|
||||||
|
sdsSerial.begin(9600, SERIAL_8N1, rxPinSDS, txPinSDS);
|
||||||
|
delay(200);
|
||||||
|
if (!sds)
|
||||||
|
{
|
||||||
|
Serial.println("no sds, creating");
|
||||||
|
sds = new SdsDustSensor(sdsSerial, retryDelayMs, maxRetriesNotAvailable);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial.println("sds already created");
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (startUp)
|
||||||
|
{
|
||||||
|
sds->wakeup();
|
||||||
|
SerialPrint("i", "Sensor Sds011", "wakeup on startUp"); // уже начинаем продувку
|
||||||
|
startUp = false;
|
||||||
|
}
|
||||||
|
String msg = sds->queryFirmwareVersion().toString(); //
|
||||||
|
SerialPrint("i", "Sensor Sds011", msg);
|
||||||
|
String ReportingMode = sds->setActiveReportingMode().toString();
|
||||||
|
|
||||||
|
if (ReportingMode == "Mode: active")
|
||||||
|
{
|
||||||
|
SerialPrint("i", "Sensor Sds011", ReportingMode);
|
||||||
SDS011_init_flag = false;
|
SDS011_init_flag = false;
|
||||||
|
if (continuous)
|
||||||
|
{
|
||||||
|
sds->wakeup();
|
||||||
|
SerialPrint("i", "Sensor Sds011", "wakeUp if continuous");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ReportingMode += " - check wiring!";
|
||||||
|
SerialPrint("E", "Sensor Sds011", ReportingMode);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"menuSection": "Сенсоры",
|
"menuSection": "Сенсоры",
|
||||||
"configItem": [
|
"configItem": [
|
||||||
{
|
{
|
||||||
"name": "SDS011 PM25 Датчик пыли",
|
"name": "SDS011 PM25 Пыль",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Sds011_25",
|
"subtype": "Sds011_25",
|
||||||
"id": "pmuart25",
|
"id": "pmuart25",
|
||||||
@@ -12,14 +12,16 @@
|
|||||||
"plus": 0,
|
"plus": 0,
|
||||||
"multiply": 1,
|
"multiply": 1,
|
||||||
"round": 10,
|
"round": 10,
|
||||||
"rxPin": 13,
|
"rxPin": 14,
|
||||||
"txPin": 12,
|
"txPin": 16,
|
||||||
"int": 15,
|
"int": 270,
|
||||||
"warmUp": 30,
|
"purge": 30,
|
||||||
"period": 300
|
"continuousMode": 0,
|
||||||
|
"maxRetriesNotAvailable": 100,
|
||||||
|
"retryDelayMs": 5
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "SDS011 PM10 Датчик пыли",
|
"name": "SDS011 PM10 Пыль",
|
||||||
"type": "Reading",
|
"type": "Reading",
|
||||||
"subtype": "Sds011_10",
|
"subtype": "Sds011_10",
|
||||||
"id": "pmuart10",
|
"id": "pmuart10",
|
||||||
@@ -29,33 +31,34 @@
|
|||||||
"plus": 0,
|
"plus": 0,
|
||||||
"multiply": 1,
|
"multiply": 1,
|
||||||
"round": 10,
|
"round": 10,
|
||||||
"rxPin": 13,
|
"rxPin": 14,
|
||||||
"txPin": 12,
|
"txPin": 16,
|
||||||
"int": 15,
|
"int": 270,
|
||||||
"warmUp": 30,
|
"purge": 30,
|
||||||
"period": 300
|
"continuousMode": 0,
|
||||||
|
"maxRetriesNotAvailable": 100,
|
||||||
|
"retryDelayMs": 5
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"about": {
|
"about": {
|
||||||
"authorName": "Alex K",
|
"authorName": "Alex K",
|
||||||
"authorContact": "https://t.me/cmche",
|
"authorContact": "https://t.me/cmche",
|
||||||
"authorGit": "",
|
"authorGit": "https://github.com/CHE77/SDS011forIotManager",
|
||||||
"specialThanks": "",
|
"specialThanks": "",
|
||||||
"moduleName": "Sds011",
|
"moduleName": "Sds011",
|
||||||
"moduleVersion": "1.0",
|
"moduleVersion": "2.0 - можно переназначать пины, за один опрос - обновляются два элемента",
|
||||||
"usedRam": 15,
|
|
||||||
"subTypes": [
|
|
||||||
"Sds011_25",
|
|
||||||
"Sds011_10"
|
|
||||||
],
|
|
||||||
"title": "Датчик пыли",
|
|
||||||
"moduleDesc": "Позволяет получить значения концентрации пыли в воздухе с Sds011.",
|
"moduleDesc": "Позволяет получить значения концентрации пыли в воздухе с Sds011.",
|
||||||
"propInfo": {
|
"propInfo": {
|
||||||
"int": "Количество секунд между опросами датчика.",
|
"plus": "поправочный коэффиент +c",
|
||||||
"rxPin": "",
|
"multiply": "поправочный коэффиент k*",
|
||||||
"txPin": "",
|
"round": "округление",
|
||||||
"warmUp": "",
|
"rxPin": "Esp8266: GPIO 14 - D5, ESP32: GPIO 16 - RX2, > подключаем к TXD сенсора",
|
||||||
"period": ""
|
"txPin": "Esp8266: GPIO 16 - D0, ESP32: GPIO 17 - TX2, > подключаем к RXD сенсора",
|
||||||
|
"int": "Количество секунд между опросами датчика",
|
||||||
|
"purge": "Время продувки сенсора перед замером. Cек.",
|
||||||
|
"continuousMode": "1 - Непрерывный режим, 0 - Режим с остановкой (щедящий)",
|
||||||
|
"maxRetriesNotAvailable": "Количество попыток ожидания ответа сенсора при опросе (не нужно менять)",
|
||||||
|
"retryDelayMs": "Задержка между попытками, миллисекунды (не нужно менять)"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"defActive": false,
|
"defActive": false,
|
||||||
|
|||||||
@@ -25,8 +25,8 @@
|
|||||||
"subTypes": [
|
"subTypes": [
|
||||||
"SoftUART"
|
"SoftUART"
|
||||||
],
|
],
|
||||||
"title": "Софтовый uart для esp8266 или harware uart для esp32",
|
"title": "Software uart для esp8266 или hardware uart для esp32",
|
||||||
"moduleDesc": "Используется вместе с Pzem004t, в последствии будет доработан для связи с arduino платами",
|
"moduleDesc": "Используется вместе с Pzem004t или с другими работающими по uart сенсорами, в последствии будет доработан для связи с arduino платами",
|
||||||
"propInfo": {
|
"propInfo": {
|
||||||
"tx": "TX пин",
|
"tx": "TX пин",
|
||||||
"rx": "RX пин",
|
"rx": "RX пин",
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"menuSection": "Виртуальные элементы",
|
"menuSection": "Виртуальные элементы",
|
||||||
"configItem": [
|
"configItem": [
|
||||||
{
|
{
|
||||||
"name": "Логирование в график",
|
"name": "График",
|
||||||
"type": "Writing",
|
"type": "Writing",
|
||||||
"subtype": "Loging",
|
"subtype": "Loging",
|
||||||
"id": "log",
|
"id": "log",
|
||||||
@@ -24,7 +24,7 @@
|
|||||||
"moduleVersion": "3.0",
|
"moduleVersion": "3.0",
|
||||||
"usedRam": 15,
|
"usedRam": 15,
|
||||||
"title": "Логирование в график",
|
"title": "Логирование в график",
|
||||||
"moduleDesc": "Расширение позволяющее логировать любую величину в график. Графики доступны в мобильном приложении и в веб интерфейсе. Данные графиков хранятся в встроенной памяти esp. В окне ввода даты нужно выбрать день, историю которого вы хотите посмотреть. Старые файлы будут удаляться автоматически после того как объем оставшейся flesh памяти устройства будет менее 20 процентов",
|
"moduleDesc": "Расширение позволяющее логировать любую величину в график. Графики доступны в мобильном приложении и в веб интерфейсе. Данные графиков хранятся в встроенной памяти esp. В окне ввода даты можно выбирать день, историю которого вы хотите посмотреть. Старые файлы будут удаляться автоматически после того как объем оставшейся flesh памяти устройства будет менее 20 процентов",
|
||||||
"propInfo": {
|
"propInfo": {
|
||||||
"int": "Интервал логирования в мнутах, рекомендуется для esp8266 использоать интервал не менее 5-ти минут",
|
"int": "Интервал логирования в мнутах, рекомендуется для esp8266 использоать интервал не менее 5-ти минут",
|
||||||
"logid": "ID величины которую будем логировать",
|
"logid": "ID величины которую будем логировать",
|
||||||
282
src/modules/virtual/LogingDaily/LogingDaily.cpp
Normal file
282
src/modules/virtual/LogingDaily/LogingDaily.cpp
Normal file
@@ -0,0 +1,282 @@
|
|||||||
|
#include "Global.h"
|
||||||
|
#include "classes/IoTItem.h"
|
||||||
|
#include "ESPConfiguration.h"
|
||||||
|
#include "NTP.h"
|
||||||
|
|
||||||
|
class LogingDaily : public IoTItem {
|
||||||
|
private:
|
||||||
|
String logid;
|
||||||
|
String id;
|
||||||
|
String filesList = "";
|
||||||
|
|
||||||
|
int _publishType = -2;
|
||||||
|
int _wsNum = -1;
|
||||||
|
|
||||||
|
int points;
|
||||||
|
|
||||||
|
IoTItem *dateIoTItem;
|
||||||
|
|
||||||
|
String prevDate = "";
|
||||||
|
bool firstTimeDate = true;
|
||||||
|
|
||||||
|
unsigned long interval;
|
||||||
|
|
||||||
|
public:
|
||||||
|
LogingDaily(String parameters) : IoTItem(parameters) {
|
||||||
|
jsonRead(parameters, F("logid"), logid);
|
||||||
|
jsonRead(parameters, F("id"), id);
|
||||||
|
jsonRead(parameters, F("points"), points);
|
||||||
|
|
||||||
|
if (points > 365) {
|
||||||
|
points = 365;
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' user set more points than allowed, value reset to 365");
|
||||||
|
}
|
||||||
|
jsonRead(parameters, F("int"), interval);
|
||||||
|
interval = interval * 1000 * 60; //приводим к милисекундам
|
||||||
|
}
|
||||||
|
|
||||||
|
void doByInterval() {
|
||||||
|
if (hasDayChanged()) {
|
||||||
|
execute();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void execute() {
|
||||||
|
//если объект логгирования не был создан
|
||||||
|
if (!isItemExist(logid)) {
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' LogingDaily object not exist, return");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String value = getItemValue(logid);
|
||||||
|
|
||||||
|
//если значение логгирования пустое
|
||||||
|
if (value == "") {
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' LogingDaily value is empty, return");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//если время не было получено из интернета
|
||||||
|
if (!isTimeSynch) {
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' Сant LogingDaily - time not synchronized, return");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String logData;
|
||||||
|
|
||||||
|
float currentValue = value.toFloat();
|
||||||
|
//прочитаем предудущее значение
|
||||||
|
float prevValue = readDataDB(id + "-v").toFloat();
|
||||||
|
//сохраним в базу данных текущее значение, понадобится в следующие сутки
|
||||||
|
saveDataDB(id + "-v", value);
|
||||||
|
|
||||||
|
float difference = currentValue - prevValue;
|
||||||
|
|
||||||
|
jsonWriteInt(logData, "x", unixTime);
|
||||||
|
jsonWriteFloat(logData, "y1", difference);
|
||||||
|
|
||||||
|
//прочитаем путь к файлу последнего сохранения
|
||||||
|
String filePath = readDataDB(id);
|
||||||
|
|
||||||
|
//если данные о файле отсутствуют, создадим новый
|
||||||
|
if (filePath == "failed" || filePath == "") {
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' file path not found, start create new file");
|
||||||
|
createNewFileWithData(logData);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//считаем количество строк и определяем размер файла
|
||||||
|
size_t size = 0;
|
||||||
|
int lines = countJsonObj(filePath, size);
|
||||||
|
SerialPrint("i", F("LogingDaily"), "'" + id + "' " + "lines = " + String(lines) + ", size = " + String(size));
|
||||||
|
|
||||||
|
//если количество строк до заданной величины и дата не менялась
|
||||||
|
if (lines <= points && !hasDayChanged()) {
|
||||||
|
//просто добавим в существующий файл новые данные
|
||||||
|
addNewDataToExistingFile(filePath, logData);
|
||||||
|
//если больше или поменялась дата то создадим следующий файл
|
||||||
|
} else {
|
||||||
|
createNewFileWithData(logData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void createNewFileWithData(String &logData) {
|
||||||
|
logData = logData + ",";
|
||||||
|
|
||||||
|
String path = "/lgd/" + id + "/" + id + ".txt"; //создадим путь вида /lgd/id/id.txt
|
||||||
|
//создадим пустой файл
|
||||||
|
if (writeEmptyFile(path) != "sucсess") {
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' file writing error, return");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//запишем в него данные
|
||||||
|
if (addFile(path, logData) != "sucсess") {
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' data writing error, return");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
//запишем путь к нему в базу данных
|
||||||
|
if (saveDataDB(id, path) != "sucсess") {
|
||||||
|
SerialPrint("E", F("LogingDaily"), "'" + id + "' db file writing error, return");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
SerialPrint("i", F("LogingDaily"), "'" + id + "' file created http://" + WiFi.localIP().toString() + path);
|
||||||
|
}
|
||||||
|
|
||||||
|
void addNewDataToExistingFile(String &path, String &logData) {
|
||||||
|
logData = logData + ",";
|
||||||
|
if (addFile(path, logData) != "sucсess") {
|
||||||
|
SerialPrint("i", F("LogingDaily"), "'" + id + "' file writing error, return");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
SerialPrint("i", F("LogingDaily"), "'" + id + "' LogingDaily in file http://" + WiFi.localIP().toString() + path);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool hasDayChanged() {
|
||||||
|
bool changed = false;
|
||||||
|
String currentDate = getTodayDateDotFormated();
|
||||||
|
if (!firstTimeDate) {
|
||||||
|
if (prevDate != currentDate) {
|
||||||
|
changed = true;
|
||||||
|
SerialPrint("i", F("NTP"), "Change day event");
|
||||||
|
#if defined(ESP8266)
|
||||||
|
FileFS.gc();
|
||||||
|
#endif
|
||||||
|
#if defined(ESP32)
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
firstTimeDate = false;
|
||||||
|
prevDate = currentDate;
|
||||||
|
return changed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void publishValue() {
|
||||||
|
String dir = "/lgd/" + id;
|
||||||
|
filesList = getFilesList(dir);
|
||||||
|
|
||||||
|
SerialPrint("i", F("LogingDaily"), "file list: " + filesList);
|
||||||
|
|
||||||
|
int f = 0;
|
||||||
|
|
||||||
|
while (filesList.length()) {
|
||||||
|
String path = selectToMarker(filesList, ";");
|
||||||
|
|
||||||
|
path = "/lgd/" + id + path;
|
||||||
|
|
||||||
|
f++;
|
||||||
|
|
||||||
|
if (_publishType == TO_MQTT) {
|
||||||
|
publishChartFileToMqtt(path);
|
||||||
|
} else if (_publishType == TO_WS) {
|
||||||
|
publishChartToWs(path, _wsNum, 1000);
|
||||||
|
} else if (_publishType == TO_MQTT_WS) {
|
||||||
|
publishChartFileToMqtt(path);
|
||||||
|
publishChartToWs(path, _wsNum, 1000);
|
||||||
|
}
|
||||||
|
SerialPrint("i", F("LogingDaily"), String(f) + ") " + path + ", sent");
|
||||||
|
|
||||||
|
filesList = deleteBeforeDelimiter(filesList, ";");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void clearHistory() {
|
||||||
|
String dir = "/lgd/" + id;
|
||||||
|
cleanDirectory(dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool publishChartFileToMqtt(String path) {
|
||||||
|
File configFile = FileFS.open(path, FILE_READ);
|
||||||
|
if (!configFile) {
|
||||||
|
SerialPrint("E", F("LogingDaily"), path + " file reading error, json not created, return");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
String oneSingleJson = configFile.readString();
|
||||||
|
configFile.close();
|
||||||
|
String topic = mqttRootDevice + "/" + id;
|
||||||
|
oneSingleJson = "{\"maxCount\":" + String(calculateMaxCount()) + ",\"topic\":\"" + topic + "\",\"status\":[" + oneSingleJson + "]}";
|
||||||
|
oneSingleJson.replace("},]}", "}]}");
|
||||||
|
SerialPrint("i", "LogingDaily", "json size: " + String(oneSingleJson.length()));
|
||||||
|
publishChartMqtt(id, oneSingleJson);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//особая функция отправки графиков в веб
|
||||||
|
void publishChartToWs(String filename, int num, size_t frameSize) {
|
||||||
|
String json;
|
||||||
|
jsonWriteStr(json, "topic", mqttRootDevice + "/" + id);
|
||||||
|
jsonWriteInt(json, "maxCount", calculateMaxCount());
|
||||||
|
|
||||||
|
String st = "/st/chart.json|";
|
||||||
|
if (num == -1) {
|
||||||
|
standWebSocket.broadcastTXT(st);
|
||||||
|
} else {
|
||||||
|
standWebSocket.sendTXT(num, st);
|
||||||
|
}
|
||||||
|
String path = filepath(filename);
|
||||||
|
auto file = FileFS.open(path, "r");
|
||||||
|
if (!file) {
|
||||||
|
SerialPrint(F("E"), F("FS"), F("reed file error"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
size_t fileSize = file.size();
|
||||||
|
SerialPrint(F("i"), F("FS"), "Send file '" + String(filename) + "', file size: " + String(fileSize));
|
||||||
|
uint8_t payload[frameSize];
|
||||||
|
int countRead = file.read(payload, sizeof(payload));
|
||||||
|
while (countRead > 0) {
|
||||||
|
if (num == -1) {
|
||||||
|
standWebSocket.broadcastBIN(payload, countRead);
|
||||||
|
} else {
|
||||||
|
standWebSocket.sendBIN(num, payload, countRead);
|
||||||
|
}
|
||||||
|
countRead = file.read(payload, sizeof(payload));
|
||||||
|
}
|
||||||
|
file.close();
|
||||||
|
String end = "/end/chart.json|" + json;
|
||||||
|
if (num == -1) {
|
||||||
|
standWebSocket.broadcastTXT(end);
|
||||||
|
} else {
|
||||||
|
standWebSocket.sendTXT(num, end);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void publishChartToWsSinglePoint(String value) {
|
||||||
|
String topic = mqttRootDevice + "/" + id;
|
||||||
|
String json = "{\"maxCount\":" + String(calculateMaxCount()) + ",\"topic\":\"" + topic + "\",\"status\":[{\"x\":" + String(unixTime) + ",\"y1\":" + value + "}]}";
|
||||||
|
String pk = "/string/chart.json|" + json;
|
||||||
|
standWebSocket.broadcastTXT(pk);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setPublishDestination(int publishType, int wsNum = -1) {
|
||||||
|
_publishType = publishType;
|
||||||
|
_wsNum = wsNum;
|
||||||
|
}
|
||||||
|
|
||||||
|
String getValue() {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop() {
|
||||||
|
if (enableDoByInt) {
|
||||||
|
currentMillis = millis();
|
||||||
|
difference = currentMillis - prevMillis;
|
||||||
|
if (difference >= interval) {
|
||||||
|
prevMillis = millis();
|
||||||
|
this->doByInterval();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//просто максимальное количество точек
|
||||||
|
int calculateMaxCount() {
|
||||||
|
return 86400;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void *getAPI_LogingDaily(String subtype, String param) {
|
||||||
|
if (subtype == F("LogingDaily")) {
|
||||||
|
return new LogingDaily(param);
|
||||||
|
} else {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
39
src/modules/virtual/LogingDaily/modinfo.json
Normal file
39
src/modules/virtual/LogingDaily/modinfo.json
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
{
|
||||||
|
"menuSection": "Виртуальные элементы",
|
||||||
|
"configItem": [
|
||||||
|
{
|
||||||
|
"name": "График дневного расхода",
|
||||||
|
"type": "Writing",
|
||||||
|
"subtype": "LogingDaily",
|
||||||
|
"id": "log",
|
||||||
|
"widget": "chart3",
|
||||||
|
"page": "Графики",
|
||||||
|
"descr": "Температура",
|
||||||
|
"num": 1,
|
||||||
|
"int": 1,
|
||||||
|
"logid": "t",
|
||||||
|
"points": 365
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"about": {
|
||||||
|
"authorName": "Dmitry Borisenko",
|
||||||
|
"authorContact": "https://t.me/Dmitry_Borisenko",
|
||||||
|
"authorGit": "https://github.com/DmitryBorisenko33",
|
||||||
|
"specialThanks": "@itsid1 @Valiuhaaa Serg",
|
||||||
|
"moduleName": "LogingDaily",
|
||||||
|
"moduleVersion": "3.0",
|
||||||
|
"usedRam": 15,
|
||||||
|
"title": "График дневного расхода",
|
||||||
|
"moduleDesc": "Расширение позволяющее логировать накопительные величины и видеть их дневное изменение. Графики доступны в мобильном приложении и в веб интерфейсе. Данные графиков хранятся в встроенной памяти esp",
|
||||||
|
"propInfo": {
|
||||||
|
"int": "Интервал логирования в мнутах, частота проверки смены суток в минутах. Не рекомендуется менять",
|
||||||
|
"logid": "ID накопительной величины которую будем логировать",
|
||||||
|
"points": "Максимальное количество точек"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"defActive": true,
|
||||||
|
"devices": {
|
||||||
|
"esp32_4mb": [],
|
||||||
|
"esp8266_4mb": []
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -214,9 +214,10 @@ String readDataDB(String id) {
|
|||||||
void cleanLogs() {
|
void cleanLogs() {
|
||||||
SerialPrint("i", "Files", "cleanLogs");
|
SerialPrint("i", "Files", "cleanLogs");
|
||||||
cleanDirectory("/db");
|
cleanDirectory("/db");
|
||||||
|
//обращение к логированию из ядра
|
||||||
//очистка данных всех экземпляров графиков
|
//очистка данных всех экземпляров графиков
|
||||||
for (std::list<IoTItem*>::iterator it = IoTItems.begin(); it != IoTItems.end(); ++it) {
|
for (std::list<IoTItem*>::iterator it = IoTItems.begin(); it != IoTItems.end(); ++it) {
|
||||||
if ((*it)->getSubtype() == "Loging") {
|
if ((*it)->getSubtype() == "Loging" || "LogingDaily") {
|
||||||
(*it)->clearHistory();
|
(*it)->clearHistory();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user