2020-11-06 22:12:48 +01:00
# include "wled.h"
/*
* Methods to handle saving and loading presets to / from the filesystem
*/
2022-01-01 12:52:50 +01:00
# ifdef ARDUINO_ARCH_ESP32
static char * tmpRAMbuffer = nullptr ;
# endif
static volatile byte presetToApply = 0 ;
static volatile byte callModeToApply = 0 ;
2022-10-08 18:25:51 +02:00
static volatile byte presetToSave = 0 ;
2022-11-16 20:55:21 +01:00
static volatile int8_t saveLedmap = - 1 ;
2022-10-08 18:25:51 +02:00
static char quickLoad [ 3 ] ;
static char saveName [ 33 ] ;
static bool includeBri = true , segBounds = true , selectedOnly = false , playlistSave = false ; ;
static const char * getName ( bool persist = true ) {
return persist ? " /presets.json " : " /tmp.json " ;
}
static void doSaveState ( ) {
bool persist = ( presetToSave < 251 ) ;
const char * filename = getName ( persist ) ;
if ( ! requestJSONBufferLock ( 10 ) ) return ; // will set fileDoc
2022-11-16 20:55:21 +01:00
initPresetsFile ( ) ; // just in case if someone deleted presets.json using /edit
2022-10-08 18:25:51 +02:00
JsonObject sObj = doc . to < JsonObject > ( ) ;
DEBUG_PRINTLN ( F ( " Serialize current state " ) ) ;
if ( playlistSave ) {
serializePlaylist ( sObj ) ;
if ( includeBri ) sObj [ " on " ] = true ;
} else {
serializeState ( sObj , true , includeBri , segBounds , selectedOnly ) ;
}
sObj [ " n " ] = saveName ;
if ( quickLoad [ 0 ] ) sObj [ F ( " ql " ) ] = quickLoad ;
2022-11-16 20:55:21 +01:00
if ( saveLedmap > = 0 ) sObj [ F ( " ledmap " ) ] = saveLedmap ;
2022-10-09 12:09:46 +02:00
/*
2022-10-08 18:25:51 +02:00
# ifdef WLED_DEBUG
DEBUG_PRINTLN ( F ( " Serialized preset " ) ) ;
serializeJson ( doc , Serial ) ;
DEBUG_PRINTLN ( ) ;
# endif
2022-10-09 12:09:46 +02:00
*/
2022-10-08 18:25:51 +02:00
# if defined(ARDUINO_ARCH_ESP32)
if ( ! persist ) {
if ( tmpRAMbuffer ! = nullptr ) free ( tmpRAMbuffer ) ;
size_t len = measureJson ( * fileDoc ) + 1 ;
DEBUG_PRINTLN ( len ) ;
// if possible use SPI RAM on ESP32
# ifdef WLED_USE_PSRAM
if ( psramFound ( ) )
tmpRAMbuffer = ( char * ) ps_malloc ( len ) ;
else
# endif
tmpRAMbuffer = ( char * ) malloc ( len ) ;
if ( tmpRAMbuffer ! = nullptr ) {
serializeJson ( * fileDoc , tmpRAMbuffer , len ) ;
} else {
writeObjectToFileUsingId ( filename , presetToSave , fileDoc ) ;
}
} else
# endif
writeObjectToFileUsingId ( filename , presetToSave , fileDoc ) ;
if ( persist ) presetsModifiedTime = toki . second ( ) ; //unix time
releaseJSONBufferLock ( ) ;
updateFSInfo ( ) ;
// clean up
2022-11-16 20:55:21 +01:00
saveLedmap = - 1 ;
2022-10-08 18:25:51 +02:00
presetToSave = 0 ;
saveName [ 0 ] = ' \0 ' ;
quickLoad [ 0 ] = ' \0 ' ;
playlistSave = false ;
}
2022-01-01 12:52:50 +01:00
2022-10-25 23:42:26 +02:00
bool getPresetName ( byte index , String & name )
{
if ( ! requestJSONBufferLock ( 9 ) ) return false ;
bool presetExists = false ;
2022-11-16 20:55:21 +01:00
if ( readObjectFromFileUsingId ( getName ( ) , index , & doc ) )
2022-10-25 23:42:26 +02:00
{
JsonObject fdo = doc . as < JsonObject > ( ) ;
if ( fdo [ " n " ] ) {
name = ( const char * ) ( fdo [ " n " ] ) ;
presetExists = true ;
}
}
releaseJSONBufferLock ( ) ;
return presetExists ;
}
2022-11-16 20:55:21 +01:00
void initPresetsFile ( )
{
if ( WLED_FS . exists ( getName ( ) ) ) return ;
StaticJsonDocument < 64 > doc ;
JsonObject sObj = doc . to < JsonObject > ( ) ;
sObj . createNestedObject ( " 0 " ) ;
File f = WLED_FS . open ( getName ( ) , " w " ) ;
if ( ! f ) {
errorFlag = ERR_FS_GENERAL ;
return ;
}
serializeJson ( doc , f ) ;
f . close ( ) ;
}
2022-09-20 21:17:44 +02:00
bool applyPreset ( byte index , byte callMode )
2020-11-06 22:12:48 +01:00
{
2022-06-01 22:11:25 +02:00
DEBUG_PRINT ( F ( " Request to apply preset: " ) ) ;
DEBUG_PRINTLN ( index ) ;
2022-01-01 12:52:50 +01:00
presetToApply = index ;
callModeToApply = callMode ;
return true ;
}
2022-09-20 21:17:44 +02:00
void handlePresets ( )
2022-01-01 12:52:50 +01:00
{
2022-10-08 18:25:51 +02:00
if ( presetToSave ) {
doSaveState ( ) ;
return ;
}
2022-04-24 19:30:14 +02:00
bool changePreset = false ;
2022-06-01 22:11:25 +02:00
uint8_t tmpPreset = presetToApply ; // store temporary since deserializeState() may call applyPreset()
uint8_t tmpMode = callModeToApply ;
2022-04-24 19:30:14 +02:00
2022-09-20 21:17:44 +02:00
if ( tmpPreset = = 0 | | ( fileDoc /*&& !force*/ ) ) return ; // JSON buffer already allocated and not force apply or no preset waiting
2022-01-01 12:52:50 +01:00
JsonObject fdo ;
2022-10-08 18:25:51 +02:00
const char * filename = getName ( tmpPreset < 255 ) ;
2022-01-01 12:52:50 +01:00
// allocate buffer
if ( ! requestJSONBufferLock ( 9 ) ) return ; // will also assign fileDoc
2022-06-01 22:11:25 +02:00
presetToApply = 0 ; //clear request for preset
callModeToApply = 0 ;
2022-07-17 15:58:41 +02:00
DEBUG_PRINT ( F ( " Applying preset: " ) ) ;
2022-06-01 22:11:25 +02:00
DEBUG_PRINTLN ( tmpPreset ) ;
2022-01-01 12:52:50 +01:00
# ifdef ARDUINO_ARCH_ESP32
2022-06-01 22:11:25 +02:00
if ( tmpPreset = = 255 & & tmpRAMbuffer ! = nullptr ) {
2022-01-01 12:52:50 +01:00
deserializeJson ( * fileDoc , tmpRAMbuffer ) ;
errorFlag = ERR_NONE ;
} else
# endif
{
2022-06-01 22:11:25 +02:00
errorFlag = readObjectFromFileUsingId ( filename , tmpPreset , fileDoc ) ? ERR_NONE : ERR_FS_PLOAD ;
2020-11-06 22:12:48 +01:00
}
2022-01-01 12:52:50 +01:00
fdo = fileDoc - > as < JsonObject > ( ) ;
2022-01-03 22:23:03 +01:00
//HTTP API commands
const char * httpwin = fdo [ " win " ] ;
if ( httpwin ) {
2022-04-24 19:47:55 +02:00
String apireq = " win " ; // reduce flash string usage
apireq + = F ( " &IN& " ) ; // internal call
2022-01-03 22:23:03 +01:00
apireq + = httpwin ;
2022-06-01 22:11:25 +02:00
handleSet ( nullptr , apireq , false ) ; // may call applyPreset() via PL=
2022-04-16 16:28:43 +02:00
setValuesFromFirstSelectedSeg ( ) ; // fills legacy values
2022-04-24 19:30:14 +02:00
changePreset = true ;
2022-01-03 22:23:03 +01:00
} else {
2022-06-01 22:11:25 +02:00
if ( ! fdo [ " seg " ] . isNull ( ) | | ! fdo [ " on " ] . isNull ( ) | | ! fdo [ " bri " ] . isNull ( ) | | ! fdo [ " nl " ] . isNull ( ) | | ! fdo [ " ps " ] . isNull ( ) | | ! fdo [ F ( " playlist " ) ] . isNull ( ) ) changePreset = true ;
2022-01-03 22:23:03 +01:00
fdo . remove ( " ps " ) ; //remove load request for presets to prevent recursive crash
2022-06-01 22:11:25 +02:00
deserializeState ( fdo , CALL_MODE_NO_NOTIFY , tmpPreset ) ; // may change presetToApply by calling applyPreset()
2022-01-03 22:23:03 +01:00
}
2022-06-03 22:22:18 +02:00
if ( ! errorFlag & & tmpPreset < 255 & & changePreset ) presetCycCurr = currentPreset = tmpPreset ;
2020-11-06 22:12:48 +01:00
2022-01-01 12:52:50 +01:00
# if defined(ARDUINO_ARCH_ESP32)
//Aircoookie recommended not to delete buffer
2022-06-01 22:11:25 +02:00
if ( tmpPreset = = 255 & & tmpRAMbuffer ! = nullptr ) {
2022-01-01 12:52:50 +01:00
free ( tmpRAMbuffer ) ;
tmpRAMbuffer = nullptr ;
2020-11-06 22:12:48 +01:00
}
2022-01-01 12:52:50 +01:00
# endif
releaseJSONBufferLock ( ) ; // will also clear fileDoc
2022-06-01 22:11:25 +02:00
colorUpdated ( tmpMode ) ;
updateInterfaces ( tmpMode ) ;
2020-11-06 22:12:48 +01:00
}
2022-01-01 12:52:50 +01:00
//called from handleSet(PS=) [network callback (fileDoc==nullptr), IR (irrational), deserializeState, UDP] and deserializeState() [network callback (filedoc!=nullptr)]
2022-10-08 18:25:51 +02:00
void savePreset ( byte index , const char * pname , JsonObject sObj )
2020-11-06 22:12:48 +01:00
{
2022-03-15 09:55:23 +01:00
if ( index = = 0 | | ( index > 250 & & index < 255 ) ) return ;
2022-10-08 18:25:51 +02:00
if ( pname ) strlcpy ( saveName , pname , 33 ) ;
else {
if ( sObj [ " n " ] . is < const char * > ( ) ) strlcpy ( saveName , sObj [ " n " ] . as < const char * > ( ) , 33 ) ;
else sprintf_P ( saveName , PSTR ( " Preset %d " ) , index ) ;
2022-01-01 12:52:50 +01:00
}
2020-11-29 22:07:12 +01:00
2022-10-08 18:25:51 +02:00
DEBUG_PRINT ( F ( " Saving preset ( " ) ) ; DEBUG_PRINT ( index ) ; DEBUG_PRINT ( F ( " ) " ) ) ; DEBUG_PRINTLN ( saveName ) ;
2021-11-03 14:52:22 +01:00
2022-10-08 18:25:51 +02:00
presetToSave = index ;
playlistSave = false ;
if ( sObj [ F ( " ql " ) ] . is < const char * > ( ) ) strlcpy ( quickLoad , sObj [ F ( " ql " ) ] . as < const char * > ( ) , 3 ) ; // only 2 chars for QL
sObj . remove ( " v " ) ;
2022-10-09 12:09:46 +02:00
sObj . remove ( " time " ) ;
2022-10-08 18:25:51 +02:00
sObj . remove ( F ( " error " ) ) ;
sObj . remove ( F ( " psave " ) ) ;
if ( sObj [ " o " ] . isNull ( ) ) { // "o" marks a playlist or manually entered API
2022-10-08 21:31:59 +02:00
includeBri = sObj [ " ib " ] . as < bool > ( ) | | index = = 255 ; // temporary preset needs brightness
segBounds = sObj [ " sb " ] . as < bool > ( ) | | index = = 255 ; // temporary preset needs bounds
2022-10-08 18:25:51 +02:00
selectedOnly = sObj [ F ( " sc " ) ] . as < bool > ( ) ;
2022-11-16 20:55:21 +01:00
saveLedmap = sObj [ F ( " ledmap " ) ] | - 1 ;
2022-10-08 18:25:51 +02:00
sObj . remove ( " ib " ) ;
sObj . remove ( " sb " ) ;
sObj . remove ( F ( " sc " ) ) ;
} else {
// this is a playlist or API
sObj . remove ( " o " ) ;
if ( sObj [ F ( " playlist " ) ] . isNull ( ) ) {
presetToSave = 0 ; // we will save API immediately
if ( index < 251 & & fileDoc ) {
if ( sObj [ " n " ] . isNull ( ) ) sObj [ " n " ] = saveName ;
2022-11-16 20:55:21 +01:00
initPresetsFile ( ) ; // just in case if someone deleted presets.json using /edit
2022-10-08 18:25:51 +02:00
writeObjectToFileUsingId ( getName ( index ) , index , fileDoc ) ;
presetsModifiedTime = toki . second ( ) ; //unix time
updateFSInfo ( ) ;
}
2022-01-01 12:52:50 +01:00
} else {
2022-10-08 18:25:51 +02:00
// store playlist
includeBri = true ; // !sObj["on"].isNull();
playlistSave = true ;
2020-11-29 22:07:12 +01:00
}
2022-10-08 18:25:51 +02:00
}
2020-11-06 22:12:48 +01:00
}
void deletePreset ( byte index ) {
StaticJsonDocument < 24 > empty ;
2022-10-09 12:09:46 +02:00
writeObjectToFileUsingId ( getName ( ) , index , & empty ) ;
2021-05-25 09:59:19 +02:00
presetsModifiedTime = toki . second ( ) ; //unix time
2020-11-06 22:12:48 +01:00
updateFSInfo ( ) ;
}