2016-12-31 00:38:51 +01:00
|
|
|
/*
|
|
|
|
* Methods to handle saving and loading to non-volatile memory
|
2018-02-28 00:27:10 +01:00
|
|
|
* EEPROM Map: https://github.com/Aircoookie/WLED/wiki/EEPROM-Map
|
2016-12-31 00:38:51 +01:00
|
|
|
*/
|
|
|
|
|
2018-02-28 00:27:10 +01:00
|
|
|
#define EEPSIZE 3072
|
|
|
|
|
|
|
|
//eeprom Version code, enables default settings instead of 0 init on update
|
2018-12-04 00:58:06 +01:00
|
|
|
#define EEPVER 10
|
2018-02-28 00:27:10 +01:00
|
|
|
//0 -> old version, default
|
|
|
|
//1 -> 0.4p 1711272 and up
|
|
|
|
//2 -> 0.4p 1711302 and up
|
|
|
|
//3 -> 0.4 1712121 and up
|
|
|
|
//4 -> 0.5.0 and up
|
2018-03-15 12:04:14 +01:00
|
|
|
//5 -> 0.5.1 and up
|
|
|
|
//6 -> 0.6.0 and up
|
2018-08-11 22:57:13 +02:00
|
|
|
//7 -> 0.7.1 and up
|
2018-10-04 16:50:12 +02:00
|
|
|
//8 -> 0.8.0-a and up
|
|
|
|
//9 -> 0.8.0
|
2018-12-04 00:58:06 +01:00
|
|
|
//10-> 0.8.2
|
2017-12-28 00:37:13 +01:00
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-08-11 22:57:13 +02:00
|
|
|
/*
|
|
|
|
* Erase all configuration data
|
|
|
|
*/
|
2016-11-19 19:39:17 +01:00
|
|
|
void clearEEPROM()
|
|
|
|
{
|
2017-12-28 00:37:13 +01:00
|
|
|
for (int i = 0; i < EEPSIZE; i++)
|
2016-11-19 19:39:17 +01:00
|
|
|
{
|
|
|
|
EEPROM.write(i, 0);
|
|
|
|
}
|
|
|
|
EEPROM.commit();
|
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
void writeStringToEEPROM(uint16_t pos, char* str, uint16_t len)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < len; ++i)
|
|
|
|
{
|
|
|
|
EEPROM.write(pos + i, str[i]);
|
|
|
|
if (str[i] == 0) return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
void readStringFromEEPROM(uint16_t pos, char* str, uint16_t len)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < len; ++i)
|
|
|
|
{
|
|
|
|
str[i] = EEPROM.read(pos + i);
|
|
|
|
if (str[i] == 0) return;
|
|
|
|
}
|
|
|
|
str[len] = 0; //make sure every string is properly terminated. str must be at least len +1 big.
|
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-08-11 22:57:13 +02:00
|
|
|
/*
|
|
|
|
* Write configuration to flash
|
|
|
|
*/
|
2016-11-19 19:39:17 +01:00
|
|
|
void saveSettingsToEEPROM()
|
|
|
|
{
|
|
|
|
if (EEPROM.read(233) != 233) //set no first boot flag
|
|
|
|
{
|
|
|
|
clearEEPROM();
|
|
|
|
EEPROM.write(233, 233);
|
|
|
|
}
|
2017-11-28 00:21:29 +01:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM( 0, clientSSID, 32);
|
|
|
|
writeStringToEEPROM( 32, clientPass, 64);
|
|
|
|
writeStringToEEPROM( 96, cmDNS, 32);
|
|
|
|
writeStringToEEPROM(128, apSSID, 32);
|
|
|
|
writeStringToEEPROM(160, apPass, 64);
|
|
|
|
|
2018-11-21 23:28:20 +01:00
|
|
|
EEPROM.write(224, nightlightDelayMinsDefault);
|
2016-11-20 00:07:04 +01:00
|
|
|
EEPROM.write(225, nightlightFade);
|
2017-05-08 21:46:04 +02:00
|
|
|
EEPROM.write(226, notifyDirectDefault);
|
2018-03-14 13:16:28 +01:00
|
|
|
EEPROM.write(227, apChannel);
|
|
|
|
EEPROM.write(228, apHide);
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(229, ledCount & 0xFF);
|
2016-11-20 01:47:15 +01:00
|
|
|
EEPROM.write(230, notifyButton);
|
2018-03-06 23:47:08 +01:00
|
|
|
EEPROM.write(231, notifyTwice);
|
2016-11-19 19:39:17 +01:00
|
|
|
EEPROM.write(232, buttonEnabled);
|
2016-11-20 01:47:15 +01:00
|
|
|
//233 reserved for first boot flag
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-07-21 23:21:07 +02:00
|
|
|
for (int i = 0; i<4; i++) //ip addresses
|
|
|
|
{
|
|
|
|
EEPROM.write(234+i, staticIP[i]);
|
|
|
|
EEPROM.write(238+i, staticGateway[i]);
|
|
|
|
EEPROM.write(242+i, staticSubnet[i]);
|
|
|
|
}
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
EEPROM.write(246, colS[0]);
|
|
|
|
EEPROM.write(247, colS[1]);
|
|
|
|
EEPROM.write(248, colS[2]);
|
|
|
|
EEPROM.write(249, briS);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-02-20 22:29:48 +01:00
|
|
|
EEPROM.write(250, receiveNotificationBrightness);
|
2016-11-19 19:39:17 +01:00
|
|
|
EEPROM.write(251, fadeTransition);
|
2018-03-06 23:47:08 +01:00
|
|
|
EEPROM.write(252, reverseMode);
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(253, transitionDelayDefault & 0xFF);
|
2018-04-11 23:50:35 +02:00
|
|
|
EEPROM.write(254, (transitionDelayDefault >> 8) & 0xFF);
|
2018-02-20 22:29:48 +01:00
|
|
|
EEPROM.write(255, briMultiplier);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2016-11-20 01:47:15 +01:00
|
|
|
//255,250,231,230,226 notifier bytes
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(256, otaPass, 32);
|
|
|
|
|
2018-02-20 22:29:48 +01:00
|
|
|
EEPROM.write(288, nightlightTargetBri);
|
2017-11-20 00:07:37 +01:00
|
|
|
EEPROM.write(289, otaLock);
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(290, udpPort & 0xFF);
|
2016-12-11 20:11:14 +01:00
|
|
|
EEPROM.write(291, (udpPort >> 8) & 0xFF);
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(292, serverDescription, 32);
|
|
|
|
|
2016-12-29 00:03:58 +01:00
|
|
|
EEPROM.write(324, effectDefault);
|
|
|
|
EEPROM.write(325, effectSpeedDefault);
|
2018-01-27 23:28:20 +01:00
|
|
|
EEPROM.write(326, effectIntensityDefault);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2016-12-29 00:03:58 +01:00
|
|
|
EEPROM.write(327, ntpEnabled);
|
2018-03-06 23:47:08 +01:00
|
|
|
EEPROM.write(328, currentTimezone);
|
|
|
|
EEPROM.write(329, useAMPM);
|
2017-02-01 19:25:36 +01:00
|
|
|
EEPROM.write(330, useGammaCorrectionBri);
|
|
|
|
EEPROM.write(331, useGammaCorrectionRGB);
|
2017-02-24 23:21:48 +01:00
|
|
|
EEPROM.write(332, overlayDefault);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2017-02-24 23:21:48 +01:00
|
|
|
EEPROM.write(333, alexaEnabled);
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(334, alexaInvocationName, 32);
|
2018-09-15 17:29:01 +02:00
|
|
|
EEPROM.write(366, notifyAlexa);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
EEPROM.write(367, (arlsOffset>=0));
|
2017-02-24 23:21:48 +01:00
|
|
|
EEPROM.write(368, abs(arlsOffset));
|
2017-03-07 22:05:18 +01:00
|
|
|
EEPROM.write(369, turnOnAtBoot);
|
2017-05-07 23:51:42 +02:00
|
|
|
EEPROM.write(370, useHSBDefault);
|
2019-02-05 21:53:39 +01:00
|
|
|
EEPROM.write(371, colS[3]); //white default
|
2017-10-12 22:20:37 +02:00
|
|
|
EEPROM.write(372, useRGBW);
|
2018-09-06 02:05:56 +02:00
|
|
|
EEPROM.write(373, effectPaletteDefault);
|
2018-09-08 16:21:44 +02:00
|
|
|
EEPROM.write(374, strip.paletteFade);
|
2017-11-20 00:07:37 +01:00
|
|
|
EEPROM.write(375, apWaitTimeSecs);
|
|
|
|
EEPROM.write(376, recoveryAPDisabled);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2017-11-28 00:21:29 +01:00
|
|
|
EEPROM.write(377, EEPVER); //eeprom was updated to latest
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
EEPROM.write(378, colSecS[0]);
|
|
|
|
EEPROM.write(379, colSecS[1]);
|
|
|
|
EEPROM.write(380, colSecS[2]);
|
2019-02-05 21:53:39 +01:00
|
|
|
EEPROM.write(381, colSecS[3]);
|
2018-09-11 00:20:12 +02:00
|
|
|
EEPROM.write(382, strip.paletteBlend);
|
2018-11-28 12:24:32 +01:00
|
|
|
EEPROM.write(383, strip.colorOrder);
|
2018-11-21 23:28:20 +01:00
|
|
|
|
|
|
|
EEPROM.write(385, irEnabled);
|
2018-12-04 00:58:06 +01:00
|
|
|
|
|
|
|
EEPROM.write(387, strip.ablMilliampsMax & 0xFF);
|
|
|
|
EEPROM.write(388, (strip.ablMilliampsMax >> 8) & 0xFF);
|
2017-12-12 14:52:28 +01:00
|
|
|
EEPROM.write(389, bootPreset);
|
2018-02-20 22:29:48 +01:00
|
|
|
EEPROM.write(390, aOtaEnabled);
|
|
|
|
EEPROM.write(391, receiveNotificationColor);
|
|
|
|
EEPROM.write(392, receiveNotificationEffects);
|
2018-02-23 01:33:32 +01:00
|
|
|
EEPROM.write(393, wifiLock);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(394, abs(utcOffsetSecs) & 0xFF);
|
2018-03-06 23:47:08 +01:00
|
|
|
EEPROM.write(395, (abs(utcOffsetSecs) >> 8) & 0xFF);
|
|
|
|
EEPROM.write(396, (utcOffsetSecs<0)); //is negative
|
2018-03-15 12:04:14 +01:00
|
|
|
EEPROM.write(397, initLedsLast);
|
2018-04-15 15:27:54 +02:00
|
|
|
EEPROM.write(398, (ledCount >> 8) & 0xFF);
|
2018-09-15 17:29:01 +02:00
|
|
|
EEPROM.write(399, !enableSecTransition);
|
2018-02-23 01:33:32 +01:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
//favorite setting (preset) memory (25 slots/ each 20byte)
|
|
|
|
//400 - 899 reserved
|
|
|
|
|
2018-02-23 01:33:32 +01:00
|
|
|
for (int k=0;k<6;k++){
|
|
|
|
int in = 900+k*8;
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(in, cssCol[k], 8);
|
|
|
|
}
|
2018-02-23 01:33:32 +01:00
|
|
|
|
2018-02-20 22:29:48 +01:00
|
|
|
EEPROM.write(948,currentTheme);
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(950, cssFont, 32);
|
2018-02-28 00:27:10 +01:00
|
|
|
|
|
|
|
EEPROM.write(2048, huePollingEnabled);
|
|
|
|
//EEPROM.write(2049, hueUpdatingEnabled);
|
|
|
|
for (int i = 2050; i < 2054; ++i)
|
|
|
|
{
|
|
|
|
EEPROM.write(i, hueIP[i-2050]);
|
|
|
|
}
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(2054, hueApiKey, 46);
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(2100, huePollIntervalMs & 0xFF);
|
2018-02-28 00:27:10 +01:00
|
|
|
EEPROM.write(2101, (huePollIntervalMs >> 8) & 0xFF);
|
|
|
|
EEPROM.write(2102, notifyHue);
|
|
|
|
EEPROM.write(2103, hueApplyOnOff);
|
|
|
|
EEPROM.write(2104, hueApplyBri);
|
|
|
|
EEPROM.write(2105, hueApplyColor);
|
|
|
|
EEPROM.write(2106, huePollLightId);
|
2018-03-14 00:25:54 +01:00
|
|
|
|
|
|
|
EEPROM.write(2150, overlayMin);
|
|
|
|
EEPROM.write(2151, overlayMax);
|
|
|
|
EEPROM.write(2152, analogClock12pixel);
|
|
|
|
EEPROM.write(2153, analogClock5MinuteMarks);
|
|
|
|
EEPROM.write(2154, analogClockSecondsTrail);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-03-14 00:25:54 +01:00
|
|
|
EEPROM.write(2155, countdownMode);
|
|
|
|
EEPROM.write(2156, countdownYear);
|
|
|
|
EEPROM.write(2157, countdownMonth);
|
|
|
|
EEPROM.write(2158, countdownDay);
|
|
|
|
EEPROM.write(2159, countdownHour);
|
|
|
|
EEPROM.write(2160, countdownMin);
|
|
|
|
EEPROM.write(2161, countdownSec);
|
|
|
|
setCountdown();
|
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(2165, cronixieDisplay, 6);
|
2018-03-14 00:25:54 +01:00
|
|
|
EEPROM.write(2171, cronixieBacklight);
|
|
|
|
setCronixie();
|
|
|
|
|
|
|
|
EEPROM.write(2175, macroBoot);
|
|
|
|
EEPROM.write(2176, macroAlexaOn);
|
|
|
|
EEPROM.write(2177, macroAlexaOff);
|
|
|
|
EEPROM.write(2178, macroButton);
|
|
|
|
EEPROM.write(2179, macroLongPress);
|
|
|
|
EEPROM.write(2180, macroCountdown);
|
|
|
|
EEPROM.write(2181, macroNl);
|
2018-05-10 19:55:58 +02:00
|
|
|
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(2190, e131Universe & 0xFF);
|
2018-08-11 22:57:13 +02:00
|
|
|
EEPROM.write(2191, (e131Universe >> 8) & 0xFF);
|
|
|
|
EEPROM.write(2192, e131Multicast);
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(2193, realtimeTimeoutMs & 0xFF);
|
2018-09-15 17:29:01 +02:00
|
|
|
EEPROM.write(2194, (realtimeTimeoutMs >> 8) & 0xFF);
|
2018-08-11 22:57:13 +02:00
|
|
|
EEPROM.write(2195, arlsForceMaxBri);
|
|
|
|
EEPROM.write(2196, arlsDisableGammaCorrection);
|
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
EEPROM.write(2200, !receiveDirect);
|
|
|
|
EEPROM.write(2201, enableRealtimeUI);
|
|
|
|
EEPROM.write(2202, uiConfiguration);
|
|
|
|
EEPROM.write(2203, autoRGBtoRGBW);
|
|
|
|
EEPROM.write(2204, skipFirstLed);
|
2018-06-24 01:20:15 +02:00
|
|
|
|
|
|
|
if (saveCurrPresetCycConf)
|
|
|
|
{
|
2018-10-07 11:56:29 +02:00
|
|
|
EEPROM.write(2205, presetCyclingEnabled);
|
2018-12-04 00:58:06 +01:00
|
|
|
EEPROM.write(2206, presetCycleTime & 0xFF);
|
2018-10-07 11:56:29 +02:00
|
|
|
EEPROM.write(2207, (presetCycleTime >> 8) & 0xFF);
|
|
|
|
EEPROM.write(2208, presetCycleMin);
|
|
|
|
EEPROM.write(2209, presetCycleMax);
|
|
|
|
EEPROM.write(2210, presetApplyBri);
|
|
|
|
EEPROM.write(2211, presetApplyCol);
|
|
|
|
EEPROM.write(2212, presetApplyFx);
|
2018-06-24 01:20:15 +02:00
|
|
|
saveCurrPresetCycConf = false;
|
|
|
|
}
|
2018-07-16 11:50:09 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(2220, blynkApiKey, 35);
|
2018-09-22 22:49:24 +02:00
|
|
|
|
|
|
|
for (int i = 0; i < 8; ++i)
|
|
|
|
{
|
|
|
|
EEPROM.write(2260 + i, timerHours[i] );
|
|
|
|
EEPROM.write(2270 + i, timerMinutes[i]);
|
|
|
|
EEPROM.write(2280 + i, timerWeekday[i]);
|
|
|
|
EEPROM.write(2290 + i, timerMacro[i] );
|
|
|
|
}
|
2018-10-04 16:50:12 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
writeStringToEEPROM(2300, mqttServer, 32);
|
|
|
|
writeStringToEEPROM(2333, mqttDeviceTopic, 32);
|
|
|
|
writeStringToEEPROM(2366, mqttGroupTopic, 32);
|
2018-02-20 22:29:48 +01:00
|
|
|
|
2018-11-18 00:31:45 +01:00
|
|
|
commit();
|
2016-11-19 19:39:17 +01:00
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-08-11 22:57:13 +02:00
|
|
|
/*
|
|
|
|
* Read all configuration from flash
|
|
|
|
*/
|
2017-12-11 23:59:12 +01:00
|
|
|
void loadSettingsFromEEPROM(bool first)
|
2016-11-19 19:39:17 +01:00
|
|
|
{
|
|
|
|
if (EEPROM.read(233) != 233) //first boot/reset to default
|
|
|
|
{
|
2018-11-21 23:28:20 +01:00
|
|
|
DEBUG_PRINT("Settings invalid, restoring defaults...");
|
2016-11-19 19:39:17 +01:00
|
|
|
saveSettingsToEEPROM();
|
2018-11-21 23:28:20 +01:00
|
|
|
DEBUG_PRINTLN("done");
|
2016-11-19 19:39:17 +01:00
|
|
|
return;
|
|
|
|
}
|
2017-11-28 00:21:29 +01:00
|
|
|
int lastEEPROMversion = EEPROM.read(377); //last EEPROM version before update
|
|
|
|
|
2018-07-21 23:21:07 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM( 0, clientSSID, 32);
|
|
|
|
readStringFromEEPROM( 32, clientPass, 64);
|
|
|
|
readStringFromEEPROM( 96, cmDNS, 32);
|
|
|
|
readStringFromEEPROM(128, apSSID, 32);
|
|
|
|
readStringFromEEPROM(160, apPass, 64);
|
2018-07-21 23:21:07 +02:00
|
|
|
|
2018-11-21 23:28:20 +01:00
|
|
|
nightlightDelayMinsDefault = EEPROM.read(224);
|
|
|
|
nightlightDelayMins = nightlightDelayMinsDefault;
|
2016-11-20 01:47:15 +01:00
|
|
|
nightlightFade = EEPROM.read(225);
|
2017-05-08 21:46:04 +02:00
|
|
|
notifyDirectDefault = EEPROM.read(226);
|
|
|
|
notifyDirect = notifyDirectDefault;
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
apChannel = EEPROM.read(227);
|
|
|
|
if (apChannel > 13 || apChannel < 1) apChannel = 1;
|
|
|
|
apHide = EEPROM.read(228);
|
|
|
|
if (apHide > 1) apHide = 1;
|
2018-12-04 00:58:06 +01:00
|
|
|
ledCount = EEPROM.read(229) + ((EEPROM.read(398) << 8) & 0xFF00); if (ledCount > 1200 || ledCount == 0) ledCount = 30;
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2016-11-20 01:47:15 +01:00
|
|
|
notifyButton = EEPROM.read(230);
|
2018-03-06 23:47:08 +01:00
|
|
|
notifyTwice = EEPROM.read(231);
|
2016-11-19 19:39:17 +01:00
|
|
|
buttonEnabled = EEPROM.read(232);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
staticIP[0] = EEPROM.read(234);
|
|
|
|
staticIP[1] = EEPROM.read(235);
|
|
|
|
staticIP[2] = EEPROM.read(236);
|
|
|
|
staticIP[3] = EEPROM.read(237);
|
|
|
|
staticGateway[0] = EEPROM.read(238);
|
|
|
|
staticGateway[1] = EEPROM.read(239);
|
|
|
|
staticGateway[2] = EEPROM.read(240);
|
|
|
|
staticGateway[3] = EEPROM.read(241);
|
|
|
|
staticSubnet[0] = EEPROM.read(242);
|
|
|
|
staticSubnet[1] = EEPROM.read(243);
|
|
|
|
staticSubnet[2] = EEPROM.read(244);
|
|
|
|
staticSubnet[3] = EEPROM.read(245);
|
2018-10-07 11:56:29 +02:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
colS[0] = EEPROM.read(246); col[0] = colS[0];
|
|
|
|
colS[1] = EEPROM.read(247); col[1] = colS[1];
|
|
|
|
colS[2] = EEPROM.read(248); col[2] = colS[2];
|
|
|
|
briS = EEPROM.read(249); bri = briS;
|
2017-12-11 23:59:12 +01:00
|
|
|
if (!EEPROM.read(369) && first)
|
2017-03-07 22:05:18 +01:00
|
|
|
{
|
2018-03-14 13:16:28 +01:00
|
|
|
bri = 0; briLast = briS;
|
2017-03-07 22:05:18 +01:00
|
|
|
}
|
2018-02-20 22:29:48 +01:00
|
|
|
receiveNotificationBrightness = EEPROM.read(250);
|
2016-11-19 19:39:17 +01:00
|
|
|
fadeTransition = EEPROM.read(251);
|
2018-03-06 23:47:08 +01:00
|
|
|
reverseMode = EEPROM.read(252);
|
2018-12-04 00:58:06 +01:00
|
|
|
transitionDelayDefault = EEPROM.read(253) + ((EEPROM.read(254) << 8) & 0xFF00);
|
2018-04-11 23:50:35 +02:00
|
|
|
transitionDelay = transitionDelayDefault;
|
2018-02-20 22:29:48 +01:00
|
|
|
briMultiplier = EEPROM.read(255);
|
2018-07-21 23:21:07 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(256, otaPass, 32);
|
|
|
|
|
2018-02-20 22:29:48 +01:00
|
|
|
nightlightTargetBri = EEPROM.read(288);
|
2017-11-20 00:07:37 +01:00
|
|
|
otaLock = EEPROM.read(289);
|
2018-12-04 00:58:06 +01:00
|
|
|
udpPort = EEPROM.read(290) + ((EEPROM.read(291) << 8) & 0xFF00);
|
2018-07-21 23:21:07 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(292, serverDescription, 32);
|
|
|
|
|
2016-12-29 00:03:58 +01:00
|
|
|
effectDefault = EEPROM.read(324); effectCurrent = effectDefault;
|
|
|
|
effectSpeedDefault = EEPROM.read(325); effectSpeed = effectSpeedDefault;
|
|
|
|
ntpEnabled = EEPROM.read(327);
|
2018-03-06 23:47:08 +01:00
|
|
|
currentTimezone = EEPROM.read(328);
|
|
|
|
useAMPM = EEPROM.read(329);
|
2017-02-01 19:25:36 +01:00
|
|
|
useGammaCorrectionBri = EEPROM.read(330);
|
|
|
|
useGammaCorrectionRGB = EEPROM.read(331);
|
2017-02-24 23:21:48 +01:00
|
|
|
overlayDefault = EEPROM.read(332);
|
2018-09-04 15:51:38 +02:00
|
|
|
if (lastEEPROMversion < 8 && overlayDefault > 0) overlayDefault--; //overlay mode 1 (solid) was removed
|
|
|
|
|
2017-02-24 23:21:48 +01:00
|
|
|
alexaEnabled = EEPROM.read(333);
|
2018-07-21 23:21:07 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(334, alexaInvocationName, 32);
|
|
|
|
|
2018-09-15 17:29:01 +02:00
|
|
|
notifyAlexa = EEPROM.read(366);
|
2017-02-24 23:21:48 +01:00
|
|
|
arlsOffset = EEPROM.read(368);
|
2018-03-14 13:16:28 +01:00
|
|
|
if (!EEPROM.read(367)) arlsOffset = -arlsOffset;
|
2017-03-07 22:05:18 +01:00
|
|
|
turnOnAtBoot = EEPROM.read(369);
|
2017-05-07 23:51:42 +02:00
|
|
|
useHSBDefault = EEPROM.read(370);
|
2019-02-05 21:53:39 +01:00
|
|
|
colS[3] = EEPROM.read(371); col[3] = colS[3];
|
2017-10-12 22:20:37 +02:00
|
|
|
useRGBW = EEPROM.read(372);
|
2018-09-06 02:05:56 +02:00
|
|
|
effectPaletteDefault = EEPROM.read(373); effectPalette = effectPaletteDefault;
|
2018-09-11 00:20:12 +02:00
|
|
|
//374 - strip.paletteFade
|
2018-09-04 15:51:38 +02:00
|
|
|
|
2017-11-28 00:21:29 +01:00
|
|
|
if (lastEEPROMversion > 0) {
|
|
|
|
apWaitTimeSecs = EEPROM.read(375);
|
|
|
|
recoveryAPDisabled = EEPROM.read(376);
|
|
|
|
}
|
2017-11-30 23:35:22 +01:00
|
|
|
//377 = lastEEPROMversion
|
|
|
|
if (lastEEPROMversion > 1) {
|
2019-02-05 21:53:39 +01:00
|
|
|
for (byte i=0; i<4; i++)
|
|
|
|
{
|
|
|
|
colSecS[i] = EEPROM.read(378+i); colSec[i] = colSecS[i];
|
|
|
|
}
|
2017-11-30 23:35:22 +01:00
|
|
|
}
|
2018-01-27 23:28:20 +01:00
|
|
|
if (lastEEPROMversion > 3) {
|
|
|
|
effectIntensityDefault = EEPROM.read(326); effectIntensity = effectIntensityDefault;
|
2018-02-20 22:29:48 +01:00
|
|
|
aOtaEnabled = EEPROM.read(390);
|
|
|
|
receiveNotificationColor = EEPROM.read(391);
|
|
|
|
receiveNotificationEffects = EEPROM.read(392);
|
2018-07-21 23:21:07 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(950, cssFont, 32);
|
2018-02-23 22:18:31 +01:00
|
|
|
} else //keep receiving notification behavior from pre0.5.0 after update
|
|
|
|
{
|
|
|
|
receiveNotificationColor = receiveNotificationBrightness;
|
|
|
|
receiveNotificationEffects = receiveNotificationBrightness;
|
2018-01-27 23:28:20 +01:00
|
|
|
}
|
2018-02-20 22:29:48 +01:00
|
|
|
receiveNotifications = (receiveNotificationBrightness || receiveNotificationColor || receiveNotificationEffects);
|
2018-02-28 00:27:10 +01:00
|
|
|
if (lastEEPROMversion > 4) {
|
|
|
|
huePollingEnabled = EEPROM.read(2048);
|
|
|
|
//hueUpdatingEnabled = EEPROM.read(2049);
|
|
|
|
for (int i = 2050; i < 2054; ++i)
|
|
|
|
{
|
|
|
|
hueIP[i-2050] = EEPROM.read(i);
|
|
|
|
}
|
2018-07-21 23:21:07 +02:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(2054, hueApiKey, 46);
|
|
|
|
|
2018-12-04 00:58:06 +01:00
|
|
|
huePollIntervalMs = EEPROM.read(2100) + ((EEPROM.read(2101) << 8) & 0xFF00);
|
2018-02-28 00:27:10 +01:00
|
|
|
notifyHue = EEPROM.read(2102);
|
|
|
|
hueApplyOnOff = EEPROM.read(2103);
|
|
|
|
hueApplyBri = EEPROM.read(2104);
|
|
|
|
hueApplyColor = EEPROM.read(2105);
|
|
|
|
huePollLightId = EEPROM.read(2106);
|
2018-03-15 12:04:14 +01:00
|
|
|
}
|
|
|
|
if (lastEEPROMversion > 5) {
|
2018-03-14 00:25:54 +01:00
|
|
|
overlayMin = EEPROM.read(2150);
|
|
|
|
overlayMax = EEPROM.read(2151);
|
|
|
|
analogClock12pixel = EEPROM.read(2152);
|
|
|
|
analogClock5MinuteMarks = EEPROM.read(2153);
|
|
|
|
analogClockSecondsTrail = EEPROM.read(2154);
|
|
|
|
countdownMode = EEPROM.read(2155);
|
|
|
|
countdownYear = EEPROM.read(2156);
|
|
|
|
countdownMonth = EEPROM.read(2157);
|
|
|
|
countdownDay = EEPROM.read(2158);
|
|
|
|
countdownHour = EEPROM.read(2159);
|
|
|
|
countdownMin = EEPROM.read(2160);
|
|
|
|
countdownSec = EEPROM.read(2161);
|
|
|
|
setCountdown();
|
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(2165, cronixieDisplay, 6);
|
2018-03-14 00:25:54 +01:00
|
|
|
cronixieBacklight = EEPROM.read(2171);
|
|
|
|
|
|
|
|
macroBoot = EEPROM.read(2175);
|
|
|
|
macroAlexaOn = EEPROM.read(2176);
|
|
|
|
macroAlexaOff = EEPROM.read(2177);
|
|
|
|
macroButton = EEPROM.read(2178);
|
|
|
|
macroLongPress = EEPROM.read(2179);
|
|
|
|
macroCountdown = EEPROM.read(2180);
|
|
|
|
macroNl = EEPROM.read(2181);
|
2018-02-28 00:27:10 +01:00
|
|
|
}
|
2018-08-11 22:57:13 +02:00
|
|
|
|
|
|
|
if (lastEEPROMversion > 6)
|
|
|
|
{
|
2018-12-04 00:58:06 +01:00
|
|
|
e131Universe = EEPROM.read(2190) + ((EEPROM.read(2191) << 8) & 0xFF00);
|
2018-08-11 22:57:13 +02:00
|
|
|
e131Multicast = EEPROM.read(2192);
|
2018-12-04 00:58:06 +01:00
|
|
|
realtimeTimeoutMs = EEPROM.read(2193) + ((EEPROM.read(2194) << 8) & 0xFF00);
|
2018-08-11 22:57:13 +02:00
|
|
|
arlsForceMaxBri = EEPROM.read(2195);
|
|
|
|
arlsDisableGammaCorrection = EEPROM.read(2196);
|
|
|
|
}
|
2018-09-11 00:20:12 +02:00
|
|
|
|
|
|
|
if (lastEEPROMversion > 7)
|
|
|
|
{
|
2018-09-22 22:49:24 +02:00
|
|
|
strip.paletteFade = EEPROM.read(374);
|
2018-09-11 00:20:12 +02:00
|
|
|
strip.paletteBlend = EEPROM.read(382);
|
2018-09-22 22:49:24 +02:00
|
|
|
|
|
|
|
for (int i = 0; i < 8; ++i)
|
|
|
|
{
|
|
|
|
timerHours[i] = EEPROM.read(2260 + i);
|
|
|
|
timerMinutes[i] = EEPROM.read(2270 + i);
|
|
|
|
timerWeekday[i] = EEPROM.read(2280 + i);
|
2018-12-04 00:58:06 +01:00
|
|
|
timerMacro[i] = EEPROM.read(2290 + i);
|
2018-09-22 22:49:24 +02:00
|
|
|
if (timerWeekday[i] == 0) timerWeekday[i] = 255;
|
|
|
|
}
|
2018-09-11 00:20:12 +02:00
|
|
|
}
|
2018-10-04 16:50:12 +02:00
|
|
|
|
|
|
|
if (lastEEPROMversion > 8)
|
|
|
|
{
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(2300, mqttServer, 32);
|
|
|
|
readStringFromEEPROM(2333, mqttDeviceTopic, 32);
|
|
|
|
readStringFromEEPROM(2366, mqttGroupTopic, 32);
|
2018-10-04 16:50:12 +02:00
|
|
|
}
|
2018-12-04 00:58:06 +01:00
|
|
|
|
|
|
|
if (lastEEPROMversion > 9)
|
|
|
|
{
|
2018-12-06 00:27:36 +01:00
|
|
|
strip.colorOrder = EEPROM.read(383);
|
|
|
|
irEnabled = EEPROM.read(385);
|
2018-12-04 00:58:06 +01:00
|
|
|
strip.ablMilliampsMax = EEPROM.read(387) + ((EEPROM.read(388) << 8) & 0xFF00);
|
2019-01-09 22:52:42 +01:00
|
|
|
} else if (lastEEPROMversion > 1) //ABL is off by default when updating from version older than 0.8.3
|
2018-12-04 00:58:06 +01:00
|
|
|
{
|
2018-12-06 00:27:36 +01:00
|
|
|
strip.ablMilliampsMax = 65000;
|
|
|
|
} else {
|
2018-12-04 00:58:06 +01:00
|
|
|
strip.ablMilliampsMax = ABL_MILLIAMPS_DEFAULT;
|
|
|
|
}
|
2018-08-11 22:57:13 +02:00
|
|
|
|
2018-05-10 19:55:58 +02:00
|
|
|
receiveDirect = !EEPROM.read(2200);
|
|
|
|
enableRealtimeUI = EEPROM.read(2201);
|
2018-05-22 21:11:19 +02:00
|
|
|
uiConfiguration = EEPROM.read(2202);
|
2018-06-24 01:20:15 +02:00
|
|
|
|
2018-11-01 15:36:13 +01:00
|
|
|
#ifdef WLED_DISABLE_MOBILE_UI
|
2018-06-24 01:20:15 +02:00
|
|
|
uiConfiguration = 1;
|
|
|
|
//force default UI since mobile is unavailable
|
|
|
|
#endif
|
|
|
|
|
2018-05-31 19:26:16 +02:00
|
|
|
autoRGBtoRGBW = EEPROM.read(2203);
|
2018-06-24 01:20:15 +02:00
|
|
|
skipFirstLed = EEPROM.read(2204);
|
|
|
|
|
|
|
|
if (EEPROM.read(2210) || EEPROM.read(2211) || EEPROM.read(2212))
|
|
|
|
{
|
|
|
|
presetCyclingEnabled = EEPROM.read(2205);
|
2018-12-04 00:58:06 +01:00
|
|
|
presetCycleTime = EEPROM.read(2206) + ((EEPROM.read(2207) << 8) & 0xFF00);
|
2018-06-24 01:20:15 +02:00
|
|
|
presetCycleMin = EEPROM.read(2208);
|
|
|
|
presetCycleMax = EEPROM.read(2209);
|
|
|
|
presetApplyBri = EEPROM.read(2210);
|
|
|
|
presetApplyCol = EEPROM.read(2211);
|
|
|
|
presetApplyFx = EEPROM.read(2212);
|
|
|
|
}
|
2018-02-20 22:29:48 +01:00
|
|
|
|
2017-12-12 14:52:28 +01:00
|
|
|
bootPreset = EEPROM.read(389);
|
2018-02-23 01:33:32 +01:00
|
|
|
wifiLock = EEPROM.read(393);
|
2018-12-04 00:58:06 +01:00
|
|
|
utcOffsetSecs = EEPROM.read(394) + ((EEPROM.read(395) << 8) & 0xFF00);
|
2018-03-06 23:47:08 +01:00
|
|
|
if (EEPROM.read(396)) utcOffsetSecs = -utcOffsetSecs; //negative
|
2018-03-15 12:04:14 +01:00
|
|
|
initLedsLast = EEPROM.read(397);
|
2018-09-15 17:29:01 +02:00
|
|
|
enableSecTransition = !EEPROM.read(399);
|
2017-11-28 00:21:29 +01:00
|
|
|
|
2018-05-10 19:55:58 +02:00
|
|
|
//favorite setting (preset) memory (25 slots/ each 20byte)
|
2017-11-28 00:21:29 +01:00
|
|
|
//400 - 899 reserved
|
2017-12-28 00:37:13 +01:00
|
|
|
|
2018-02-20 22:29:48 +01:00
|
|
|
currentTheme = EEPROM.read(948);
|
2018-02-23 01:33:32 +01:00
|
|
|
for (int k=0;k<6;k++){
|
|
|
|
int in=900+k*8;
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(in, cssCol[k], 8);
|
|
|
|
}
|
2018-02-20 22:29:48 +01:00
|
|
|
|
2017-12-28 00:37:13 +01:00
|
|
|
//custom macro memory (16 slots/ each 64byte)
|
|
|
|
//1024-2047 reserved
|
2018-02-28 00:27:10 +01:00
|
|
|
|
2018-10-07 11:56:29 +02:00
|
|
|
readStringFromEEPROM(2220, blynkApiKey, 35);
|
|
|
|
|
2018-02-28 00:27:10 +01:00
|
|
|
//user MOD memory
|
|
|
|
//2944 - 3071 reserved
|
2017-11-28 00:21:29 +01:00
|
|
|
|
2017-05-07 23:51:42 +02:00
|
|
|
useHSB = useHSBDefault;
|
2017-12-11 23:59:12 +01:00
|
|
|
|
|
|
|
overlayCurrent = overlayDefault;
|
2016-11-19 19:39:17 +01:00
|
|
|
}
|
2017-11-28 00:21:29 +01:00
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2017-11-28 00:21:29 +01:00
|
|
|
//PRESET PROTOCOL 20 bytes
|
2017-12-02 23:58:22 +01:00
|
|
|
//0: preset purpose byte 0:invalid 1:valid preset 1.0
|
2018-09-06 02:05:56 +02:00
|
|
|
//1:a 2:r 3:g 4:b 5:w 6:er 7:eg 8:eb 9:ew 10:fx 11:sx | custom chase 12:numP 13:numS 14:(0:fs 1:both 2:fe) 15:step 16:ix 17: fp 18-19:Zeros
|
2017-11-28 00:21:29 +01:00
|
|
|
|
2018-11-18 00:31:45 +01:00
|
|
|
bool applyPreset(byte index, bool loadBri = true, bool loadCol = true, bool loadFX = true)
|
2017-11-28 00:21:29 +01:00
|
|
|
{
|
2018-11-18 00:31:45 +01:00
|
|
|
if (index == 255 || index == 0)
|
|
|
|
{
|
|
|
|
loadSettingsFromEEPROM(false);//load boot defaults
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (index > 25 || index < 1) return false;
|
2017-12-11 23:59:12 +01:00
|
|
|
uint16_t i = 380 + index*20;
|
2018-11-18 00:31:45 +01:00
|
|
|
if (EEPROM.read(i) == 0) return false;
|
2017-12-02 23:58:22 +01:00
|
|
|
if (loadBri) bri = EEPROM.read(i+1);
|
|
|
|
if (loadCol)
|
2017-11-28 00:21:29 +01:00
|
|
|
{
|
2019-02-05 21:53:39 +01:00
|
|
|
for (byte j=0; j<4; j++)
|
|
|
|
{
|
|
|
|
col[j] = EEPROM.read(i+j+2);
|
|
|
|
colSec[j] = EEPROM.read(i+j+6);
|
|
|
|
}
|
2017-11-30 23:35:22 +01:00
|
|
|
}
|
2017-12-02 23:58:22 +01:00
|
|
|
if (loadFX)
|
2017-11-30 23:35:22 +01:00
|
|
|
{
|
|
|
|
effectCurrent = EEPROM.read(i+10);
|
|
|
|
effectSpeed = EEPROM.read(i+11);
|
2018-01-27 23:28:20 +01:00
|
|
|
effectIntensity = EEPROM.read(i+16);
|
2018-10-01 21:31:31 +02:00
|
|
|
effectPalette = EEPROM.read(i+17);
|
2017-11-28 00:21:29 +01:00
|
|
|
}
|
2018-11-18 00:31:45 +01:00
|
|
|
return true;
|
2017-11-28 00:21:29 +01:00
|
|
|
}
|
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
void savePreset(byte index)
|
2017-11-28 00:21:29 +01:00
|
|
|
{
|
2017-12-11 23:59:12 +01:00
|
|
|
if (index > 25) return;
|
|
|
|
if (index < 1) {saveSettingsToEEPROM();return;}
|
|
|
|
uint16_t i = 380 + index*20;//min400
|
2017-12-02 23:58:22 +01:00
|
|
|
EEPROM.write(i, 1);
|
|
|
|
EEPROM.write(i+1, bri);
|
2019-02-05 21:53:39 +01:00
|
|
|
for (uint16_t j=0; j<4; j++)
|
|
|
|
{
|
|
|
|
EEPROM.write(i+j+2, col[j]);
|
|
|
|
EEPROM.write(i+j+6, colSec[j]);
|
|
|
|
}
|
2017-12-02 23:58:22 +01:00
|
|
|
EEPROM.write(i+10, effectCurrent);
|
|
|
|
EEPROM.write(i+11, effectSpeed);
|
2018-09-04 15:51:38 +02:00
|
|
|
|
2018-01-27 23:28:20 +01:00
|
|
|
EEPROM.write(i+16, effectIntensity);
|
2018-09-06 02:05:56 +02:00
|
|
|
EEPROM.write(i+17, effectPalette);
|
2018-11-18 00:31:45 +01:00
|
|
|
commit();
|
2017-11-28 00:21:29 +01:00
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
String loadMacro(byte index)
|
2017-12-28 00:37:13 +01:00
|
|
|
{
|
2018-03-06 23:47:08 +01:00
|
|
|
index-=1;
|
|
|
|
String m="";
|
|
|
|
if (index > 15) return m;
|
2017-12-28 00:37:13 +01:00
|
|
|
for (int i = 1024+64*index; i < 1088+64*index; i++)
|
|
|
|
{
|
|
|
|
if (EEPROM.read(i) == 0) break;
|
2018-03-06 23:47:08 +01:00
|
|
|
m += char(EEPROM.read(i));
|
2017-12-28 00:37:13 +01:00
|
|
|
}
|
2018-03-06 23:47:08 +01:00
|
|
|
return m;
|
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
void applyMacro(byte index)
|
2018-03-06 23:47:08 +01:00
|
|
|
{
|
|
|
|
index-=1;
|
|
|
|
if (index > 15) return;
|
|
|
|
String mc="win&";
|
|
|
|
mc += loadMacro(index+1);
|
2017-12-28 00:37:13 +01:00
|
|
|
mc += "&IN"; //internal, no XML response
|
2018-09-15 17:29:01 +02:00
|
|
|
if (!notifyMacro) mc += "&NN";
|
2017-12-28 00:37:13 +01:00
|
|
|
String forbidden = "&M="; //dont apply if called by the macro itself to prevent loop
|
|
|
|
/*
|
|
|
|
* NOTE: loop is still possible if you call a different macro from a macro, which then calls the first macro again.
|
|
|
|
* To prevent that, but also disable calling macros within macros, comment the next line out.
|
|
|
|
*/
|
|
|
|
forbidden = forbidden + index;
|
|
|
|
if (mc.indexOf(forbidden) >= 0) return;
|
|
|
|
handleSet(mc);
|
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-03-14 13:16:28 +01:00
|
|
|
void saveMacro(byte index, String mc, bool sing=true) //only commit on single save, not in settings
|
2017-12-28 00:37:13 +01:00
|
|
|
{
|
2018-03-06 23:47:08 +01:00
|
|
|
index-=1;
|
2017-12-28 00:37:13 +01:00
|
|
|
if (index > 15) return;
|
|
|
|
int s = 1024+index*64;
|
|
|
|
for (int i = s; i < s+64; i++)
|
|
|
|
{
|
|
|
|
EEPROM.write(i, mc.charAt(i-s));
|
|
|
|
}
|
2018-11-18 00:31:45 +01:00
|
|
|
if (sing) commit();
|
|
|
|
}
|
|
|
|
|
2018-11-24 11:52:23 +01:00
|
|
|
|
2018-11-18 00:31:45 +01:00
|
|
|
void commit()
|
|
|
|
{
|
2018-11-21 23:28:20 +01:00
|
|
|
DEBUG_PRINT("s");
|
2018-11-24 11:52:23 +01:00
|
|
|
//this is to support IR on ESP32, needs work
|
2018-11-21 23:28:20 +01:00
|
|
|
/*#ifdef ARDUINO_ARCH_ESP32
|
|
|
|
portMUX_TYPE mMux = portMUX_INITIALIZER_UNLOCKED;
|
|
|
|
portENTER_CRITICAL(&mMux);
|
|
|
|
#endif*/
|
|
|
|
|
|
|
|
EEPROM.commit();
|
|
|
|
|
|
|
|
/*#ifdef ARDUINO_ARCH_ESP32
|
|
|
|
portEXIT_CRITICAL(&mMux);
|
|
|
|
#endif*/
|
|
|
|
DEBUG_PRINT(".");
|
2017-12-28 00:37:13 +01:00
|
|
|
}
|