-
Notifications
You must be signed in to change notification settings - Fork 22
/
Copy pathwordclock_esp8266.ino
1157 lines (1036 loc) · 37.2 KB
/
wordclock_esp8266.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/**
* Wordclock 2.0 - Wordclock with ESP8266 and NTP time update
*
* created by techniccontroller 04.12.2021
*
* components:
* - ESP8266
* - Neopixelstrip
*
* Board settings:
* - Board: NodeMCU 1.0 (ESP-12E Module)
* - Flash Size: 4MB (FS:2MB OTA:~1019KB)
* - Upload Speed: 115200
*
*
* with code parts from:
* - Adafruit NeoPixel strandtest.ino, https://github.com/adafruit/Adafruit_NeoPixel/blob/master/examples/strandtest/strandtest.ino
* - Esp8266 und Esp32 webserver https://fipsok.de/
* - https://github.com/pmerlin/PMR-LED-Table/blob/master/tetrisGame.ino
* - https://randomnerdtutorials.com/wifimanager-with-esp8266-autoconnect-custom-parameter-and-manage-your-ssid-and-password/
*
*/
#include "secrets.h" // rename the file example_secrets.h to secrets.h after cloning the project. More information in README.md
#include <LittleFS.h>
#include <Adafruit_GFX.h> // https://github.com/adafruit/Adafruit-GFX-Library
#include <Adafruit_NeoMatrix.h> // https://github.com/adafruit/Adafruit_NeoMatrix
#include <Adafruit_NeoPixel.h> // NeoPixel library used to run the NeoPixel LEDs: https://github.com/adafruit/Adafruit_NeoPixel
#include <ESP8266WiFi.h>
#include <WiFiUdp.h>
#include <ArduinoOTA.h>
#include <ESP8266WebServer.h>
#include "Base64.h" // copied from https://github.com/Xander-Electronics/Base64
#include <DNSServer.h>
#include <WiFiManager.h> //https://github.com/tzapu/WiFiManager WiFi Configuration Magic
#include <EEPROM.h> //from ESP8266 Arduino Core (automatically installed when ESP8266 was installed via Boardmanager)
// own libraries
#include "udplogger.h"
#include "ntp_client_plus.h"
#include "ledmatrix.h"
#include "tetris.h"
#include "snake.h"
#include "pong.h"
// ----------------------------------------------------------------------------------
// CONSTANTS
// ----------------------------------------------------------------------------------
#define EEPROM_SIZE 30 // size of EEPROM to save persistent variables
#define ADR_NM_START_H 0
#define ADR_NM_END_H 4
#define ADR_NM_START_M 8
#define ADR_NM_END_M 12
#define ADR_BRIGHTNESS 16
#define ADR_MC_RED 20
#define ADR_MC_GREEN 22
#define ADR_MC_BLUE 24
#define ADR_STATE 26
#define ADR_NM_ACTIVATED 27
#define ADR_COLSHIFTSPEED 28
#define ADR_COLSHIFTACTIVE 29
#define NEOPIXELPIN 5 // pin to which the NeoPixels are attached
#define BUTTONPIN 14 // pin to which the button is attached
#define LEFT 1
#define RIGHT 2
#define LINE 10
#define RECT 5
#define PERIOD_HEARTBEAT 5000
#define PERIOD_ANIMATION 200
#define PERIOD_TETRIS 50
#define PERIOD_SNAKE 50
#define PERIOD_PONG 10
#define TIMEOUT_LEDDIRECT 5000
#define PERIOD_STATECHANGE 10000
#define PERIOD_NTPUPDATE 30000
#define PERIOD_TIMEVISUUPDATE 1000
#define PERIOD_MATRIXUPDATE 100
#define PERIOD_NIGHTMODECHECK 20000
#define SHORTPRESS 100
#define LONGPRESS 2000
#define CURRENT_LIMIT_LED 2500 // limit the total current sonsumed by LEDs (mA)
#define DEFAULT_SMOOTHING_FACTOR 0.5
// number of colors in colors array
#define NUM_COLORS 7
// own datatype for matrix movement (snake and spiral)
enum direction {right, left, up, down};
// width of the led matrix
#define WIDTH 11
// height of the led matrix
#define HEIGHT 11
// own datatype for state machine states
#define NUM_STATES 6
enum ClockState {st_clock, st_diclock, st_spiral, st_tetris, st_snake, st_pingpong};
const String stateNames[] = {"Clock", "DiClock", "Sprial", "Tetris", "Snake", "PingPong"};
// ports
const unsigned int localPort = 2390;
const unsigned int HTTPPort = 80;
const unsigned int logMulticastPort = 8123;
const unsigned int DNSPort = 53;
// ip addresses for multicast logging
IPAddress logMulticastIP = IPAddress(230, 120, 10, 2);
// ip addresses for Access Point
IPAddress IPAdress_AccessPoint(192,168,10,2);
IPAddress Gateway_AccessPoint(192,168,10,0);
IPAddress Subnetmask_AccessPoint(255,255,255,0);
// hostname
const String hostname = "wordclock";
// URL DNS server
const char WebserverURL[] = "www.wordclock.local";
// ----------------------------------------------------------------------------------
// GLOBAL VARIABLES
// ----------------------------------------------------------------------------------
// Webserver
ESP8266WebServer server(HTTPPort);
//DNS Server
DNSServer DnsServer;
// Wifi server. keep around to support resetting.
WiFiManager wifiManager;
// When we setup the NeoPixel library, we tell it how many pixels, and which pin to use to send signals.
// Note that for older NeoPixel strips you might need to change the third parameter--see the strandtest
// example for more information on possible values.
Adafruit_NeoMatrix matrix = Adafruit_NeoMatrix(WIDTH, HEIGHT+1, NEOPIXELPIN,
NEO_MATRIX_TOP + NEO_MATRIX_LEFT +
NEO_MATRIX_ROWS + NEO_MATRIX_ZIGZAG,
NEO_GRB + NEO_KHZ800);
// seven predefined colors24bit (green, red, yellow, purple, orange, lightgreen, blue)
const uint32_t colors24bit[NUM_COLORS] = {
LEDMatrix::Color24bit(0, 255, 0),
LEDMatrix::Color24bit(255, 0, 0),
LEDMatrix::Color24bit(200, 200, 0),
LEDMatrix::Color24bit(255, 0, 200),
LEDMatrix::Color24bit(255, 128, 0),
LEDMatrix::Color24bit(0, 128, 0),
LEDMatrix::Color24bit(0, 0, 255) };
uint8_t brightness = 40; // current brightness of leds
bool sprialDir = false;
// timestamp variables
long lastheartbeat = millis(); // time of last heartbeat sending
long lastStep = millis(); // time of last animation step
long lastLEDdirect = 0; // time of last direct LED command (=> fall back to normal mode after timeout)
long lastStateChange = millis(); // time of last state change
long lastNTPUpdate = millis() - (PERIOD_NTPUPDATE-3000); // time of last NTP update
long lastAnimationStep = millis(); // time of last Matrix update
long lastNightmodeCheck = millis() - (PERIOD_NIGHTMODECHECK-3000); // time of last nightmode check
long buttonPressStart = 0; // time of push button press start
uint16_t behaviorUpdatePeriod = PERIOD_TIMEVISUUPDATE; // holdes the period in which the behavior should be updated
// Create necessary global objects
UDPLogger logger;
WiFiUDP NTPUDP;
NTPClientPlus ntp = NTPClientPlus(NTPUDP, "pool.ntp.org", 1, true);
LEDMatrix ledmatrix = LEDMatrix(&matrix, brightness, &logger);
Tetris mytetris = Tetris(&ledmatrix, &logger);
Snake mysnake = Snake(&ledmatrix, &logger);
Pong mypong = Pong(&ledmatrix, &logger);
float filterFactor = DEFAULT_SMOOTHING_FACTOR;// stores smoothing factor for led transition
uint8_t currentState = st_clock; // stores current state
bool stateAutoChange = false; // stores state of automatic state change
bool nightMode = false; // stores state of nightmode
bool nightModeActivated = true; // stores if the function nightmode is activated (its not the state of nightmode)
bool ledOff = false; // stores state of led off
uint32_t maincolor_clock = colors24bit[2]; // color of the clock and digital clock
uint32_t maincolor_snake = colors24bit[1]; // color of the random snake animation
bool apmode = false; // stores if WiFi AP mode is active
bool dynColorShiftActive = false; // stores if dynamic color shift is active
uint8_t dynColorShiftPhase = 0; // stores the phase of the dynamic color shift
uint8_t dynColorShiftSpeed = 1; // stores the speed of the dynamic color shift -> used to calc update period
// nightmode settings
uint8_t nightModeStartHour = 22;
uint8_t nightModeStartMin = 0;
uint8_t nightModeEndHour = 7;
uint8_t nightModeEndMin = 0;
// Watchdog counter to trigger restart if NTP update was not possible 30 times in a row (5min)
int watchdogCounter = 30;
bool waitForTimeAfterReboot = false; // wait for time update after reboot
// ----------------------------------------------------------------------------------
// SETUP
// ----------------------------------------------------------------------------------
void setup() {
// put your setup code here, to run once:
Serial.begin(115200);
delay(100);
Serial.println();
Serial.printf("\nSketchname: %s\nBuild: %s\n", (__FILE__), (__TIMESTAMP__));
Serial.println();
//Init EEPROM
EEPROM.begin(EEPROM_SIZE);
// configure button pin as input
pinMode(BUTTONPIN, INPUT_PULLUP);
// setup Matrix LED functions
ledmatrix.setupMatrix();
ledmatrix.setCurrentLimit(CURRENT_LIMIT_LED);
if(!ESP.getResetReason().equals("Software/System restart")){
// Turn on minutes leds (blue)
ledmatrix.setMinIndicator(15, colors24bit[6]);
ledmatrix.drawOnMatrixInstant();
}
/** Use WiFiMaanger for handling initial Wifi setup **/
// Local intialization. Once its business is done, there is no need to keep it around
// Uncomment and run it once, if you want to erase all the stored information
//wifiManager.resetSettings();
// set custom ip for portal
//wifiManager.setAPStaticIPConfig(IPAdress_AccessPoint, Gateway_AccessPoint, Subnetmask_AccessPoint);
// fetches ssid and pass from eeprom and tries to connect
// if it does not connect it starts an access point with the specified name
// here "wordclockAP"
// and goes into a blocking loop awaiting configuration
wifiManager.autoConnect(AP_SSID);
// if you get here you have connected to the WiFi
Serial.println("Connected.");
Serial.println("IP address: ");
Serial.println(WiFi.localIP());
if(!ESP.getResetReason().equals("Software/System restart")){
// Turn off minutes leds
ledmatrix.setMinIndicator(15, 0);
ledmatrix.drawOnMatrixInstant();
}
/** (alternative) Use directly STA/AP Mode of ESP8266 **/
/*
// We start by connecting to a WiFi network
Serial.print("Connecting to ");
Serial.println(WIFI_SSID);
// We start by connecting to a WiFi network
WiFi.mode(WIFI_STA);
//Set new hostname
WiFi.hostname(hostname.c_str());
WiFi.begin(WIFI_SSID, WIFI_PASS);
//wifi_station_set_hostname("esplamp");
int timeoutcounter = 0;
while (WiFi.status() != WL_CONNECTED && timeoutcounter < 30) {
ledmatrix.setMinIndicator(15, colors24bit[6]);
ledmatrix.drawOnMatrixInstant();
delay(250);
ledmatrix.setMinIndicator(15, 0);
ledmatrix.drawOnMatrixInstant();
delay(250);
Serial.print(".");
timeoutcounter++;
}
// start request of program
if (WiFi.status() == WL_CONNECTED) { //Check WiFi connection status
Serial.println("");
Serial.println("WiFi connected");
Serial.println("IP address: ");
Serial.println(WiFi.localIP());
WiFi.setAutoReconnect(true);
WiFi.persistent(true);
} else {
// no wifi found -> open access point
WiFi.mode(WIFI_AP);
WiFi.softAPConfig(IPAdress_AccessPoint, Gateway_AccessPoint, Subnetmask_AccessPoint);
WiFi.softAP(AP_SSID, AP_PASS);
apmode = true;
// start DNS Server
DnsServer.setTTL(300);
DnsServer.start(DNSPort, WebserverURL, IPAdress_AccessPoint);
IPAddress myIP = WiFi.softAPIP();
Serial.print("AP IP address: ");
Serial.println(myIP);
}*/
// init ESP8266 File manager (LittleFS)
setupFS();
// setup OTA
setupOTA(hostname);
server.on("/cmd", handleCommand); // process commands
server.on("/data", handleDataRequest); // process datarequests
server.on("/leddirect", HTTP_POST, handleLEDDirect); // Call the 'handleLEDDirect' function when a POST request is made to URI "/leddirect"
server.begin();
// create UDP Logger to send logging messages via UDP multicast
logger = UDPLogger(WiFi.localIP(), logMulticastIP, logMulticastPort);
logger.setName("Wordclock 2.0");
logger.logString("Start program\n");
delay(10);
logger.logString("Sketchname: "+ String(__FILE__));
delay(10);
logger.logString("Build: " + String(__TIMESTAMP__));
delay(10);
logger.logString("IP: " + WiFi.localIP().toString());
delay(10);
logger.logString("Reset Reason: " + ESP.getResetReason());
// setup NTP
ntp.setupNTPClient();
logger.logString("NTP running");
logger.logString("Time: " + ntp.getFormattedTime());
logger.logString("TimeOffset (seconds): " + String(ntp.getTimeOffset()));
// load persistent variables from EEPROM
loadMainColorFromEEPROM();
loadCurrentStateFromEEPROM();
loadNightmodeSettingsFromEEPROM();
loadBrightnessSettingsFromEEPROM();
loadColorShiftStateFromEEPROM();
if(!ESP.getResetReason().equals("Software/System restart")){
// test quickly each LED
for(int r = 0; r < HEIGHT; r++){
for(int c = 0; c < WIDTH; c++){
matrix.fillScreen(0);
matrix.drawPixel(c, r, LEDMatrix::color24to16bit(colors24bit[2]));
matrix.show();
delay(10);
}
}
// clear Matrix
matrix.fillScreen(0);
matrix.show();
delay(200);
// display IP
uint8_t address = WiFi.localIP()[3];
ledmatrix.printChar(1, 0, 'I', maincolor_clock);
ledmatrix.printChar(5, 0, 'P', maincolor_clock);
ledmatrix.printNumber(0, 6, (address/100), maincolor_clock);
ledmatrix.printNumber(4, 6, (address/10)%10, maincolor_clock);
ledmatrix.printNumber(8, 6, address%10, maincolor_clock);
ledmatrix.drawOnMatrixInstant();
delay(2000);
// clear matrix
ledmatrix.gridFlush();
ledmatrix.drawOnMatrixInstant();
}
else {
waitForTimeAfterReboot = true;
}
// run the entry action for the initial state
entryAction(currentState);
}
// ----------------------------------------------------------------------------------
// LOOP
// ----------------------------------------------------------------------------------
void loop() {
// handle OTA
handleOTA();
// handle Webserver
server.handleClient();
// send regularly heartbeat messages via UDP multicast
if(millis() - lastheartbeat > PERIOD_HEARTBEAT){
logger.logString("Heartbeat, state: " + stateNames[currentState] + ", FreeHeap: " + ESP.getFreeHeap() + ", HeapFrag: " + ESP.getHeapFragmentation() + ", MaxFreeBlock: " + ESP.getMaxFreeBlockSize() + "\n");
lastheartbeat = millis();
// Check wifi status (only if no apmode)
if(!apmode && WiFi.status() != WL_CONNECTED){
Serial.println("connection lost");
ledmatrix.gridAddPixel(0, 5, colors24bit[1]);
ledmatrix.drawOnMatrixInstant();
}
}
// handle state behaviours (trigger loopCycles of different states depending on current state)
if(!nightMode && !ledOff && (millis() - lastStep > behaviorUpdatePeriod) && (millis() - lastLEDdirect > TIMEOUT_LEDDIRECT)){
updateStateBehavior(currentState);
lastStep = millis();
}
// Turn off LEDs if ledOff is true or nightmode is active
if((ledOff || nightMode) && !waitForTimeAfterReboot){
ledmatrix.gridFlush();
ledmatrix.drawOnMatrixInstant();
}
// periodically write colors to matrix
if(millis() - lastAnimationStep > PERIOD_MATRIXUPDATE && !waitForTimeAfterReboot){
ledmatrix.drawOnMatrixSmooth(filterFactor);
lastAnimationStep = millis();
}
// handle button press
handleButton();
// handle state changes
if(stateAutoChange && (millis() - lastStateChange > PERIOD_STATECHANGE) && !nightMode && !ledOff){
// increment state variable and trigger state change
stateChange((currentState + 1) % NUM_STATES, false);
// save last automatic state change
lastStateChange = millis();
}
// NTP time update
if(millis() - lastNTPUpdate > PERIOD_NTPUPDATE){
int res = ntp.updateNTP();
if(res == 0){
ntp.calcDate();
logger.logString("NTP-Update successful");
logger.logString("Time: " + ntp.getFormattedTime());
logger.logString("Date: " + ntp.getFormattedDate());
logger.logString("Day of Week (Mon=1, Sun=7): " + String(ntp.getDayOfWeek()));
logger.logString("TimeOffset (seconds): " + String(ntp.getTimeOffset()));
logger.logString("Summertime: " + String(ntp.updateSWChange()));
lastNTPUpdate = millis();
watchdogCounter = 30;
checkNightmode();
if(waitForTimeAfterReboot && !nightMode){
// update mode (e.g. write the current time onto the matrix) first time after reboot
entryAction(currentState);
updateStateBehavior(currentState);
ledmatrix.drawOnMatrixInstant();
}
waitForTimeAfterReboot = false;
}
else if(res == -1){
logger.logString("NTP-Update not successful. Reason: Timeout");
lastNTPUpdate += 10000;
watchdogCounter--;
}
else if(res == 1){
logger.logString("NTP-Update not successful. Reason: Too large time difference");
logger.logString("Time: " + ntp.getFormattedTime());
logger.logString("Date: " + ntp.getFormattedDate());
logger.logString("Day of Week (Mon=1, Sun=7): " + ntp.getDayOfWeek());
logger.logString("TimeOffset (seconds): " + String(ntp.getTimeOffset()));
logger.logString("Summertime: " + String(ntp.updateSWChange()));
lastNTPUpdate += 10000;
watchdogCounter--;
}
else {
logger.logString("NTP-Update not successful. Reason: NTP time not valid (<1970)");
lastNTPUpdate += 10000;
watchdogCounter--;
}
logger.logString("Watchdog Counter: " + String(watchdogCounter));
if(watchdogCounter <= 0){
logger.logString("Trigger restart due to watchdog...");
delay(100);
ESP.restart();
}
}
// check if nightmode need to be activated
if(millis() - lastNightmodeCheck > PERIOD_NIGHTMODECHECK && !waitForTimeAfterReboot){
checkNightmode();
lastNightmodeCheck = millis();
}
}
// ----------------------------------------------------------------------------------
// OTHER FUNCTIONS
// ----------------------------------------------------------------------------------
/**
* @brief Update mode behaviour depending on current state
*/
void updateStateBehavior(uint8_t state){
switch(state){
// state clock
case st_clock:
{
if(dynColorShiftActive){
dynColorShiftPhase = (dynColorShiftPhase + 1) % 256;
ledmatrix.setDynamicColorShiftPhase(dynColorShiftPhase);
filterFactor = 1.0; // no smoothing
behaviorUpdatePeriod = PERIOD_TIMEVISUUPDATE / dynColorShiftSpeed;
} else {
ledmatrix.setDynamicColorShiftPhase(-1);
filterFactor = DEFAULT_SMOOTHING_FACTOR;
behaviorUpdatePeriod = PERIOD_TIMEVISUUPDATE;
}
uint8_t hours = ntp.getHours24();
uint8_t minutes = ntp.getMinutes();
static uint8_t lastMinutes = 0;
static String timeAsString = "";
if(lastMinutes != minutes){
timeAsString = timeToString(hours, minutes);
lastMinutes = minutes;
}
showStringOnClock(timeAsString, maincolor_clock);
drawMinuteIndicator(minutes, maincolor_clock);
}
break;
// state diclock
case st_diclock:
{
int hours = ntp.getHours24();
int minutes = ntp.getMinutes();
showDigitalClock(hours, minutes, maincolor_clock);
}
break;
// state spiral
case st_spiral:
{
int res = spiral(false, sprialDir, WIDTH-6);
if(res && sprialDir == 0){
// change spiral direction to closing (draw empty leds)
sprialDir = 1;
// init spiral with new spiral direction
spiral(true, sprialDir, WIDTH-6);
}else if(res && sprialDir == 1){
// reset spiral direction to normal drawing leds
sprialDir = 0;
// init spiral with new spiral direction
spiral(true, sprialDir, WIDTH-6);
}
}
break;
// state tetris
case st_tetris:
{
if(stateAutoChange){
randomtetris(false);
}
else{
mytetris.loopCycle();
}
}
break;
// state snake
case st_snake:
{
if(stateAutoChange){
ledmatrix.gridFlush();
int res = randomsnake(false, 8, maincolor_snake, -1);
if(res){
// init snake for next run
randomsnake(true, 8, maincolor_snake, -1);
}
}
else{
mysnake.loopCycle();
}
}
break;
// state pingpong
case st_pingpong:
{
mypong.loopCycle();
}
break;
}
}
/**
* @brief Check if nightmode should be activated
*
*/
void checkNightmode(){
logger.logString("Check nightmode");
int hours = ntp.getHours24();
int minutes = ntp.getMinutes();
nightMode = false; // Initial assumption
// Convert all times to minutes for easier comparison
int currentTimeInMinutes = hours * 60 + minutes;
int startInMinutes = nightModeStartHour * 60 + nightModeStartMin;
int endInMinutes = nightModeEndHour * 60 + nightModeEndMin;
if (startInMinutes < endInMinutes && nightModeActivated) { // Same day scenario
if (startInMinutes < currentTimeInMinutes && currentTimeInMinutes < endInMinutes) {
nightMode = true;
logger.logString("Nightmode active");
}
} else if (startInMinutes > endInMinutes && nightModeActivated) { // Overnight scenario
if (currentTimeInMinutes >= startInMinutes || currentTimeInMinutes < endInMinutes) {
nightMode = true;
logger.logString("Nightmode active");
}
}
}
/**
* @brief call entry action of given state
*
* @param state
*/
void entryAction(uint8_t state){
filterFactor = DEFAULT_SMOOTHING_FACTOR;
switch(state){
case st_clock:
behaviorUpdatePeriod = PERIOD_TIMEVISUUPDATE;
break;
case st_diclock:
behaviorUpdatePeriod = PERIOD_TIMEVISUUPDATE;
ledmatrix.setDynamicColorShiftPhase(-1); // disable dyn. color shift
break;
case st_spiral:
behaviorUpdatePeriod = PERIOD_ANIMATION;
ledmatrix.setDynamicColorShiftPhase(-1); // disable dyn. color shift
// Init spiral with normal drawing mode
sprialDir = 0;
spiral(true, sprialDir, WIDTH-6);
break;
case st_tetris:
ledmatrix.setDynamicColorShiftPhase(-1); // disable dyn. color shift
filterFactor = 1.0; // no smoothing
if(stateAutoChange){
behaviorUpdatePeriod = PERIOD_ANIMATION;
randomtetris(true);
}
else{
behaviorUpdatePeriod = PERIOD_TETRIS;
mytetris.ctrlStart();
}
break;
case st_snake:
ledmatrix.setDynamicColorShiftPhase(-1); // disable dyn. color shift
if(stateAutoChange){
behaviorUpdatePeriod = PERIOD_ANIMATION;
randomsnake(true, 8, colors24bit[1], -1);
}
else{
behaviorUpdatePeriod = PERIOD_SNAKE;
filterFactor = 1.0; // no smoothing
mysnake.initGame();
}
break;
case st_pingpong:
behaviorUpdatePeriod = PERIOD_PONG;
ledmatrix.setDynamicColorShiftPhase(-1); // disable dyn. color shift
if(stateAutoChange){
mypong.initGame(2);
}
else{
filterFactor = 1.0; // no smoothing
mypong.initGame(1);
}
break;
}
}
/**
* @brief execute a state change to given newState
*
* @param newState the new state to be changed to
* @param persistant if true, the state will be saved to EEPROM
*/
void stateChange(uint8_t newState, bool persistant){
if(ledOff){
ledOff = false;
}
// first clear matrix
ledmatrix.gridFlush();
// set new state
currentState = newState;
entryAction(currentState);
logger.logString("State change to: " + stateNames[currentState]);
if(persistant){
// save state to EEPROM
EEPROM.write(ADR_STATE, currentState);
EEPROM.commit();
}
}
/**
* @brief Handler for POST requests to /leddirect.
*
* Allows the control of all LEDs from external source.
* It will overwrite the normal program for 5 seconds.
* A 11x11 picture can be sent as base64 encoded string to be displayed on matrix.
*
*/
void handleLEDDirect() {
if (server.method() != HTTP_POST) {
server.send(405, "text/plain", "Method Not Allowed");
} else {
String message = "POST data was:\n";
/*logger.logString(message);
delay(10);
for (uint8_t i = 0; i < server.args(); i++) {
message += " " + server.argName(i) + ": " + server.arg(i) + "\n";
logger.logString(server.arg(i));
delay(10);
}*/
if(server.args() == 1){
String data = String(server.arg(0));
int dataLength = data.length();
//char byteArray[dataLength];
//data.toCharArray(byteArray, dataLength);
// base64 decoding
char base64data[dataLength];
data.toCharArray(base64data, dataLength);
int base64dataLen = dataLength;
int decodedLength = Base64.decodedLength(base64data, base64dataLen);
char byteArray[decodedLength];
Base64.decode(byteArray, base64data, base64dataLen);
/*for(int i = 0; i < 10; i++){
logger.logString(String((int)(byteArray[i])));
delay(10);
}*/
for(int i = 0; i < dataLength; i += 4) {
uint8_t red = byteArray[i]; // red
uint8_t green = byteArray[i + 1]; // green
uint8_t blue = byteArray[i + 2]; // blue
ledmatrix.gridAddPixel((i/4) % WIDTH, (i/4) / HEIGHT, LEDMatrix::Color24bit(red, green, blue));
}
ledmatrix.drawOnMatrixInstant();
lastLEDdirect = millis();
}
server.send(200, "text/plain", message);
}
}
/**
* @brief Check button commands
*
*/
void handleButton(){
static bool lastButtonState = false;
bool buttonPressed = !digitalRead(BUTTONPIN);
// check rising edge
if(buttonPressed == true && lastButtonState == false){
// button press start
logger.logString("Button press started");
buttonPressStart = millis();
}
// check falling edge
if(buttonPressed == false && lastButtonState == true){
// button press ended
if((millis() - buttonPressStart) > LONGPRESS){
// longpress -> nightmode
logger.logString("Button press ended - longpress");
ledOff = true;
}
else if((millis() - buttonPressStart) > SHORTPRESS){
// shortpress -> state change
logger.logString("Button press ended - shortpress");
if(ledOff){
ledOff = false;
}else{
stateChange((currentState + 1) % NUM_STATES, true);
}
}
}
lastButtonState = buttonPressed;
}
/**
* @brief Set main color
*
*/
void setMainColor(uint8_t red, uint8_t green, uint8_t blue){
maincolor_clock = LEDMatrix::Color24bit(red, green, blue);
EEPROM.put(ADR_MC_RED, red);
EEPROM.put(ADR_MC_GREEN, green);
EEPROM.put(ADR_MC_BLUE, blue);
EEPROM.commit();
}
/**
* @brief Load maincolor from EEPROM
*
*/
void loadMainColorFromEEPROM(){
uint8_t red = EEPROM.read(ADR_MC_RED);
uint8_t green = EEPROM.read(ADR_MC_GREEN);
uint8_t blue = EEPROM.read(ADR_MC_BLUE);
if(int(red) + int(green) + int(blue) < 50){
maincolor_clock = colors24bit[2];
}else{
maincolor_clock = LEDMatrix::Color24bit(red, green, blue);
}
}
/**
* @brief Load the current state from EEPROM
*
*/
void loadCurrentStateFromEEPROM(){
currentState = EEPROM.read(ADR_STATE);
if(currentState >= NUM_STATES){
currentState = st_clock;
EEPROM.write(ADR_STATE, currentState);
EEPROM.commit();
}
}
/**
* @brief Load the nightmode settings from EEPROM
*/
void loadNightmodeSettingsFromEEPROM()
{
nightModeStartHour = EEPROM.read(ADR_NM_START_H);
nightModeStartMin = EEPROM.read(ADR_NM_START_M);
nightModeEndHour = EEPROM.read(ADR_NM_END_H);
nightModeEndMin = EEPROM.read(ADR_NM_END_M);
nightModeActivated = EEPROM.read(ADR_NM_ACTIVATED);
if(nightModeStartHour < 0 || nightModeStartHour > 23) nightModeStartHour = 22;
if(nightModeStartMin < 0 || nightModeStartMin > 59) nightModeStartMin = 0;
if(nightModeEndHour < 0 || nightModeEndHour > 23) nightModeEndHour = 7;
if(nightModeEndMin < 0 || nightModeEndMin > 59) nightModeEndMin = 0;
logger.logString("Nightmode activated: " + String(nightModeActivated));
logger.logString("Nightmode starts at: " + String(nightModeStartHour) + ":" + String(nightModeStartMin));
logger.logString("Nightmode ends at: " + String(nightModeEndHour) + ":" + String(nightModeEndMin));
}
/**
* @brief Load the brightness settings from EEPROM
*
* lower limit is 10 so that the LEDs are not completely off
*/
void loadBrightnessSettingsFromEEPROM()
{
brightness = EEPROM.read(ADR_BRIGHTNESS);
if(brightness < 10) brightness = 10;
logger.logString("Brightness: " + String(brightness));
ledmatrix.setBrightness(brightness);
}
/**
* @brief load the color shift speed from EEPROM
*
*/
void loadColorShiftStateFromEEPROM()
{
dynColorShiftSpeed = EEPROM.read(ADR_COLSHIFTSPEED);
if (dynColorShiftSpeed == 0) dynColorShiftSpeed = 1;
logger.logString("ColorShiftSpeed: " + String(dynColorShiftSpeed));
dynColorShiftActive = EEPROM.read(ADR_COLSHIFTACTIVE);
logger.logString("ColorShiftActive: " + String(dynColorShiftActive));
}
/**
* @brief Handler for handling commands sent to "/cmd" url
*
*/
void handleCommand() {
// receive command and handle accordingly
for (uint8_t i = 0; i < server.args(); i++) {
String log_str = "Command received: " + server.argName(i) + " " + server.arg(i);
logger.logString(log_str);
}
if (server.argName(0) == "led") // the parameter which was sent to this server is led color
{
String colorstr = server.arg(0) + "-";
String redstr = split(colorstr, '-', 0);
String greenstr= split(colorstr, '-', 1);
String bluestr = split(colorstr, '-', 2);
logger.logString(colorstr);
logger.logString("r: " + String(redstr.toInt()));
logger.logString("g: " + String(greenstr.toInt()));
logger.logString("b: " + String(bluestr.toInt()));
// set new main color
setMainColor(redstr.toInt(), greenstr.toInt(), bluestr.toInt());
}
else if (server.argName(0) == "mode") // the parameter which was sent to this server is mode change
{
String modestr = server.arg(0);
logger.logString("Mode change via Webserver to: " + modestr);
// set current mode/state accordant sent mode
if(modestr == "clock"){
stateChange(st_clock, true);
}
else if(modestr == "diclock"){
stateChange(st_diclock, true);
}
else if(modestr == "spiral"){
stateChange(st_spiral, true);
}
else if(modestr == "tetris"){
stateChange(st_tetris, true);
}
else if(modestr == "snake"){
stateChange(st_snake, true);
}
else if(modestr == "pingpong"){
stateChange(st_pingpong, true);
}
}
else if(server.argName(0) == "ledoff"){
String modestr = server.arg(0);
logger.logString("LED off change via Webserver to: " + modestr);
if(modestr == "1") ledOff = true;
else ledOff = false;
}
else if(server.argName(0) == "nightmodeactivated"){
String modestr = server.arg(0);
logger.logString("nightModeActivated change via Webserver to: " + modestr);
if(modestr == "1") nightModeActivated = true;
else nightModeActivated = false;
EEPROM.write(ADR_NM_ACTIVATED, nightModeActivated);
EEPROM.commit();
checkNightmode();
}
else if(server.argName(0) == "setting"){
String timestr = server.arg(0) + "-";
logger.logString("Nightmode setting change via Webserver to: " + timestr);
nightModeStartHour = split(timestr, '-', 0).toInt();
nightModeStartMin = split(timestr, '-', 1).toInt();
nightModeEndHour = split(timestr, '-', 2).toInt();
nightModeEndMin = split(timestr, '-', 3).toInt();
brightness = split(timestr, '-', 4).toInt();
dynColorShiftSpeed = split(timestr, '-', 5).toInt();
if(nightModeStartHour < 0 || nightModeStartHour > 23) nightModeStartHour = 22;
if(nightModeStartMin < 0 || nightModeStartMin > 59) nightModeStartMin = 0;
if(nightModeEndHour < 0 || nightModeEndHour > 23) nightModeEndHour = 7;
if(nightModeEndMin < 0 || nightModeEndMin > 59) nightModeEndMin = 0;
if(brightness < 10) brightness = 10;
if(dynColorShiftSpeed == 0) dynColorShiftSpeed = 1;
EEPROM.write(ADR_NM_START_H, nightModeStartHour);
EEPROM.write(ADR_NM_START_M, nightModeStartMin);
EEPROM.write(ADR_NM_END_H, nightModeEndHour);
EEPROM.write(ADR_NM_END_M, nightModeEndMin);
EEPROM.write(ADR_BRIGHTNESS, brightness);
EEPROM.write(ADR_COLSHIFTSPEED, dynColorShiftSpeed);
EEPROM.commit();
logger.logString("Nightmode starts at: " + String(nightModeStartHour) + ":" + String(nightModeStartMin));
logger.logString("Nightmode ends at: " + String(nightModeEndHour) + ":" + String(nightModeEndMin));
logger.logString("Brightness: " + String(brightness));
logger.logString("ColorShiftSpeed: " + String(dynColorShiftSpeed));
ledmatrix.setBrightness(brightness);
lastNightmodeCheck = millis() - PERIOD_NIGHTMODECHECK;
}
else if (server.argName(0) == "resetwifi"){
wifiManager.resetSettings();
// run LED test.
for(int r = 0; r < HEIGHT; r++){
for(int c = 0; c < WIDTH; c++){
matrix.fillScreen(0);
matrix.drawPixel(c, r, LEDMatrix::color24to16bit(colors24bit[2]));
matrix.show();
delay(10);
}
}
// clear Matrix
matrix.fillScreen(0);
matrix.show();
delay(200);