mirror of
				https://github.com/AlexGyver/GyverLamp2.git
				synced 2025-10-25 05:40:53 +03:00 
			
		
		
		
	Merge branch 'main' into platformio
This commit is contained in:
		
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,3 +1,12 @@ | ||||
| Отправляем на адрес x.x.x.255, первые 3 октета - адрес сети, к которой подключен смартфон | ||||
| UDP пакет вида <ключ>,<канал>,<тип>,<дата1>,<дата2>... разделитель - запятая | ||||
| Ключ зашит в прошивке и задаётся в приложении (защита от управления "чужаком") | ||||
|  | ||||
| Порт UDP формируется из имени сети: | ||||
| GLkey = "ключ" | ||||
| portNum = 17;		// uint16_t (или % 65536) | ||||
| for (byte i = 0; i < длина ключа; i++) portNum *= GLkey[i]; | ||||
| portNum %= 15000; | ||||
| portNum += 50000; | ||||
| portNum += номер группы | ||||
| Таким образом порт лежит в диапазоне 50 001... 65 010 | ||||
|  | ||||
| UDP пакет вида <GL>,<тип>,<дата1>,<дата2>... разделитель - запятая | ||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										80
									
								
								firmware/GyverLamp2/0_func.ino
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								firmware/GyverLamp2/0_func.ino
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | ||||
| void sendUDP(char *data) { | ||||
|   Udp.beginPacket(deviceIP, portNum + cfg.group); | ||||
|   Udp.write(data); | ||||
|   Udp.endPacket(); | ||||
| } | ||||
| void sendUDP(byte cmd, int data1 = 0, int data2 = 0, int data3 = 0) { | ||||
|   char reply[20] = ""; | ||||
|   mString packet(reply, sizeof(reply)); | ||||
|   packet = packet + "GL," + cmd + ',' + data1 + ',' + data2 + ',' + data3; | ||||
|   sendUDP(reply); | ||||
|   //DEBUG("Sending: "); | ||||
|   //DEBUGLN(cmd); | ||||
| } | ||||
|  | ||||
| void restartUDP() { | ||||
|   Udp.stop(); | ||||
|   Udp.begin(portNum + cfg.group); | ||||
|   deviceIP = WiFi.localIP(); | ||||
|   deviceIP[3] = 255; | ||||
|   DEBUG("UDP port: "); | ||||
|   DEBUGLN(portNum + cfg.group); | ||||
| } | ||||
|  | ||||
| void blink16(CRGB color) { | ||||
|   FOR_i(0, 3) { | ||||
|     fill_solid(leds, 16, color); | ||||
|     FastLED.show(); | ||||
|     delay(300); | ||||
|     FastLED.clear(); | ||||
|     FastLED.show(); | ||||
|     delay(300); | ||||
|   } | ||||
| } | ||||
|  | ||||
| const uint8_t font5x7[][5] = { | ||||
|   {0x3e, 0x51, 0x49, 0x45, 0x3e}, // 0 0x30 48 | ||||
|   {0x00, 0x42, 0x7f, 0x40, 0x00}, // 1 0x31 49 | ||||
|   {0x42, 0x61, 0x51, 0x49, 0x46}, // 2 0x32 50 | ||||
|   {0x21, 0x41, 0x45, 0x4b, 0x31}, // 3 0x33 51 | ||||
|   {0x18, 0x14, 0x12, 0x7f, 0x10}, // 4 0x34 52 | ||||
|   {0x27, 0x45, 0x45, 0x45, 0x39}, // 5 0x35 53 | ||||
|   {0x3c, 0x4a, 0x49, 0x49, 0x30}, // 6 0x36 54 | ||||
|   {0x01, 0x71, 0x09, 0x05, 0x03}, // 7 0x37 55 | ||||
|   {0x36, 0x49, 0x49, 0x49, 0x36}, // 8 0x38 56 | ||||
|   {0x06, 0x49, 0x49, 0x29, 0x1e}, // 9 0x39 57 | ||||
|   {0x00, 0x08, 0x08, 0x08, 0x00}, // 10 - | ||||
|   {0x00, 0x00, 0x00, 0x00, 0x00}, // 11 empty | ||||
| }; | ||||
|  | ||||
| void drawDigit(byte digit, int X, int Y, CRGB color) { | ||||
|   FOR_i(0, 5) { | ||||
|     FOR_j(0, 7) { | ||||
|       if (font5x7[digit][i] & (1 << 6 - j)) setPix(i + X, j + Y, color); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| void drawDots(int X, int Y, CRGB color) { | ||||
|   setPix(X, Y + 2, color); | ||||
|   setPix(X, Y + 4, color); | ||||
| } | ||||
|  | ||||
| void drawClock(byte Y, byte speed, CRGB color) { | ||||
|   if (cfg.deviceType == 1 || cfg.width < 16) return;   // лента или мелкая матрица - на выход | ||||
|   byte h1, h2, m1, m2; | ||||
|   if (gotNTP || gotTime) { | ||||
|     h1 = now.hour / 10; | ||||
|     if (h1 == 0) h1 = 11; | ||||
|     h2 = now.hour % 10; | ||||
|     m1 = now.min / 10; | ||||
|     m2 = now.min % 10; | ||||
|   } else { | ||||
|     h1 = h2 = m1 = m2 = 10; | ||||
|   } | ||||
|   int pos = cfg.width - (now.weekMs / (speed * 2)) % (cfg.width + 26); | ||||
|   drawDigit(h1, pos, Y, color); | ||||
|   drawDigit(h2, pos + 6, Y, color); | ||||
|   if (now.getMs() < 500) drawDots(pos + 12, Y, color); | ||||
|   drawDigit(m1, pos + 14, Y, color); | ||||
|   drawDigit(m2, pos + 20, Y, color); | ||||
| } | ||||
| @@ -8,10 +8,10 @@ | ||||
|  | ||||
| class FastFilter { | ||||
|   public: | ||||
| 	FastFilter(byte k = 20, int dt = 0) { | ||||
| 		setK(k); | ||||
| 		setDt(dt); | ||||
| 	} | ||||
|     FastFilter(byte k = 20, int dt = 0) { | ||||
|       setK(k); | ||||
|       setDt(dt); | ||||
|     } | ||||
|     void setK(byte k) { | ||||
|       _k1 = k; | ||||
|       _k2 = 32 - k; | ||||
|   | ||||
| @@ -1,4 +1,22 @@ | ||||
| /* | ||||
|   Версия 0.19b | ||||
|   Минимальная версия приложения 1.17!!! | ||||
|   Почищен мусор, оптимизация, повышена стабильность и производительность | ||||
|   Мигает теперь 16 светиков | ||||
|   Снова переделана сетевая политика, упрощён и сильно ускорен парсинг | ||||
|   Изменены пределы по светодиодам, что сильно увеличило производительность | ||||
|   Выключенная (программно) лампа не принимает сервисные команды кроме команды включиться | ||||
|   Добавлены часы, в том числе в рассвет | ||||
|   Slave работает со светомузыкой сам, если не получает данные с мастера | ||||
|  | ||||
|   Версия 0.18b | ||||
|   Уменьшена чувствительность хлопков | ||||
|   Увеличена плавность светомузыки | ||||
|   Переделана сетевая политика | ||||
|   Микрофон и датчик света опрашивает только мастер и отсылает данные слейвам своей группы | ||||
|   4 клика - включить первый режим | ||||
|   Отправка точного времени на лампу в режиме АР для работы рассвета и синхронизации эффектов | ||||
|  | ||||
|   Версия 0.17b | ||||
|   Автосмена отключается 30 сек во время настройки режимов | ||||
|   Убрана кнопка upload в режимах | ||||
| @@ -7,17 +25,17 @@ | ||||
|   Вкл выкл двумя хлопками | ||||
|   Плавное выключение | ||||
|   Починил рассвет | ||||
|    | ||||
|  | ||||
|   Версия 0.16b | ||||
|   Исправлен масштаб огня 2020 | ||||
|   Фикс невыключения рассвета | ||||
|    | ||||
|  | ||||
|   Версия 0.14b | ||||
|   Мелкие баги | ||||
|   Вернул искры огню | ||||
|   Добавлены палитры | ||||
|   Добавлен огонь 2020 | ||||
|    | ||||
|  | ||||
|   Версия 0.13b | ||||
|   Улучшена стабильность | ||||
|  | ||||
| @@ -36,14 +54,11 @@ | ||||
|   Выключение по таймеру теперь плавное | ||||
|   Добавлен рассвет | ||||
|  | ||||
|   TODO:   | ||||
|   TODO: | ||||
|   плавная смена режимов | ||||
|   4 клика вкл выкл смену? | ||||
|   Mqtt? | ||||
|   Базовый пак | ||||
|   Предложения Серёги крутского | ||||
|   Эффект погода https://it4it.club/topic/40-esp8266-i-parsing-pogodyi-s-openweathermap/ | ||||
|   Эффект часы | ||||
| */ | ||||
|  | ||||
| // ВНИМАНИЕ! ВНИМАНИЕ! ВНИМАНИЕ! ВНИМАНИЕ! ВНИМАНИЕ! ВНИМАНИЕ! ВНИМАНИЕ! | ||||
| @@ -69,7 +84,7 @@ | ||||
|  | ||||
| // ------------ Лента ------------- | ||||
| #define STRIP_PIN 2         // пин ленты GPIO2 (D4 на wemos/node) | ||||
| #define MAX_LEDS 600        // макс. светодиодов | ||||
| #define MAX_LEDS 300        // макс. светодиодов | ||||
| #define STRIP_CHIP WS2812   // чип ленты | ||||
| #define STRIP_COLOR GRB     // порядок цветов в ленте | ||||
| #define STRIP_VOLT 5        // напряжение ленты, V | ||||
| @@ -86,15 +101,15 @@ const char AP_NameChar[] = "GyverLamp2"; | ||||
| const char WiFiPassword[] = "12345678"; | ||||
|  | ||||
| // ------------ Прочее ------------- | ||||
| #define GL_VERSION 017      // код версии прошивки | ||||
| #define GL_VERSION 19       // код версии прошивки | ||||
| #define EE_TOUT 30000       // таймаут сохранения епром после изменения, мс | ||||
| //#define DEBUG_SERIAL_LAMP // закомментируй чтобы выключить отладку (скорость 115200) | ||||
| #define DEBUG_SERIAL_LAMP   // закомментируй чтобы выключить отладку (скорость 115200) | ||||
| #define EE_KEY 55           // ключ сброса WiFi (измени для сброса всех настроек) | ||||
| #define NTP_UPD_PRD 5       // период обновления времени с NTP сервера, минут | ||||
| //#define SKIP_WIFI         // пропустить подключение к вафле (для отладки) | ||||
|  | ||||
| // ------------ БИЛДЕР ------------- | ||||
| //#define MAX_LEDS 1200 | ||||
| //#define MAX_LEDS 900 | ||||
|  | ||||
| // esp01 | ||||
| //#define BTN_PIN 0 | ||||
| @@ -105,6 +120,7 @@ const char WiFiPassword[] = "12345678"; | ||||
| //#define STRIP_PIN 5     // GPIO5 на gl module (D1 на wemos/node) | ||||
|  | ||||
| // ---------- БИБЛИОТЕКИ ----------- | ||||
| #define FASTLED_ALLOW_INTERRUPTS 0 | ||||
| #include "data.h"         // данные | ||||
| #include "Time.h"         // часы | ||||
| #include "TimeRandom.h"   // случайные числа по времени | ||||
| @@ -115,7 +131,6 @@ const char WiFiPassword[] = "12345678"; | ||||
| #include "timerMillis.h"  // таймер миллис | ||||
| #include "VolAnalyzer.h"  // анализатор громкости | ||||
| #include "FFT_C.h"        // фурье | ||||
| #define FASTLED_ALLOW_INTERRUPTS 0 | ||||
| #include <FastLED.h>      // лента | ||||
| #include <ESP8266WiFi.h>  // базовая либа есп | ||||
| #include <WiFiUdp.h>      // общение по UDP | ||||
| @@ -132,31 +147,35 @@ Palette pal; | ||||
| WiFiServer server(80); | ||||
| WiFiUDP Udp; | ||||
| WiFiUDP ntpUDP; | ||||
| IPAddress deviceIP; | ||||
| NTPClient ntp(ntpUDP); | ||||
| CRGB leds[MAX_LEDS]; | ||||
| Time now; | ||||
| Button btn(BTN_PIN); | ||||
| timerMillis EEtmr(EE_TOUT), turnoffTmr, connTmr(120000ul), dawnTmr, holdPresTmr(30000ul), blinkTmr(300); | ||||
| timerMillis effTmr(30, true); | ||||
| TimeRandom trnd; | ||||
| VolAnalyzer vol(A0), low, high; | ||||
| FastFilter phot; | ||||
| Clap clap; | ||||
|  | ||||
| uint16_t portNum; | ||||
| uint32_t udpTmr = 0, gotADCtmr = 0; | ||||
| byte btnClicks = 0, brTicks = 0; | ||||
| unsigned char matrixValue[11][16]; | ||||
| bool gotNTP = false, gotTime = false; | ||||
| bool loading = true; | ||||
| void blink8(CRGB color); | ||||
| int udpLength = 0; | ||||
| byte udpScale = 0, udpBright = 0; | ||||
|  | ||||
| // ------------------- SETUP -------------------- | ||||
| void setup() { | ||||
|   misc(); | ||||
|   delay(2000);          // ждём старта есп | ||||
|   memset(matrixValue, 0, sizeof(matrixValue)); | ||||
| #ifdef DEBUG_SERIAL_LAMP | ||||
|   Serial.begin(115200); | ||||
|   DEBUGLN(); | ||||
| #endif | ||||
|   EEPROM.begin(1000);   // старт епром | ||||
|   startStrip();         // старт ленты | ||||
|   btn.setLevel(digitalRead(BTN_PIN));   // смотрим что за кнопка | ||||
|   EE_startup();         // читаем епром | ||||
| @@ -174,6 +193,7 @@ void setup() { | ||||
|  | ||||
| void loop() { | ||||
|   timeTicker();       // обновляем время | ||||
|   yield(); | ||||
| #ifndef SKIP_WIFI | ||||
|   tryReconnect();     // пробуем переподключиться если WiFi упал | ||||
|   yield(); | ||||
| @@ -186,4 +206,5 @@ void loop() { | ||||
|   yield(); | ||||
|   button();           // проверяем кнопку | ||||
|   checkAnalog();      // чтение звука и датчика | ||||
|   yield(); | ||||
| } | ||||
|   | ||||
| @@ -1,23 +1,23 @@ | ||||
| /** | ||||
|  * The MIT License (MIT) | ||||
|  * Copyright (c) 2015 by Fabrice Weinberg | ||||
|  * | ||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|  * of this software and associated documentation files (the "Software"), to deal | ||||
|  * in the Software without restriction, including without limitation the rights | ||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|  * copies of the Software, and to permit persons to whom the Software is | ||||
|  * furnished to do so, subject to the following conditions: | ||||
|  * The above copyright notice and this permission notice shall be included in all | ||||
|  * copies or substantial portions of the Software. | ||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|  * SOFTWARE. | ||||
|  */ | ||||
|    The MIT License (MIT) | ||||
|    Copyright (c) 2015 by Fabrice Weinberg | ||||
|  | ||||
|    Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
|    of this software and associated documentation files (the "Software"), to deal | ||||
|    in the Software without restriction, including without limitation the rights | ||||
|    to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
|    copies of the Software, and to permit persons to whom the Software is | ||||
|    furnished to do so, subject to the following conditions: | ||||
|    The above copyright notice and this permission notice shall be included in all | ||||
|    copies or substantial portions of the Software. | ||||
|    THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
|    IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
|    FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
|    AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
|    LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
|    OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
|    SOFTWARE. | ||||
| */ | ||||
|  | ||||
| #include "NTPClient-Gyver.h" | ||||
|  | ||||
| @@ -61,9 +61,9 @@ void NTPClient::begin(int port) { | ||||
| } | ||||
|  | ||||
| bool NTPClient::forceUpdate() { | ||||
|   #ifdef DEBUG_NTPClient | ||||
|     Serial.println("Update from NTP Server"); | ||||
|   #endif | ||||
| #ifdef DEBUG_NTPClient | ||||
|   Serial.println("Update from NTP Server"); | ||||
| #endif | ||||
|  | ||||
|   this->sendNTPPacket(); | ||||
|  | ||||
| @@ -83,12 +83,12 @@ bool NTPClient::forceUpdate() { | ||||
|  | ||||
|   unsigned long highWord = word(this->_packetBuffer[40], this->_packetBuffer[41]); | ||||
|   unsigned long lowWord = word(this->_packetBuffer[42], this->_packetBuffer[43]); | ||||
|    | ||||
|  | ||||
|   /// добавлено AlexGyver | ||||
|   uint32_t frac  = (uint32_t) _packetBuffer[44] << 24 | ||||
|                | (uint32_t) _packetBuffer[45] << 16 | ||||
|                | (uint32_t) _packetBuffer[46] <<  8 | ||||
|                | (uint32_t) _packetBuffer[47] <<  0; | ||||
|                    | (uint32_t) _packetBuffer[45] << 16 | ||||
|                    | (uint32_t) _packetBuffer[46] <<  8 | ||||
|                    | (uint32_t) _packetBuffer[47] <<  0; | ||||
|   uint16_t mssec = ((uint64_t) frac * 1000) >> 32; | ||||
|   //https://arduino.stackexchange.com/questions/49567/synching-local-clock-usign-ntp-to-milliseconds | ||||
|   _lastUpdate -= mssec; | ||||
| @@ -105,7 +105,7 @@ bool NTPClient::forceUpdate() { | ||||
|  | ||||
| bool NTPClient::update() { | ||||
|   if ((millis() - this->_lastUpdate >= this->_updateInterval)     // Update after _updateInterval | ||||
|     || this->_lastUpdate == 0) {                                // Update if there was no update yet. | ||||
|       || this->_lastUpdate == 0) {                                // Update if there was no update yet. | ||||
|     if (!this->_udpSetup) this->begin();                         // setup the UDP client if needed | ||||
|     return this->forceUpdate(); | ||||
|   } | ||||
| @@ -166,7 +166,7 @@ void NTPClient::setUpdateInterval(unsigned long updateInterval) { | ||||
| } | ||||
|  | ||||
| void NTPClient::setPoolServerName(const char* poolServerName) { | ||||
|     this->_poolServerName = poolServerName; | ||||
|   this->_poolServerName = poolServerName; | ||||
| } | ||||
|  | ||||
| void NTPClient::sendNTPPacket() { | ||||
|   | ||||
| @@ -36,67 +36,67 @@ class NTPClient { | ||||
|     NTPClient(UDP& udp, const char* poolServerName, long timeOffset, unsigned long updateInterval); | ||||
|  | ||||
|     /** | ||||
|      * Set time server name | ||||
|      * | ||||
|      * @param poolServerName | ||||
|      */ | ||||
|        Set time server name | ||||
|  | ||||
|        @param poolServerName | ||||
|     */ | ||||
|     void setPoolServerName(const char* poolServerName); | ||||
|  | ||||
|     /** | ||||
|      * Starts the underlying UDP client with the default local port | ||||
|      */ | ||||
|        Starts the underlying UDP client with the default local port | ||||
|     */ | ||||
|     void begin(); | ||||
|  | ||||
|     /** | ||||
|      * Starts the underlying UDP client with the specified local port | ||||
|      */ | ||||
|        Starts the underlying UDP client with the specified local port | ||||
|     */ | ||||
|     void begin(int port); | ||||
|  | ||||
|     /** | ||||
|      * This should be called in the main loop of your application. By default an update from the NTP Server is only | ||||
|      * made every 60 seconds. This can be configured in the NTPClient constructor. | ||||
|      * | ||||
|      * @return true on success, false on failure | ||||
|      */ | ||||
|        This should be called in the main loop of your application. By default an update from the NTP Server is only | ||||
|        made every 60 seconds. This can be configured in the NTPClient constructor. | ||||
|  | ||||
|        @return true on success, false on failure | ||||
|     */ | ||||
|     bool update(); | ||||
|  | ||||
|     /** | ||||
|      * This will force the update from the NTP Server. | ||||
|      * | ||||
|      * @return true on success, false on failure | ||||
|      */ | ||||
|        This will force the update from the NTP Server. | ||||
|  | ||||
|        @return true on success, false on failure | ||||
|     */ | ||||
|     bool forceUpdate(); | ||||
|  | ||||
|     int getDay() const; | ||||
|     int getHours() const; | ||||
|     int getMinutes() const; | ||||
|     int getSeconds() const; | ||||
| 	int getMillis() const; | ||||
| 	int getMillisLastUpd() const; | ||||
|     int getMillis() const; | ||||
|     int getMillisLastUpd() const; | ||||
|  | ||||
|     /** | ||||
|      * Changes the time offset. Useful for changing timezones dynamically | ||||
|      */ | ||||
|        Changes the time offset. Useful for changing timezones dynamically | ||||
|     */ | ||||
|     void setTimeOffset(int timeOffset); | ||||
|  | ||||
|     /** | ||||
|      * Set the update interval to another frequency. E.g. useful when the | ||||
|      * timeOffset should not be set in the constructor | ||||
|      */ | ||||
|        Set the update interval to another frequency. E.g. useful when the | ||||
|        timeOffset should not be set in the constructor | ||||
|     */ | ||||
|     void setUpdateInterval(unsigned long updateInterval); | ||||
|  | ||||
|     /** | ||||
|      * @return time formatted like `hh:mm:ss` | ||||
|      */ | ||||
|        @return time formatted like `hh:mm:ss` | ||||
|     */ | ||||
|     String getFormattedTime() const; | ||||
|  | ||||
|     /** | ||||
|      * @return time in seconds since Jan. 1, 1970 | ||||
|      */ | ||||
|        @return time in seconds since Jan. 1, 1970 | ||||
|     */ | ||||
|     unsigned long getEpochTime() const; | ||||
|  | ||||
|     /** | ||||
|      * Stops the underlying UDP client | ||||
|      */ | ||||
|        Stops the underlying UDP client | ||||
|     */ | ||||
|     void end(); | ||||
| }; | ||||
|   | ||||
| @@ -9,7 +9,7 @@ class Time { | ||||
|     uint32_t weekS = 0; | ||||
|  | ||||
|     int getMs() { | ||||
|       return (millis() - tmr); | ||||
|       return (tmr - millis()); | ||||
|     } | ||||
|     void setMs(int ms) { | ||||
|       tmr = millis() + ms; | ||||
|   | ||||
| @@ -123,6 +123,6 @@ class VolAnalyzer { | ||||
|     int max = 0, count = 0; | ||||
|     int maxs = 0, mins = 1023; | ||||
|     int _volMin = 0, _volMax = 100, _trsh = 30; | ||||
|     bool _pulse = 0, _first = 0;     | ||||
|     bool _pulse = 0, _first = 0; | ||||
|     FastFilter minF, maxF, volF; | ||||
| }; | ||||
|   | ||||
| @@ -1,6 +1,12 @@ | ||||
| #if (USE_ADC == 1) | ||||
| void setupADC() { | ||||
|   clap.setTimeout(500); | ||||
|   clap.setTrsh(250); | ||||
|  | ||||
|   vol.setDt(700); | ||||
|   vol.setPeriod(5); | ||||
|   vol.setWindow(map(MAX_LEDS, 300, 1200, 20, 1)); | ||||
|  | ||||
|   low.setDt(0); | ||||
|   low.setPeriod(0); | ||||
|   low.setWindow(0); | ||||
| @@ -8,9 +14,9 @@ void setupADC() { | ||||
|   high.setPeriod(0); | ||||
|   high.setWindow(0); | ||||
|  | ||||
|   vol.setVolK(20); | ||||
|   low.setVolK(20); | ||||
|   high.setVolK(20); | ||||
|   vol.setVolK(26); | ||||
|   low.setVolK(26); | ||||
|   high.setVolK(26); | ||||
|  | ||||
|   vol.setTrsh(50); | ||||
|   low.setTrsh(50); | ||||
| @@ -33,26 +39,26 @@ void setupADC() { | ||||
|  | ||||
|  | ||||
| void checkAnalog() { | ||||
|   //if (cfg.state) { | ||||
|   switch (cfg.adcMode) { | ||||
|     case GL_ADC_NONE: break; | ||||
|     case GL_ADC_BRI: checkPhot(); break; | ||||
|     case GL_ADC_MIC: checkMusic(); break; | ||||
|     case GL_ADC_BOTH: | ||||
|       { | ||||
|         static timerMillis tmr(1000, 1); | ||||
|         if (tmr.isReady()) { | ||||
|           switchToPhot(); | ||||
|           phot.setRaw(analogRead(A0)); | ||||
|           switchToMic(); | ||||
|         } else { | ||||
|           checkMusic(); | ||||
|   if (cfg.role || millis() - gotADCtmr >= 2000) {   // только мастер или слейв по таймауту опрашивает АЦП! | ||||
|     switch (cfg.adcMode) { | ||||
|       case GL_ADC_NONE: break; | ||||
|       case GL_ADC_BRI: checkPhot(); break; | ||||
|       case GL_ADC_MIC: checkMusic(); break; | ||||
|       case GL_ADC_BOTH: | ||||
|         { | ||||
|           static timerMillis tmr(1000, 1); | ||||
|           if (tmr.isReady()) { | ||||
|             switchToPhot(); | ||||
|             phot.setRaw(analogRead(A0)); | ||||
|             switchToMic(); | ||||
|           } else { | ||||
|             checkMusic(); | ||||
|           } | ||||
|           phot.compute(); | ||||
|         } | ||||
|         phot.compute(); | ||||
|       } | ||||
|       break; | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
|   //} | ||||
| } | ||||
|  | ||||
| void checkMusic() { | ||||
| @@ -61,10 +67,11 @@ void checkMusic() { | ||||
|   clap.tick(vol.getRawMax()); | ||||
|   if (clap.hasClaps(2)) controlHandler(!cfg.state); | ||||
| #endif | ||||
|    | ||||
|   yield(); | ||||
|   if (CUR_PRES.advMode == GL_ADV_LOW || CUR_PRES.advMode == GL_ADV_HIGH) {   // частоты | ||||
|     int raw[FFT_SIZE], spectr[FFT_SIZE]; | ||||
|     for (int i = 0; i < FFT_SIZE; i++) raw[i] = analogRead(A0); | ||||
|     yield(); | ||||
|     FFT(raw, spectr); | ||||
|     int low_raw = 0; | ||||
|     int high_raw = 0; | ||||
|   | ||||
| @@ -27,11 +27,17 @@ void button() { | ||||
|         changePreset(-1); | ||||
|         sendToSlaves(1, cfg.curPreset); | ||||
|         break; | ||||
|       case 4: | ||||
|         setPreset(0); | ||||
|         sendToSlaves(1, cfg.curPreset); | ||||
|         break; | ||||
|       case 5: | ||||
|         cfg.role = 0; | ||||
|         blink16(CRGB::DarkSlateBlue); | ||||
|         break; | ||||
|       case 6: | ||||
|         cfg.role = 1; | ||||
|         blink16(CRGB::Maroon); | ||||
|         break; | ||||
|     } | ||||
|     EE_updateCfg(); | ||||
|   | ||||
| @@ -41,12 +41,12 @@ int mapFF(byte x, byte min, byte max) { | ||||
| const char OTAhost[] = "http://ota.alexgyver.ru/"; | ||||
| const char *OTAfile[] = { | ||||
|   "GL2_latest.bin", | ||||
|   "com_600.bin", | ||||
|   "com_1200.bin", | ||||
|   "esp1_600.bin", | ||||
|   "esp1_1200.bin", | ||||
|   "module_600.bin", | ||||
|   "module_1200.bin", | ||||
|   "com_300.bin", | ||||
|   "com_900.bin", | ||||
|   "esp1_300.bin", | ||||
|   "esp1_900.bin", | ||||
|   "module_300.bin", | ||||
|   "module_900.bin", | ||||
| }; | ||||
|  | ||||
| const char NTPserver[] = "pool.ntp.org"; | ||||
| @@ -56,6 +56,7 @@ const char NTPserver[] = "pool.ntp.org"; | ||||
| //"ntp2.stratum2.ru" | ||||
| //"ntp.msk-ix.ru" | ||||
|  | ||||
| #define PAL_SIZE 49 | ||||
| struct Palette { | ||||
|   byte size = 1; | ||||
|   byte strip[16 * 3]; | ||||
| @@ -120,6 +121,7 @@ struct Preset { | ||||
|   byte rnd = 0;           // случайный (0/1) | ||||
| }; | ||||
|  | ||||
| #define DAWN_SIZE 23 | ||||
| struct Dawn { | ||||
|   byte state[7] = {0, 0, 0, 0, 0, 0, 0};  // (1/0) | ||||
|   byte hour[7] = {0, 0, 0, 0, 0, 0, 0};   // (0.. 59) | ||||
| @@ -127,3 +129,9 @@ struct Dawn { | ||||
|   byte bright = 100;      // (0.. 255) | ||||
|   byte time = 1;          // (5,10,15,20..) | ||||
| }; | ||||
|  | ||||
| /* | ||||
|   - Каждые 5 минут лампа AP отправляет время (день час минута) на Local лампы всех ролей в сети с ней (GL,6,день,час,мин) | ||||
|   - Если включен АЦП, Мастер отправляет своей группе данные с него на каждой итерации отрисовки эффектов (GL,1,длина,масштаб,яркость) | ||||
|   - Установка времени с мобилы - получают все роли АР и Local (не получившие ntp) | ||||
| */ | ||||
|   | ||||
| @@ -5,6 +5,8 @@ bool EEpalFlag = false; | ||||
|  | ||||
| void EE_startup() { | ||||
|   // старт епром | ||||
|   EEPROM.begin(1000);   // старт епром | ||||
|   delay(100); | ||||
|   if (EEPROM.read(0) != EE_KEY) { | ||||
|     EEPROM.write(0, EE_KEY); | ||||
|     EEPROM.put(1, cfg); | ||||
| @@ -12,7 +14,7 @@ void EE_startup() { | ||||
|     EEPROM.put(sizeof(cfg) + sizeof(dawn) + 1, pal); | ||||
|     EEPROM.put(sizeof(cfg) + sizeof(dawn) + sizeof(pal) + 1, preset); | ||||
|     EEPROM.commit(); | ||||
|     blink8(CRGB::Magenta); | ||||
|     blink16(CRGB::Magenta); | ||||
|     DEBUGLN("First start"); | ||||
|   } | ||||
|   EEPROM.get(1, cfg); | ||||
|   | ||||
| @@ -1,12 +1,11 @@ | ||||
| void effectsRoutine() { | ||||
|   static timerMillis effTmr(30, true); | ||||
|   static byte prevEff = 255; | ||||
|   if (!effTmr.isReady()) return; | ||||
|  | ||||
|   if (dawnTmr.running()) { | ||||
|     if (effTmr.isReady()) { | ||||
|       fill_solid(leds, MAX_LEDS, ColorFromPalette(HeatColors_p, dawnTmr.getLength8(), scaleFF(dawnTmr.getLength8(), dawn.bright), LINEARBLEND)); | ||||
|       FastLED.show(); | ||||
|     } | ||||
|     fill_solid(leds, MAX_LEDS, ColorFromPalette(HeatColors_p, dawnTmr.getLength8(), scaleFF(dawnTmr.getLength8(), dawn.bright), LINEARBLEND)); | ||||
|     drawClock(cfg.length / 2 - 4, 150, 0); | ||||
|     FastLED.show(); | ||||
|     if (dawnTmr.isReady()) { | ||||
|       dawnTmr.stop(); | ||||
|       FastLED.clear(); | ||||
| @@ -15,207 +14,225 @@ void effectsRoutine() { | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   if (cfg.state && effTmr.isReady()) { | ||||
|     int thisLength = getLength(); | ||||
|     byte thisScale = getScale(); | ||||
|     int thisWidth = (cfg.deviceType > 1) ? cfg.width : 1; | ||||
|     byte thisBright = getBright(); | ||||
|   if (!cfg.state) return; | ||||
|   int thisLength = getLength(); | ||||
|   byte thisScale = getScale(); | ||||
|   byte thisBright = getBright(); | ||||
|  | ||||
|     if (turnoffTmr.running()) thisBright = scaleFF(thisBright, 255 - turnoffTmr.getLength8()); | ||||
|     else if (blinkTmr.runningStop()) thisBright = scaleFF(thisBright, blinkTmr.getLength8()); | ||||
|     if (turnoffTmr.isReady()) { | ||||
|       turnoffTmr.stop(); | ||||
|       setPower(0); | ||||
|       return; | ||||
|   if (cfg.adcMode > 1) {    // музыка или яркость | ||||
|     if (cfg.role) {         // мастер отправляет | ||||
|       static timerMillis adcSend(120, true); | ||||
|       if (adcSend.isReady() && millis() - udpTmr >= 1000) sendUDP(7, thisLength, thisScale, thisBright); | ||||
|     } else {                // слейв получает | ||||
|       if (millis() - gotADCtmr < 2000) {     // есть сигнал с мастера | ||||
|         thisLength = udpLength; | ||||
|         thisScale = udpScale; | ||||
|         thisBright = udpBright; | ||||
|       } | ||||
|     } | ||||
|     FastLED.setBrightness(thisBright); | ||||
|  | ||||
|     if (prevEff != CUR_PRES.effect) { | ||||
|       FastLED.clear(); | ||||
|       prevEff = CUR_PRES.effect; | ||||
|       loading = true; | ||||
|     } | ||||
|     // =================================================== ЭФФЕКТЫ =================================================== | ||||
|     switch (CUR_PRES.effect) { | ||||
|       case 1: // =================================== ПЕРЛИН =================================== | ||||
|         if (cfg.deviceType > 1) { | ||||
|           FOR_j(0, cfg.length) { | ||||
|             FOR_i(0, cfg.width) { | ||||
|               setPix(i, j, ColorFromPalette(paletteArr[CUR_PRES.palette - 1], | ||||
|                                             scalePal(inoise8( | ||||
|                                                 i * (thisScale / 5) - cfg.width * (thisScale / 5) / 2, | ||||
|                                                 j * (thisScale / 5) - cfg.length * (thisScale / 5) / 2, | ||||
|                                                 (now.weekMs >> 1) * CUR_PRES.speed / 255)), | ||||
|                                             255, LINEARBLEND)); | ||||
|             } | ||||
|           } | ||||
|  | ||||
|         } else { | ||||
|           FOR_i(0, cfg.length) { | ||||
|             leds[i] = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], | ||||
|                                        scalePal(inoise8(i * (thisScale / 5) - cfg.length * (thisScale / 5) / 2, | ||||
|                                                 (now.weekMs >> 1) * CUR_PRES.speed / 255)), | ||||
|                                        255, LINEARBLEND); | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 2: // ==================================== ЦВЕТ ==================================== | ||||
|         { | ||||
|           fill_solid(leds, cfg.length * thisWidth, CHSV(CUR_PRES.color, thisScale, 30)); | ||||
|           CRGB thisColor = CHSV(CUR_PRES.color, thisScale, thisBright); | ||||
|           if (CUR_PRES.fromCenter) { | ||||
|             fillStrip(cfg.length / 2, cfg.length / 2 + thisLength / 2, thisColor); | ||||
|             fillStrip(cfg.length / 2 - thisLength / 2, cfg.length / 2, thisColor); | ||||
|           } else { | ||||
|             fillStrip(0, thisLength, thisColor); | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 3: // ================================= СМЕНА ЦВЕТА ================================= | ||||
|         { | ||||
|           CRGB thisColor = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scalePal((now.weekMs >> 5) * CUR_PRES.speed / 255), 10, LINEARBLEND); | ||||
|           fill_solid(leds, cfg.length * thisWidth, thisColor); | ||||
|           thisColor = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scalePal((now.weekMs >> 5) * CUR_PRES.speed / 255), thisBright, LINEARBLEND); | ||||
|           if (CUR_PRES.fromCenter) { | ||||
|             fillStrip(cfg.length / 2, cfg.length / 2 + thisLength / 2, thisColor); | ||||
|             fillStrip(cfg.length / 2 - thisLength / 2, cfg.length / 2, thisColor); | ||||
|           } else { | ||||
|             fillStrip(0, thisLength, thisColor); | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 4: // ================================== ГРАДИЕНТ ================================== | ||||
|         if (CUR_PRES.fromCenter) { | ||||
|           FOR_i(cfg.length / 2, cfg.length) { | ||||
|             byte bright = 255; | ||||
|             if (CUR_PRES.soundReact == GL_REACT_LEN) bright = (i < cfg.length / 2 + thisLength / 2) ? (thisBright) : (10); | ||||
|             CRGB thisColor = ColorFromPalette( | ||||
|                                paletteArr[CUR_PRES.palette - 1],   // (x*1.9 + 25) / 255 - быстрый мап 0..255 в 0.1..2 | ||||
|                                scalePal((i * (thisScale * 1.9 + 25) / cfg.length) + ((now.weekMs >> 3) * (CUR_PRES.speed - 128) / 128)), | ||||
|                                bright, LINEARBLEND); | ||||
|             if (cfg.deviceType > 1) fillRow(i, thisColor); | ||||
|             else leds[i] = thisColor; | ||||
|           } | ||||
|           if (cfg.deviceType > 1) FOR_i(0, cfg.length / 2) fillRow(i, leds[(cfg.length - i)*cfg.width - 1]); | ||||
|           else FOR_i(0, cfg.length / 2) leds[i] = leds[cfg.length - i - 1]; | ||||
|  | ||||
|         } else { | ||||
|           FOR_i(0, cfg.length) { | ||||
|             byte bright = 255; | ||||
|             if (CUR_PRES.soundReact == GL_REACT_LEN) bright = (i < thisLength) ? (thisBright) : (10); | ||||
|             CRGB thisColor = ColorFromPalette( | ||||
|                                paletteArr[CUR_PRES.palette - 1],   // (x*1.9 + 25) / 255 - быстрый мап 0..255 в 0.1..2 | ||||
|                                scalePal((i * (thisScale * 1.9 + 25) / cfg.length) + ((now.weekMs >> 3) * (CUR_PRES.speed - 128) / 128)), | ||||
|                                bright, LINEARBLEND); | ||||
|             if (cfg.deviceType > 1) fillRow(i, thisColor); | ||||
|             else leds[i] = thisColor; | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 5: // =================================== ЧАСТИЦЫ =================================== | ||||
|         FOR_i(0, cfg.length * cfg.width) leds[i].fadeToBlackBy(70); | ||||
|         { | ||||
|           uint16_t rndVal = 0; | ||||
|           FOR_i(0, thisScale / 8 + 1) { | ||||
|             rndVal = rndVal * 2053 + 13849;     // random2053 алгоритм | ||||
|             int homeX = inoise16(i * 100000000ul + (now.weekMs << 3) * CUR_PRES.speed / 255); | ||||
|             homeX = map(homeX, 15000, 50000, 0, cfg.length); | ||||
|             int offsX = inoise8(i * 2500 + (now.weekMs >> 1) * CUR_PRES.speed / 255) - 128; | ||||
|             offsX = cfg.length / 2 * offsX / 128; | ||||
|             int thisX = homeX + offsX; | ||||
|  | ||||
|             if (cfg.deviceType > 1) { | ||||
|               int homeY = inoise16(i * 100000000ul + 2000000000ul + (now.weekMs << 3) * CUR_PRES.speed / 255); | ||||
|               homeY = map(homeY, 15000, 50000, 0, cfg.width); | ||||
|               int offsY = inoise8(i * 2500 + 30000 + (now.weekMs >> 1) * CUR_PRES.speed / 255) - 128; | ||||
|               offsY = cfg.length / 2 * offsY / 128; | ||||
|               int thisY = homeY + offsY; | ||||
|               setPix(thisX, thisY, CHSV(CUR_PRES.rnd ? rndVal : CUR_PRES.color, 255, 255)); | ||||
|             } else { | ||||
|               setLED(thisX, CHSV(CUR_PRES.rnd ? rndVal : CUR_PRES.color, 255, 255)); | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 6: // ==================================== ОГОНЬ ==================================== | ||||
|         if (cfg.deviceType > 1) {         // 2D огонь | ||||
|           fireRoutine(CUR_PRES.speed / 2); | ||||
|         } else {                          // 1D огонь | ||||
|           FastLED.clear(); | ||||
|           static byte heat[MAX_LEDS]; | ||||
|           CRGBPalette16 gPal; | ||||
|           if (CUR_PRES.color < 5) gPal = HeatColors_p; | ||||
|           else gPal = CRGBPalette16(CRGB::Black, CHSV(CUR_PRES.color, 255, 255), CRGB::White); | ||||
|           if (CUR_PRES.fromCenter) thisLength /= 2; | ||||
|  | ||||
|           for (int i = 0; i < thisLength; i++) heat[i] = qsub8(heat[i], random8(0, ((((255 - thisScale) / 2 + 20) * 10) / thisLength) + 2)); | ||||
|           for (int k = thisLength - 1; k >= 2; k--) heat[k] = (heat[k - 1] + heat[k - 2] + heat[k - 2] ) / 3; | ||||
|           if (random8() < 120 ) { | ||||
|             int y = random8(7); | ||||
|             heat[y] = qadd8(heat[y], random8(160, 255)); | ||||
|           } | ||||
|           if (CUR_PRES.fromCenter) { | ||||
|             for (int j = 0; j < thisLength; j++) leds[cfg.length / 2 + j] = ColorFromPalette(gPal, scale8(heat[j], 240)); | ||||
|             FOR_i(0, cfg.length / 2) leds[i] = leds[cfg.length - i - 1]; | ||||
|           } else { | ||||
|             for (int j = 0; j < thisLength; j++) leds[j] = ColorFromPalette(gPal, scale8(heat[j], 240)); | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 7: // ==================================== ОГОНЬ 2020 ==================================== | ||||
|         FastLED.clear(); | ||||
|         if (cfg.deviceType > 1) {         // 2D огонь | ||||
|           fire2020(CUR_PRES.scale, thisLength); | ||||
|         } else {                          // 1D огонь | ||||
|           static byte heat[MAX_LEDS]; | ||||
|           if (CUR_PRES.fromCenter) thisLength /= 2; | ||||
|  | ||||
|           for (int i = 0; i < thisLength; i++) heat[i] = qsub8(heat[i], random8(0, ((((255 - thisScale) / 2 + 20) * 10) / thisLength) + 2)); | ||||
|           for (int k = thisLength - 1; k >= 2; k--) heat[k] = (heat[k - 1] + heat[k - 2] + heat[k - 2] ) / 3; | ||||
|           if (random8() < 120 ) { | ||||
|             int y = random8(7); | ||||
|             heat[y] = qadd8(heat[y], random8(160, 255)); | ||||
|           } | ||||
|           if (CUR_PRES.fromCenter) { | ||||
|             for (int j = 0; j < thisLength; j++) leds[cfg.length / 2 + j] = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scale8(heat[j], 240)); | ||||
|             FOR_i(0, cfg.length / 2) leds[i] = leds[cfg.length - i - 1]; | ||||
|           } else { | ||||
|             for (int j = 0; j < thisLength; j++) leds[j] = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scale8(heat[j], 240)); | ||||
|           } | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 8: // ================================== КОНФЕТТИ ================================== | ||||
|         FOR_i(0, (thisScale >> 3) + 1) { | ||||
|           int x = random(0, cfg.length * cfg.width); | ||||
|           if (leds[x] == CRGB(0, 0, 0)) leds[x] = CHSV(CUR_PRES.rnd ? random(0, 255) : CUR_PRES.color, 255, 255); | ||||
|         } | ||||
|         FOR_i(0, cfg.length * cfg.width) { | ||||
|           if (leds[i].r >= 10 || leds[i].g >= 10 || leds[i].b >= 10) leds[i].fadeToBlackBy(CUR_PRES.speed / 2 + 1); | ||||
|           else leds[i] = 0; | ||||
|         } | ||||
|         break; | ||||
|  | ||||
|       case 9: // ================================== ПОГОДА ================================== | ||||
|  | ||||
|         break; | ||||
|     } | ||||
|  | ||||
|     // выводим нажатия кнопки | ||||
|     if (btnClicks > 0) fill_solid(leds, btnClicks, CRGB::White); | ||||
|     if (brTicks > 0) fill_solid(leds, brTicks, CRGB::Cyan); | ||||
|     FastLED.show(); | ||||
|   } | ||||
|  | ||||
|   if (turnoffTmr.running()) thisBright = scaleFF(thisBright, 255 - turnoffTmr.getLength8()); | ||||
|   else if (blinkTmr.runningStop()) thisBright = scaleFF(thisBright, blinkTmr.getLength8()); | ||||
|   if (turnoffTmr.isReady()) { | ||||
|     turnoffTmr.stop(); | ||||
|     setPower(0); | ||||
|     return; | ||||
|   } | ||||
|   FastLED.setBrightness(thisBright); | ||||
|  | ||||
|   if (prevEff != CUR_PRES.effect) {   // смена эффекта | ||||
|     FastLED.clear(); | ||||
|     prevEff = CUR_PRES.effect; | ||||
|     loading = true; | ||||
|   } | ||||
|  | ||||
|   // =================================================== ЭФФЕКТЫ =================================================== | ||||
|   switch (CUR_PRES.effect) { | ||||
|     case 1: // =================================== ПЕРЛИН =================================== | ||||
|       if (cfg.deviceType > 1) { | ||||
|         FOR_j(0, cfg.length) { | ||||
|           FOR_i(0, cfg.width) { | ||||
|             setPix(i, j, ColorFromPalette(paletteArr[CUR_PRES.palette - 1], | ||||
|                                           scalePal(inoise8( | ||||
|                                               i * (thisScale / 5) - cfg.width * (thisScale / 5) / 2, | ||||
|                                               j * (thisScale / 5) - cfg.length * (thisScale / 5) / 2, | ||||
|                                               (now.weekMs >> 1) * CUR_PRES.speed / 255)), | ||||
|                                           255, LINEARBLEND)); | ||||
|           } | ||||
|         } | ||||
|  | ||||
|       } else { | ||||
|         FOR_i(0, cfg.length) { | ||||
|           leds[i] = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], | ||||
|                                      scalePal(inoise8(i * (thisScale / 5) - cfg.length * (thisScale / 5) / 2, | ||||
|                                               (now.weekMs >> 1) * CUR_PRES.speed / 255)), | ||||
|                                      255, LINEARBLEND); | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 2: // ==================================== ЦВЕТ ==================================== | ||||
|       { | ||||
|         fill_solid(leds, cfg.length * cfg.width, CHSV(CUR_PRES.color, thisScale, 30)); | ||||
|         CRGB thisColor = CHSV(CUR_PRES.color, thisScale, thisBright); | ||||
|         if (CUR_PRES.fromCenter) { | ||||
|           fillStrip(cfg.length / 2, cfg.length / 2 + thisLength / 2, thisColor); | ||||
|           fillStrip(cfg.length / 2 - thisLength / 2, cfg.length / 2, thisColor); | ||||
|         } else { | ||||
|           fillStrip(0, thisLength, thisColor); | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 3: // ================================= СМЕНА ЦВЕТА ================================= | ||||
|       { | ||||
|         CRGB thisColor = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scalePal((now.weekMs >> 5) * CUR_PRES.speed / 255), 10, LINEARBLEND); | ||||
|         fill_solid(leds, cfg.length * cfg.width, thisColor); | ||||
|         thisColor = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scalePal((now.weekMs >> 5) * CUR_PRES.speed / 255), thisBright, LINEARBLEND); | ||||
|         if (CUR_PRES.fromCenter) { | ||||
|           fillStrip(cfg.length / 2, cfg.length / 2 + thisLength / 2, thisColor); | ||||
|           fillStrip(cfg.length / 2 - thisLength / 2, cfg.length / 2, thisColor); | ||||
|         } else { | ||||
|           fillStrip(0, thisLength, thisColor); | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 4: // ================================== ГРАДИЕНТ ================================== | ||||
|       if (CUR_PRES.fromCenter) { | ||||
|         FOR_i(cfg.length / 2, cfg.length) { | ||||
|           byte bright = 255; | ||||
|           if (CUR_PRES.soundReact == GL_REACT_LEN) bright = (i < cfg.length / 2 + thisLength / 2) ? (thisBright) : (10); | ||||
|           CRGB thisColor = ColorFromPalette( | ||||
|                              paletteArr[CUR_PRES.palette - 1],   // (x*1.9 + 25) / 255 - быстрый мап 0..255 в 0.1..2 | ||||
|                              scalePal((i * (thisScale * 1.9 + 25) / cfg.length) + ((now.weekMs >> 3) * (CUR_PRES.speed - 128) / 128)), | ||||
|                              bright, LINEARBLEND); | ||||
|           if (cfg.deviceType > 1) fillRow(i, thisColor); | ||||
|           else leds[i] = thisColor; | ||||
|         } | ||||
|         if (cfg.deviceType > 1) FOR_i(0, cfg.length / 2) fillRow(i, leds[(cfg.length - i)*cfg.width - 1]); | ||||
|         else FOR_i(0, cfg.length / 2) leds[i] = leds[cfg.length - i - 1]; | ||||
|  | ||||
|       } else { | ||||
|         FOR_i(0, cfg.length) { | ||||
|           byte bright = 255; | ||||
|           if (CUR_PRES.soundReact == GL_REACT_LEN) bright = (i < thisLength) ? (thisBright) : (10); | ||||
|           CRGB thisColor = ColorFromPalette( | ||||
|                              paletteArr[CUR_PRES.palette - 1],   // (x*1.9 + 25) / 255 - быстрый мап 0..255 в 0.1..2 | ||||
|                              scalePal((i * (thisScale * 1.9 + 25) / cfg.length) + ((now.weekMs >> 3) * (CUR_PRES.speed - 128) / 128)), | ||||
|                              bright, LINEARBLEND); | ||||
|           if (cfg.deviceType > 1) fillRow(i, thisColor); | ||||
|           else leds[i] = thisColor; | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 5: // =================================== ЧАСТИЦЫ =================================== | ||||
|       FOR_i(0, cfg.length * cfg.width) leds[i].fadeToBlackBy(70); | ||||
|       { | ||||
|         uint16_t rndVal = 0; | ||||
|         FOR_i(0, thisScale / 8 + 1) { | ||||
|           rndVal = rndVal * 2053 + 13849;     // random2053 алгоритм | ||||
|           int homeX = inoise16(i * 100000000ul + (now.weekMs << 3) * CUR_PRES.speed / 255); | ||||
|           homeX = map(homeX, 15000, 50000, 0, cfg.length); | ||||
|           int offsX = inoise8(i * 2500 + (now.weekMs >> 1) * CUR_PRES.speed / 255) - 128; | ||||
|           offsX = cfg.length / 2 * offsX / 128; | ||||
|           int thisX = homeX + offsX; | ||||
|  | ||||
|           if (cfg.deviceType > 1) { | ||||
|             int homeY = inoise16(i * 100000000ul + 2000000000ul + (now.weekMs << 3) * CUR_PRES.speed / 255); | ||||
|             homeY = map(homeY, 15000, 50000, 0, cfg.width); | ||||
|             int offsY = inoise8(i * 2500 + 30000 + (now.weekMs >> 1) * CUR_PRES.speed / 255) - 128; | ||||
|             offsY = cfg.length / 2 * offsY / 128; | ||||
|             int thisY = homeY + offsY; | ||||
|             setPix(thisX, thisY, CHSV(CUR_PRES.rnd ? rndVal : CUR_PRES.color, 255, 255)); | ||||
|           } else { | ||||
|             setLED(thisX, CHSV(CUR_PRES.rnd ? rndVal : CUR_PRES.color, 255, 255)); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 6: // ==================================== ОГОНЬ ==================================== | ||||
|       if (cfg.deviceType > 1) {         // 2D огонь | ||||
|         fireRoutine(CUR_PRES.speed / 2); | ||||
|       } else {                          // 1D огонь | ||||
|         FastLED.clear(); | ||||
|         static byte heat[MAX_LEDS]; | ||||
|         CRGBPalette16 gPal; | ||||
|         if (CUR_PRES.color < 5) gPal = HeatColors_p; | ||||
|         else gPal = CRGBPalette16(CRGB::Black, CHSV(CUR_PRES.color, 255, 255), CRGB::White); | ||||
|         if (CUR_PRES.fromCenter) thisLength /= 2; | ||||
|  | ||||
|         for (int i = 0; i < thisLength; i++) heat[i] = qsub8(heat[i], random8(0, ((((255 - thisScale) / 2 + 20) * 10) / thisLength) + 2)); | ||||
|         for (int k = thisLength - 1; k >= 2; k--) heat[k] = (heat[k - 1] + heat[k - 2] + heat[k - 2] ) / 3; | ||||
|         if (random8() < 120 ) { | ||||
|           int y = random8(7); | ||||
|           heat[y] = qadd8(heat[y], random8(160, 255)); | ||||
|         } | ||||
|         if (CUR_PRES.fromCenter) { | ||||
|           for (int j = 0; j < thisLength; j++) leds[cfg.length / 2 + j] = ColorFromPalette(gPal, scale8(heat[j], 240)); | ||||
|           FOR_i(0, cfg.length / 2) leds[i] = leds[cfg.length - i - 1]; | ||||
|         } else { | ||||
|           for (int j = 0; j < thisLength; j++) leds[j] = ColorFromPalette(gPal, scale8(heat[j], 240)); | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 7: // ==================================== ОГОНЬ 2020 ==================================== | ||||
|       FastLED.clear(); | ||||
|       if (cfg.deviceType > 1) {         // 2D огонь | ||||
|         fire2020(CUR_PRES.scale, thisLength); | ||||
|       } else {                          // 1D огонь | ||||
|         static byte heat[MAX_LEDS]; | ||||
|         if (CUR_PRES.fromCenter) thisLength /= 2; | ||||
|  | ||||
|         for (int i = 0; i < thisLength; i++) heat[i] = qsub8(heat[i], random8(0, ((((255 - thisScale) / 2 + 20) * 10) / thisLength) + 2)); | ||||
|         for (int k = thisLength - 1; k >= 2; k--) heat[k] = (heat[k - 1] + heat[k - 2] + heat[k - 2] ) / 3; | ||||
|         if (random8() < 120 ) { | ||||
|           int y = random8(7); | ||||
|           heat[y] = qadd8(heat[y], random8(160, 255)); | ||||
|         } | ||||
|         if (CUR_PRES.fromCenter) { | ||||
|           for (int j = 0; j < thisLength; j++) leds[cfg.length / 2 + j] = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scale8(heat[j], 240)); | ||||
|           FOR_i(0, cfg.length / 2) leds[i] = leds[cfg.length - i - 1]; | ||||
|         } else { | ||||
|           for (int j = 0; j < thisLength; j++) leds[j] = ColorFromPalette(paletteArr[CUR_PRES.palette - 1], scale8(heat[j], 240)); | ||||
|         } | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 8: // ================================== КОНФЕТТИ ================================== | ||||
|       FOR_i(0, (thisScale >> 3) + 1) { | ||||
|         int x = random(0, cfg.length * cfg.width); | ||||
|         if (leds[x] == CRGB(0, 0, 0)) leds[x] = CHSV(CUR_PRES.rnd ? random(0, 255) : CUR_PRES.color, 255, 255); | ||||
|       } | ||||
|       FOR_i(0, cfg.length * cfg.width) { | ||||
|         if (leds[i].r >= 10 || leds[i].g >= 10 || leds[i].b >= 10) leds[i].fadeToBlackBy(CUR_PRES.speed / 2 + 1); | ||||
|         else leds[i] = 0; | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case 9: // =================================== ЧАСЫ =================================== | ||||
|       FastLED.clear(); | ||||
|       drawClock(mapFF(CUR_PRES.scale, 0, cfg.length - 7), (255 - CUR_PRES.speed), CHSV(CUR_PRES.color, 255, 255)); | ||||
|       break;  | ||||
|  | ||||
|     case 10: // ================================= ПОГОДА ================================== | ||||
|  | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   if (CUR_PRES.advMode == GL_ADV_CLOCK && CUR_PRES.effect != 9) drawClock(mapFF(CUR_PRES.scale, 0, cfg.length - 7), 150, 0); | ||||
|   // выводим нажатия кнопки | ||||
|   if (btnClicks > 0) fill_solid(leds, btnClicks, CRGB::White); | ||||
|   if (brTicks > 0) fill_solid(leds, brTicks, CRGB::Cyan); | ||||
|   yield(); | ||||
|   FastLED.show(); | ||||
| } | ||||
|  | ||||
| // ==================================================================================================================== | ||||
|  | ||||
| bool musicMode() { | ||||
|   return ((cfg.adcMode == GL_ADC_MIC || cfg.adcMode == GL_ADC_BOTH) && (CUR_PRES.advMode > 1 && CUR_PRES.advMode <= 4)); | ||||
| } | ||||
| @@ -273,17 +290,6 @@ void updPal() { | ||||
|   if (pal.size < 16) paletteArr[0][pal.size] = paletteArr[0][0]; | ||||
| } | ||||
|  | ||||
| void blink8(CRGB color) { | ||||
|   FOR_i(0, 3) { | ||||
|     fill_solid(leds, 8, color); | ||||
|     FastLED.show(); | ||||
|     delay(300); | ||||
|     FastLED.clear(); | ||||
|     FastLED.show(); | ||||
|     delay(300); | ||||
|   } | ||||
| } | ||||
|  | ||||
| byte scalePal(byte val) { | ||||
|   if (CUR_PRES.palette == 1) val = val * pal.size / 16; | ||||
|   return val; | ||||
|   | ||||
| @@ -12,7 +12,7 @@ void fire2020(byte scale, int len) { | ||||
|   static float trackingObjectPosX[100]; | ||||
|   static float trackingObjectPosY[100]; | ||||
|   static uint16_t ff_x, ff_y, ff_z; | ||||
|    | ||||
|  | ||||
|  | ||||
|   if (loading) { | ||||
|     loading = false; | ||||
|   | ||||
| @@ -28,7 +28,7 @@ const unsigned char hueMask[11][16] PROGMEM = { | ||||
| byte fireLine[100]; | ||||
|  | ||||
| void fireRoutine(byte speed) { | ||||
|   static byte count = 0;   | ||||
|   static byte count = 0; | ||||
|   if (count >= 100) { | ||||
|     shiftUp(); | ||||
|     FOR_i(0, cfg.width) fireLine[i] = random(64, 255); | ||||
|   | ||||
| @@ -43,68 +43,67 @@ char* mFtoa(double value, int8_t decimals, char *buffer) { | ||||
|  | ||||
| class mString { | ||||
|   public: | ||||
|     int size = 0; | ||||
|     char* buf; | ||||
|     // system*this = buf; | ||||
|     int size = 0; | ||||
|     uint16_t length() { | ||||
|       return strlen(buf); | ||||
|     } | ||||
|     void clear() { | ||||
|       buf[0] = 0; | ||||
|       buf[0] = NULL; | ||||
|     } | ||||
|  | ||||
|     // constructor | ||||
|     mString(char* buffer, int newSize) { | ||||
|       //*this = buf; | ||||
|       buf = buffer; | ||||
|       size = newSize; | ||||
|     } | ||||
|     /*mString (const char c) { | ||||
|     	init(); | ||||
|     	add(c); | ||||
|       //init(); | ||||
|       add(c); | ||||
|       } | ||||
|       mString (const char* data) { | ||||
|     	init(); | ||||
|     	add(data); | ||||
|       //init(); | ||||
|       add(data); | ||||
|       } | ||||
|       mString (const __FlashStringHelper *data) { | ||||
|     	init(); | ||||
|     	add(data); | ||||
|       //init(); | ||||
|       add(data); | ||||
|       } | ||||
|       mString (uint32_t value) { | ||||
|     	init(); | ||||
|     	add(value); | ||||
|       //init(); | ||||
|       add(value); | ||||
|       } | ||||
|       mString (int32_t value) { | ||||
|     	init(); | ||||
|     	add(value); | ||||
|       //init(); | ||||
|       add(value); | ||||
|       } | ||||
|       mString (uint16_t value) { | ||||
|     	init(); | ||||
|     	add(value); | ||||
|       //init(); | ||||
|       add(value); | ||||
|       } | ||||
|       mString (int16_t value) { | ||||
|     	init(); | ||||
|     	add(value); | ||||
|       //init(); | ||||
|       add(value); | ||||
|       } | ||||
|       mString (uint8_t value) { | ||||
|     	init(); | ||||
|     	add(value); | ||||
|       //init(); | ||||
|       add(value); | ||||
|       } | ||||
|       mString (int8_t value) { | ||||
|     	init(); | ||||
|     	add(value); | ||||
|       //init(); | ||||
|       add(value); | ||||
|       } | ||||
|       mString (double value, byte dec = 2) { | ||||
|     	init(); | ||||
|     	add(value, dec); | ||||
|       //init(); | ||||
|       add(value, dec); | ||||
|       }*/ | ||||
|  | ||||
|     // add | ||||
|     mString& add(const char c) { | ||||
|       byte len = length(); | ||||
|       if (len + 1 >= size) return *this; | ||||
|       buf[len++] = c; | ||||
|       buf[len++] = 0; | ||||
|       buf[len++] = NULL; | ||||
|       return *this; | ||||
|     } | ||||
|     mString& add(const char* data) { | ||||
| @@ -112,11 +111,13 @@ class mString { | ||||
|         do { | ||||
|         buf[len] = *(data++); | ||||
|         } while (buf[len++] != 0);*/ | ||||
|       if (length() + strlen(data) >= size) return *this; | ||||
|       strcpy(buf + length(), data); | ||||
|       return *this; | ||||
|     } | ||||
|     mString& add(const __FlashStringHelper *data) { | ||||
|       PGM_P p = reinterpret_cast<PGM_P>(data); | ||||
|       if (length() + strlen_P(p) >= size) return *this; | ||||
|       strcpy_P(buf + length(), p); | ||||
|       return *this; | ||||
|       /*do { | ||||
| @@ -125,10 +126,9 @@ class mString { | ||||
|       */ | ||||
|     } | ||||
|     mString& add(uint32_t value) { | ||||
|       //char buf[11]; | ||||
|       //return add(mUtoa(value, buf)); | ||||
|       utoa(value, buf + length(), DEC); | ||||
|       return *this; | ||||
|       char vBuf[11]; | ||||
|       utoa(value, vBuf, DEC); | ||||
|       return add(vBuf); | ||||
|     } | ||||
|     mString& add(uint16_t value) { | ||||
|       return add((uint32_t)value); | ||||
| @@ -137,10 +137,9 @@ class mString { | ||||
|       return add((uint32_t)value); | ||||
|     } | ||||
|     mString& add(int32_t value) { | ||||
|       //char buf[11]; | ||||
|       //return add(mLtoa(value, buf)); | ||||
|       ltoa(value, buf + length(), DEC); | ||||
|       return *this; | ||||
|       char vBuf[11]; | ||||
|       ltoa(value, vBuf, DEC); | ||||
|       return add(vBuf); | ||||
|     } | ||||
|     mString& add(int16_t value) { | ||||
|       return add((int32_t)value); | ||||
| @@ -149,11 +148,13 @@ class mString { | ||||
|       return add((int32_t)value); | ||||
|     } | ||||
|     mString& add(double value, int8_t dec = 2) { | ||||
|       char buf[20]; | ||||
|       return add(mFtoa(value, dec, buf)); | ||||
|       //dtostrf(value, dec, DEC, buf+length()); | ||||
|       //return *this; | ||||
|       char vBuf[20]; | ||||
|       mFtoa(value, dec, vBuf); | ||||
|       return add(vBuf); | ||||
|     } | ||||
|     /*mString& add(mString data) { | ||||
|       return add(data.buf); | ||||
|       }*/ | ||||
|  | ||||
|     // add += | ||||
|     mString& operator += (const char c) { | ||||
| @@ -186,6 +187,44 @@ class mString { | ||||
|     mString& operator += (double value) { | ||||
|       return add(value); | ||||
|     } | ||||
|     /*mString& operator += (mString data) { | ||||
|       return add(data); | ||||
|       }*/ | ||||
|  | ||||
|     // + | ||||
|     mString operator + (const char c) { | ||||
|       return mString(*this) += c; | ||||
|     } | ||||
|     mString operator + (const char* data) { | ||||
|       return mString(*this) += data; | ||||
|     } | ||||
|     mString operator + (const __FlashStringHelper *data) { | ||||
|       return mString(*this) += data; | ||||
|     } | ||||
|     mString operator + (uint32_t value) { | ||||
|       return mString(*this) += value; | ||||
|     } | ||||
|     mString operator + (int32_t value) { | ||||
|       return mString(*this) += value; | ||||
|     } | ||||
|     mString operator + (uint16_t value) { | ||||
|       return mString(*this) += value; | ||||
|     } | ||||
|     mString operator + (int16_t value) { | ||||
|       return mString(*this) += value; | ||||
|     } | ||||
|     mString operator + (uint8_t value) { | ||||
|       return mString(*this) += value; | ||||
|     } | ||||
|     mString operator + (int8_t value) { | ||||
|       return mString(*this) += value; | ||||
|     } | ||||
|     mString operator + (double value) { | ||||
|       return mString(*this) += value; | ||||
|     } | ||||
|     /*mString operator + (mString data) { | ||||
|       return mString(*this) += data; | ||||
|       }*/ | ||||
|  | ||||
|     // assign | ||||
|     mString& operator = (const char c) { | ||||
| @@ -228,6 +267,10 @@ class mString { | ||||
|       clear(); | ||||
|       return add(value); | ||||
|     } | ||||
|     /*mString& operator = (mString data) { | ||||
|       clear(); | ||||
|       return add(data); | ||||
|       }*/ | ||||
|  | ||||
|     // compare | ||||
|     bool operator == (const char c) { | ||||
| @@ -248,15 +291,17 @@ class mString { | ||||
|       char valBuf[20]; | ||||
|       return !strcmp(buf, mFtoa(value, 2, valBuf)); | ||||
|     } | ||||
|     /*bool operator == (mString data) { | ||||
|       return (buf == data.buf); | ||||
|       }*/ | ||||
|  | ||||
|     // convert & parse | ||||
|     char operator [] (uint16_t index) const { | ||||
|       return (index < size ? buf[index] : 0); | ||||
|     } | ||||
|     char& operator [] (uint16_t index) { | ||||
|       return buf[index]; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     // convert & parse | ||||
|     uint32_t toInt() { | ||||
|       return atoi(buf); | ||||
|     } | ||||
|   | ||||
| @@ -1,26 +1,41 @@ | ||||
| void parsing() { | ||||
|   if (Udp.parsePacket()) { | ||||
|     static uint32_t tmr = 0; | ||||
|     static char buf[UDP_TX_PACKET_MAX_SIZE + 1]; | ||||
|  | ||||
|     int n = Udp.read(buf, UDP_TX_PACKET_MAX_SIZE); | ||||
|     if (millis() - tmr < 500) return;  // принимаем посылки не чаще 2 раз в секунду | ||||
|     tmr = millis(); | ||||
|  | ||||
|     buf[n] = NULL; | ||||
|     DEBUGLN(buf);   // пакет вида <ключ>,<канал>,<тип>,<дата1>,<дата2>... | ||||
|     byte keyLen = strchr(buf, ',') - buf;     // indexof | ||||
|     if (strncmp(buf, GL_KEY, keyLen)) return; // не наш ключ | ||||
|  | ||||
|     byte data[MAX_PRESETS * PRES_SIZE + 5]; | ||||
|     memset(data, 0, MAX_PRESETS * PRES_SIZE + 5); | ||||
|     // ПРЕ-ПАРСИНГ (для данных АЦП) | ||||
|     if (buf[0] != 'G' || buf[1] != 'L' || buf[2] != ',') return;  // защита от не наших данных | ||||
|     if (buf[3] == '7') {   // АЦП GL,7, | ||||
|       if (!cfg.role) {     // принимаем данные ацп если слейв | ||||
|         int data[3]; | ||||
|         mString ints(buf + 5, 20); | ||||
|         ints.parseInts(data, 3); | ||||
|         udpLength = data[0]; | ||||
|         udpScale = data[1]; | ||||
|         udpBright = data[2]; | ||||
|         effTmr.force();   // форсируем отрисовку эффекта | ||||
|         gotADCtmr = millis(); | ||||
|       } | ||||
|       return;   // выходим | ||||
|     } | ||||
|  | ||||
|     if (millis() - udpTmr < 500) return;   // принимаем остальные посылки не чаще 2 раз в секунду | ||||
|     udpTmr = millis(); | ||||
|  | ||||
|     DEBUGLN(buf);   // пакет вида <GL>,<тип>,<дата1>,<дата2>... | ||||
|  | ||||
|     // ПАРСИНГ | ||||
|     byte data[MAX_PRESETS * PRES_SIZE + 10]; | ||||
|     memset(data, 0, sizeof(data)); | ||||
|     int count = 0; | ||||
|     char *str, *p = buf + keyLen;  // сдвиг до даты | ||||
|     char *str, *p = buf; | ||||
|     char *ssid, *pass; | ||||
|  | ||||
|     while ((str = strtok_r(p, ",", &p)) != NULL) { | ||||
|       uint32_t thisInt = atoi(str); | ||||
|       data[count++] = (byte)thisInt; | ||||
|       data[count++] = (byte)thisInt;  // парс байтов | ||||
|       // парс "тяжёлых" данных | ||||
|       if (data[1] == 0) { | ||||
|         if (count == 4) ssid = str; | ||||
|         if (count == 5) pass = str; | ||||
| @@ -38,20 +53,12 @@ void parsing() { | ||||
|         if (count == 24) strcpy(cfg.mqttPass, str); | ||||
|       } | ||||
|     } | ||||
|     yield(); | ||||
|  | ||||
|     // широковещательный запрос времени для local устройств в сети AP лампы | ||||
|     if (data[0] == 0 && cfg.WiFimode && !gotNTP) { | ||||
|       now.day = data[1]; | ||||
|       now.hour = data[2]; | ||||
|       now.min = data[3]; | ||||
|       now.sec = data[4]; | ||||
|       now.setMs(0); | ||||
|     } | ||||
|  | ||||
|     if (data[0] != cfg.group) return;     // не наш адрес, выходим | ||||
|  | ||||
|     switch (data[1]) {  // тип 0 - control, 1 - config, 2 - effects, 3 - dawn, 4 - from master, 5 - palette | ||||
|     // тип 0 - control, 1 - config, 2 - effects, 3 - dawn, 4 - from master, 5 - palette, 6 - time | ||||
|     switch (data[1]) { | ||||
|       case 0: DEBUGLN("Control"); blinkTmr.restart(); | ||||
|         if (!cfg.state && data[2] != 1) return;   // если лампа выключена и это не команда на включение - не обрабатываем | ||||
|         switch (data[2]) { | ||||
|           case 0: controlHandler(0); break;               // выкл | ||||
|           case 1: controlHandler(1); break;               // вкл | ||||
| @@ -62,7 +69,7 @@ void parsing() { | ||||
|           case 6: setPreset(data[3] - 1); break;          // конкретный пресет data[3] | ||||
|           case 7: cfg.WiFimode = data[3]; EE_updCfgRst(); break;  // смена режима WiFi | ||||
|           case 8: cfg.role = data[3]; break;              // смена роли | ||||
|           case 9: cfg.group = data[3]; break;             // смена группы | ||||
|           case 9: cfg.group = data[3]; restartUDP(); break;   // смена группы | ||||
|           case 10:                                        // установка настроек WiFi | ||||
|             strcpy(cfg.ssid, ssid); | ||||
|             strcpy(cfg.pass, pass); | ||||
| @@ -90,6 +97,7 @@ void parsing() { | ||||
|             } | ||||
|             break; | ||||
|         } | ||||
|         if (data[2] < 7) setTime(data[3], data[4], data[5], data[6]); | ||||
|         EE_updCfg(); | ||||
|         break; | ||||
|  | ||||
| @@ -97,6 +105,7 @@ void parsing() { | ||||
|         FOR_i(0, CFG_SIZE) { | ||||
|           *((byte*)&cfg + i) = data[i + 2];   // загоняем в структуру | ||||
|         } | ||||
|         setTime(data[CFG_SIZE + 10 + 2], data[CFG_SIZE + 10 + 3], data[CFG_SIZE + 10 + 4], data[CFG_SIZE + 10 + 5]); | ||||
|         if (cfg.deviceType == GL_TYPE_STRIP) { | ||||
|           if (cfg.length > MAX_LEDS) cfg.length = MAX_LEDS; | ||||
|           cfg.width = 1; | ||||
| @@ -111,31 +120,37 @@ void parsing() { | ||||
|         break; | ||||
|  | ||||
|       case 2: DEBUGLN("Preset"); | ||||
|         cfg.presetAmount = data[2];   // кол-во режимов | ||||
|         FOR_j(0, cfg.presetAmount) { | ||||
|           FOR_i(0, PRES_SIZE) { | ||||
|             *((byte*)&preset + j * PRES_SIZE + i) = data[j * PRES_SIZE + i + 3]; // загоняем в структуру | ||||
|         { | ||||
|           cfg.presetAmount = data[2];   // кол-во режимов | ||||
|           FOR_j(0, cfg.presetAmount) { | ||||
|             FOR_i(0, PRES_SIZE) { | ||||
|               *((byte*)&preset + j * PRES_SIZE + i) = data[j * PRES_SIZE + i + 3]; // загоняем в структуру | ||||
|             } | ||||
|           } | ||||
|           //if (!cfg.rotation) setPreset(data[cfg.presetAmount * PRES_SIZE + 3] - 1); | ||||
|           byte dataStart = cfg.presetAmount * PRES_SIZE + 3; | ||||
|           setPreset(data[dataStart] - 1); | ||||
|           setTime(data[dataStart + 1], data[dataStart + 2], data[dataStart + 3], data[dataStart + 4]); | ||||
|  | ||||
|           EE_updatePreset(); | ||||
|           //presetRotation(true); // форсировать смену режима | ||||
|           holdPresTmr.restart(); | ||||
|           loading = true; | ||||
|         } | ||||
|         //if (!cfg.rotation) setPreset(data[cfg.presetAmount * PRES_SIZE + 3] - 1); | ||||
|         setPreset(data[cfg.presetAmount * PRES_SIZE + 3] - 1); | ||||
|         EE_updatePreset(); | ||||
|         //presetRotation(true); // форсировать смену режима | ||||
|         holdPresTmr.restart(); | ||||
|         loading = true; | ||||
|         break; | ||||
|  | ||||
|       case 3: DEBUGLN("Dawn"); blinkTmr.restart(); | ||||
|         FOR_i(0, (2 + 3 * 7)) { | ||||
|         FOR_i(0, DAWN_SIZE) { | ||||
|           *((byte*)&dawn + i) = data[i + 2]; // загоняем в структуру | ||||
|         } | ||||
|         setTime(data[DAWN_SIZE + 2], data[DAWN_SIZE + 3], data[DAWN_SIZE + 4], data[DAWN_SIZE + 5]); | ||||
|         EE_updateDawn(); | ||||
|         break; | ||||
|  | ||||
|       case 4: DEBUGLN("From master"); | ||||
|         if (cfg.role == GL_SLAVE) { | ||||
|           switch (data[2]) { | ||||
|             case 0: fade(data[3]); break;     // вкл выкл | ||||
|             case 0: fade(data[3]); break;         // вкл выкл | ||||
|             case 1: setPreset(data[3]); break;    // пресет | ||||
|             case 2: cfg.bright = data[3]; break;  // яркость | ||||
|           } | ||||
| @@ -144,20 +159,23 @@ void parsing() { | ||||
|         break; | ||||
|  | ||||
|       case 5: DEBUGLN("Palette"); blinkTmr.restart(); | ||||
|         FOR_i(0, 1 + 16 * 3) { | ||||
|         FOR_i(0, PAL_SIZE) { | ||||
|           *((byte*)&pal + i) = data[i + 2]; // загоняем в структуру | ||||
|         } | ||||
|         setTime(data[PAL_SIZE + 2], data[PAL_SIZE + 3], data[PAL_SIZE + 4], data[PAL_SIZE + 5]); | ||||
|         updPal(); | ||||
|         EE_updatePal(); | ||||
|         break; | ||||
|  | ||||
|       case 6: DEBUGLN("Time"); blinkTmr.restart(); | ||||
|         if (!cfg.WiFimode) {  // если мы AP | ||||
|       case 6: DEBUGLN("Time from AP"); | ||||
|         if (cfg.WiFimode && !gotNTP) {   // время для local устройств в сети AP лампы (не получили время из интернета) | ||||
|           now.day = data[2]; | ||||
|           now.hour = data[3]; | ||||
|           now.min = data[4]; | ||||
|           now.sec = 0; | ||||
|           now.setMs(0); | ||||
|           DEBUGLN("Got time from master"); | ||||
|         } | ||||
|         gotTime = true; | ||||
|         break; | ||||
|     } | ||||
|     FastLED.clear();    // на всякий случай | ||||
| @@ -166,27 +184,16 @@ void parsing() { | ||||
|  | ||||
| void sendToSlaves(byte data1, byte data2) { | ||||
|   if (cfg.role == GL_MASTER) { | ||||
|     IPAddress ip = WiFi.localIP(); | ||||
|     ip[3] = 255; | ||||
|  | ||||
|     char reply[20]; | ||||
|     char reply[15]; | ||||
|     mString packet(reply, sizeof(reply)); | ||||
|     packet.clear(); | ||||
|     packet += GL_KEY; | ||||
|     packet += ','; | ||||
|     packet += cfg.group; | ||||
|     packet += ",4,"; | ||||
|     packet += data1; | ||||
|     packet += ','; | ||||
|     packet += data2; | ||||
|     packet = packet + "GL,4," + data1 + ',' + data2; | ||||
|  | ||||
|     DEBUG("Sending to Slaves: "); | ||||
|     DEBUGLN(reply); | ||||
|  | ||||
|     FOR_i(0, 3) { | ||||
|       Udp.beginPacket(ip, 8888); | ||||
|       Udp.write(reply); | ||||
|       Udp.endPacket(); | ||||
|       sendUDP(reply); | ||||
|       delay(10); | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -14,14 +14,14 @@ void presetRotation(bool force) { | ||||
| } | ||||
|  | ||||
| void changePreset(int dir) { | ||||
|   if (!cfg.rotation) {    // ручная смена | ||||
|     cfg.curPreset += dir; | ||||
|     if (cfg.curPreset >= cfg.presetAmount) cfg.curPreset = 0; | ||||
|     if (cfg.curPreset < 0) cfg.curPreset = cfg.presetAmount - 1; | ||||
|     holdPresTmr.restart(); | ||||
|     DEBUG("Preset changed to "); | ||||
|     DEBUGLN(cfg.curPreset); | ||||
|   } | ||||
|   //if (!cfg.rotation) {    // ручная смена | ||||
|   cfg.curPreset += dir; | ||||
|   if (cfg.curPreset >= cfg.presetAmount) cfg.curPreset = 0; | ||||
|   if (cfg.curPreset < 0) cfg.curPreset = cfg.presetAmount - 1; | ||||
|   holdPresTmr.restart(); | ||||
|   DEBUG("Preset changed to "); | ||||
|   DEBUGLN(cfg.curPreset); | ||||
|   //} | ||||
| } | ||||
|  | ||||
| void setPreset(byte pres) { | ||||
| @@ -61,13 +61,14 @@ void fade(bool state) { | ||||
| } | ||||
|  | ||||
| void setPower(bool state) { | ||||
|   if (cfg.state != state) EE_updateCfg();   // на сохранение | ||||
|   cfg.state = state; | ||||
|   if (!state) { | ||||
|     delay(100);     // чтобы пролететь мин. частоту обновления | ||||
|     FastLED.clear(); | ||||
|     FastLED.show(); | ||||
|   } | ||||
|   sendToSlaves(0, cfg.state); | ||||
|   if (millis() - udpTmr >= 1000) sendToSlaves(0, cfg.state); // пиздец костыль | ||||
|   DEBUGLN(state ? "Power on" : "Power off"); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -86,15 +86,13 @@ void startWiFi() { | ||||
|   if (!cfg.WiFimode) setupAP();   // режим точки доступа | ||||
|   else setupLocal();              // подключаемся к точке | ||||
|  | ||||
|   DEBUG("UDP port: "); | ||||
|   DEBUGLN(8888); | ||||
|   Udp.begin(8888); | ||||
|   restartUDP(); | ||||
|   FastLED.clear(); | ||||
|   FastLED.show(); | ||||
| } | ||||
|  | ||||
| void setupAP() { | ||||
|   blink8(CRGB::Yellow); | ||||
|   blink16(CRGB::Yellow); | ||||
|   WiFi.disconnect(); | ||||
|   WiFi.mode(WIFI_AP); | ||||
|   delay(100); | ||||
| @@ -131,12 +129,12 @@ void setupLocal() { | ||||
|         leds[count] = CRGB::Yellow; | ||||
|         FastLED.show(); | ||||
|         count += dir; | ||||
|         if (count >= 7 || count <= 0) dir *= -1; | ||||
|         if (count >= 15 || count <= 0) dir *= -1; | ||||
|         delay(50); | ||||
|       } | ||||
|       if (connect) { | ||||
|         connTmr.stop(); | ||||
|         blink8(CRGB::Green); | ||||
|         blink16(CRGB::Green); | ||||
|         server.begin(); | ||||
|         DEBUG("Connected! Local IP: "); | ||||
|         DEBUGLN(WiFi.localIP()); | ||||
| @@ -144,18 +142,13 @@ void setupLocal() { | ||||
|         return; | ||||
|       } else { | ||||
|         DEBUGLN("Failed!"); | ||||
|         blink8(CRGB::Red); | ||||
|         blink16(CRGB::Red); | ||||
|         failCount++; | ||||
|         tmr = millis(); | ||||
|         if (failCount >= 3) { | ||||
|           connTmr.restart();    // попробуем позже | ||||
|           setupAP(); | ||||
|           return; | ||||
|           /*DEBUGLN("Reboot to AP!"); | ||||
|             cfg.WiFimode = 0; | ||||
|             EE_updCfg(); | ||||
|             delay(100); | ||||
|             ESP.restart();*/ | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| @@ -166,11 +159,11 @@ void checkUpdate() { | ||||
|   if (cfg.update) { | ||||
|     if (cfg.version != GL_VERSION) { | ||||
|       cfg.version = GL_VERSION; | ||||
|       blink8(CRGB::Cyan); | ||||
|       blink16(CRGB::Cyan); | ||||
|       DEBUG("Update to"); | ||||
|       DEBUGLN(GL_VERSION); | ||||
|     } else { | ||||
|       blink8(CRGB::Blue); | ||||
|       blink16(CRGB::Blue); | ||||
|       DEBUG("Update to current"); | ||||
|     } | ||||
|     cfg.update = 0; | ||||
| @@ -181,6 +174,15 @@ void checkUpdate() { | ||||
| void tryReconnect() { | ||||
|   if (connTmr.isReady()) { | ||||
|     DEBUGLN("Reconnect"); | ||||
|     setupLocal(); | ||||
|     startWiFi(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| void misc() { | ||||
|   memset(matrixValue, 0, sizeof(matrixValue)); | ||||
|   char GLkey[] = GL_KEY; | ||||
|   portNum = 17; | ||||
|   for (byte i = 0; i < strlen(GLkey); i++) portNum *= GLkey[i]; | ||||
|   portNum %= 15000; | ||||
|   portNum += 50000; | ||||
| } | ||||
|   | ||||
| @@ -1,58 +1,63 @@ | ||||
| void setupTime() { | ||||
|   ntp.setUpdateInterval(NTP_UPD_PRD / 2 * 60000ul);   // меньше в два раза, ибо апдейт вручную | ||||
|   ntp.setUpdateInterval(NTP_UPD_PRD * 60000ul); | ||||
|   ntp.setTimeOffset((cfg.GMT - 13) * 3600); | ||||
|   ntp.setPoolServerName(NTPserver); | ||||
|   if (cfg.WiFimode) { | ||||
|     // если подключены - запрашиваем время с сервера | ||||
|   if (cfg.WiFimode && !connTmr.running()) {     // если успешно подключились к WiFi | ||||
|     ntp.begin(); | ||||
|     if (ntp.update() && !gotNTP) { | ||||
|       gotNTP = true; | ||||
|       DEBUGLN("Got ntp"); | ||||
|     } | ||||
|     if (ntp.update()) gotNTP = true; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // сохраняет счёт времени после обрыва связи | ||||
| // основной тикер времени | ||||
| void timeTicker() { | ||||
|   static timerMillis tmr(10, true); | ||||
|   static timerMillis tmr(30, true); | ||||
|   if (tmr.isReady()) { | ||||
|     updateTime();                               // обновляем время | ||||
|     sendTimeToSlaves();                         // отправляем время слейвам | ||||
|     trnd.update(now.hour, now.min, now.sec);    // обновляем рандомайзер | ||||
|     if (gotNTP || gotTime) checkWorkTime();     // проверяем расписание, если знаем время | ||||
|   } | ||||
| } | ||||
|  | ||||
| void updateTime() { | ||||
|   if (cfg.WiFimode && WiFi.status() == WL_CONNECTED) {  // если вайфай подключен | ||||
|     now.sec = ntp.getSeconds(); | ||||
|     now.min = ntp.getMinutes(); | ||||
|     now.hour = ntp.getHours(); | ||||
|     now.day = ntp.getDay();   // вс 0, сб 6 | ||||
|     now.weekMs = now.getWeekS() * 1000ul + ntp.getMillis(); | ||||
|     now.setMs(ntp.getMillis()); | ||||
|     if (now.min % NTP_UPD_PRD == 0 && now.sec == 0) { | ||||
|       // берём время с интернета каждую NTP_UPD_PRD минуту, ставим флаг что данные с NTP получены, значит мы онлайн | ||||
|       if (ntp.update() && !gotNTP) gotNTP = true; | ||||
|     }     | ||||
|   } else {          // если нет | ||||
|     now.tick();     // тикаем своим счётчиком | ||||
|   } | ||||
|   if (gotNTP || gotTime) checkDawn(); | ||||
| } | ||||
|  | ||||
| void sendTimeToSlaves() { | ||||
|   if (!cfg.WiFimode) {  // если мы AP | ||||
|     static byte prevSec = 0; | ||||
|     if (prevSec != now.sec) {       // новая секунда | ||||
|       prevSec = now.sec; | ||||
|       if (now.min % 5 == 0 && now.sec == 0) sendTime(); // ровно каждые 5 мин отправляем время | ||||
|     if (cfg.WiFimode && WiFi.status() == WL_CONNECTED) {  // если вайфай подключен | ||||
|       now.sec = ntp.getSeconds(); | ||||
|       now.min = ntp.getMinutes(); | ||||
|       now.hour = ntp.getHours(); | ||||
|       now.day = ntp.getDay();   // вс 0, сб 6 | ||||
|       now.weekMs = now.getWeekS() * 1000ul + ntp.getMillis(); | ||||
|       now.setMs(ntp.getMillis()); | ||||
|       if (ntp.update()) gotNTP = true; | ||||
|     } else {          // если вайфай не подключен | ||||
|       now.tick();     // тикаем своим счётчиком | ||||
|     } | ||||
|  | ||||
|     static byte prevSec = 0; | ||||
|     if (prevSec != now.sec) {                   // новая секунда | ||||
|       prevSec = now.sec; | ||||
|       trnd.update(now.hour, now.min, now.sec);  // обновляем рандомайзер | ||||
|  | ||||
|       if (now.sec == 0) {                       // новая минута | ||||
|         if (now.min % 5 == 0) sendTimeToLocals();  // отправляем время каждые 5 мин | ||||
|         if (gotNTP || gotTime) {                // если знаем точное время | ||||
|           checkWorkTime();                      // проверяем расписание | ||||
|           checkDawn();                          // и рассвет | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| void sendTimeToLocals() { | ||||
|   if (!cfg.WiFimode) sendUDP(6, now.day, now.hour, now.min);   // мы - АР | ||||
| } | ||||
|  | ||||
| // установка времени с мобилы | ||||
| void setTime(byte day, byte hour, byte min, byte sec) { | ||||
|   if (!cfg.WiFimode || !gotNTP) {  // если мы AP или не получили NTP | ||||
|     now.day = day; | ||||
|     now.hour = hour; | ||||
|     now.min = min; | ||||
|     now.sec = sec; | ||||
|     now.setMs(0); | ||||
|     gotTime = true; | ||||
|   } | ||||
| } | ||||
|  | ||||
| void checkDawn() { | ||||
|   if (now.sec == 0 && dawn.state[now.day] && !dawnTmr.running()) {    // рассвет включен но не запущен | ||||
|   if (dawn.state[now.day] && !dawnTmr.running()) {    // рассвет включен но не запущен | ||||
|     int dawnMinute = dawn.hour[now.day] * 60 + dawn.minute[now.day] - dawn.time; | ||||
|     if (dawnMinute < 0) dawnMinute += 1440; | ||||
|     if (dawnMinute == now.hour * 60 + now.min) { | ||||
| @@ -75,31 +80,6 @@ void checkWorkTime() { | ||||
|   } | ||||
| } | ||||
|  | ||||
| void sendTime() { | ||||
|   IPAddress ip = WiFi.localIP(); | ||||
|   ip[3] = 255; | ||||
|   char reply[25] = GL_KEY; | ||||
|   mString packet(reply, sizeof(reply)); | ||||
|   packet.clear(); | ||||
|   packet += GL_KEY; | ||||
|   packet += ','; | ||||
|   packet += 0; | ||||
|   packet += ','; | ||||
|   packet += now.day; | ||||
|   packet += ','; | ||||
|   packet += now.hour; | ||||
|   packet += ','; | ||||
|   packet += now.min; | ||||
|   packet += ','; | ||||
|   packet += now.sec; | ||||
|  | ||||
|   DEBUG("Sending time: "); | ||||
|   DEBUGLN(reply); | ||||
|   Udp.beginPacket(ip, 8888); | ||||
|   Udp.write(reply); | ||||
|   Udp.endPacket(); | ||||
| } | ||||
|  | ||||
| bool isWorkTime(byte t, byte from, byte to) { | ||||
|   if (from == to) return 1; | ||||
|   else if (from < to) { | ||||
|   | ||||
| @@ -21,6 +21,9 @@ class timerMillis { | ||||
|       if (_active && millis() - _tmr >= _interval) stop(); | ||||
|       return _active; | ||||
|     } | ||||
|     void force() { | ||||
|       _tmr = millis() - _interval; | ||||
|     } | ||||
|     void reset() { | ||||
|       _tmr = millis(); | ||||
|     } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user