2021-03-25 19:00:08 +00:00
|
|
|
#ifndef WLED_FCN_DECLARE_H
|
|
|
|
#define WLED_FCN_DECLARE_H
|
|
|
|
|
|
|
|
/*
|
|
|
|
* All globally accessible functions are declared here
|
|
|
|
*/
|
|
|
|
|
|
|
|
//alexa.cpp
|
2023-01-12 19:35:34 +00:00
|
|
|
#ifndef WLED_DISABLE_ALEXA
|
2021-03-25 19:00:08 +00:00
|
|
|
void onAlexaChange(EspalexaDevice* dev);
|
|
|
|
void alexaInit();
|
|
|
|
void handleAlexa();
|
|
|
|
void onAlexaChange(EspalexaDevice* dev);
|
2023-01-12 19:35:34 +00:00
|
|
|
#endif
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//button.cpp
|
2021-05-19 16:39:16 +00:00
|
|
|
void shortPressAction(uint8_t b=0);
|
2022-01-21 16:30:52 +00:00
|
|
|
void longPressAction(uint8_t b=0);
|
|
|
|
void doublePressAction(uint8_t b=0);
|
2021-05-19 16:39:16 +00:00
|
|
|
bool isButtonPressed(uint8_t b=0);
|
2021-03-25 19:00:08 +00:00
|
|
|
void handleButton();
|
|
|
|
void handleIO();
|
2024-02-18 14:52:36 +00:00
|
|
|
void IRAM_ATTR touchButtonISR();
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//cfg.cpp
|
2021-05-10 23:11:16 +00:00
|
|
|
bool deserializeConfig(JsonObject doc, bool fromFS = false);
|
|
|
|
void deserializeConfigFromFS();
|
2021-03-25 19:00:08 +00:00
|
|
|
bool deserializeConfigSec();
|
|
|
|
void serializeConfig();
|
|
|
|
void serializeConfigSec();
|
|
|
|
|
2021-07-05 21:14:57 +00:00
|
|
|
template<typename DestType>
|
|
|
|
bool getJsonValue(const JsonVariant& element, DestType& destination) {
|
|
|
|
if (element.isNull()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
destination = element.as<DestType>();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename DestType, typename DefaultType>
|
|
|
|
bool getJsonValue(const JsonVariant& element, DestType& destination, const DefaultType defaultValue) {
|
|
|
|
if(!getJsonValue(element, destination)) {
|
|
|
|
destination = defaultValue;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-01-20 23:30:15 +00:00
|
|
|
typedef struct WiFiConfig {
|
|
|
|
char clientSSID[33];
|
|
|
|
char clientPass[65];
|
|
|
|
IPAddress staticIP;
|
|
|
|
IPAddress staticGW;
|
|
|
|
IPAddress staticSN;
|
|
|
|
WiFiConfig(const char *ssid="", const char *pass="", uint32_t ip=0, uint32_t gw=0, uint32_t subnet=0x00FFFFFF) // little endian
|
|
|
|
: staticIP(ip)
|
|
|
|
, staticGW(gw)
|
|
|
|
, staticSN(subnet)
|
|
|
|
{
|
|
|
|
strncpy(clientSSID, ssid, 32); clientSSID[32] = 0;
|
|
|
|
strncpy(clientPass, pass, 64); clientPass[64] = 0;
|
|
|
|
}
|
|
|
|
} wifi_config;
|
2021-07-05 21:14:57 +00:00
|
|
|
|
2021-03-25 19:00:08 +00:00
|
|
|
//colors.cpp
|
2023-06-10 18:43:27 +00:00
|
|
|
// similar to NeoPixelBus NeoGammaTableMethod but allows dynamic changes (superseded by NPB::NeoGammaDynamicTableMethod)
|
|
|
|
class NeoGammaWLEDMethod {
|
|
|
|
public:
|
2024-09-11 15:14:59 +00:00
|
|
|
[[gnu::hot]] static uint8_t Correct(uint8_t value); // apply Gamma to single channel
|
|
|
|
[[gnu::hot]] static uint32_t Correct32(uint32_t color); // apply Gamma to RGBW32 color (WLED specific, not used by NPB)
|
|
|
|
static void calcGammaTable(float gamma); // re-calculates & fills gamma table
|
2023-06-10 18:43:27 +00:00
|
|
|
static inline uint8_t rawGamma8(uint8_t val) { return gammaT[val]; } // get value from Gamma table (WLED specific, not used by NPB)
|
|
|
|
private:
|
|
|
|
static uint8_t gammaT[];
|
|
|
|
};
|
|
|
|
#define gamma32(c) NeoGammaWLEDMethod::Correct32(c)
|
|
|
|
#define gamma8(c) NeoGammaWLEDMethod::rawGamma8(c)
|
2024-09-11 15:14:59 +00:00
|
|
|
[[gnu::hot]] uint32_t color_blend(uint32_t,uint32_t,uint16_t,bool b16=false);
|
|
|
|
[[gnu::hot]] uint32_t color_add(uint32_t,uint32_t, bool fast=false);
|
|
|
|
[[gnu::hot]] uint32_t color_fade(uint32_t c1, uint8_t amount, bool video=false);
|
2024-01-30 21:28:40 +00:00
|
|
|
CRGBPalette16 generateHarmonicRandomPalette(CRGBPalette16 &basepalette);
|
2024-09-11 15:14:59 +00:00
|
|
|
CRGBPalette16 generateRandomPalette();
|
2022-01-14 13:27:11 +00:00
|
|
|
inline uint32_t colorFromRgbw(byte* rgbw) { return uint32_t((byte(rgbw[3]) << 24) | (byte(rgbw[0]) << 16) | (byte(rgbw[1]) << 8) | (byte(rgbw[2]))); }
|
2021-03-25 19:00:08 +00:00
|
|
|
void colorHStoRGB(uint16_t hue, byte sat, byte* rgb); //hue, sat to rgb
|
|
|
|
void colorKtoRGB(uint16_t kelvin, byte* rgb);
|
|
|
|
void colorCTtoRGB(uint16_t mired, byte* rgb); //white spectrum to rgb
|
|
|
|
void colorXYtoRGB(float x, float y, byte* rgb); // only defined if huesync disabled TODO
|
|
|
|
void colorRGBtoXY(byte* rgb, float* xy); // only defined if huesync disabled TODO
|
|
|
|
void colorFromDecOrHexString(byte* rgb, char* in);
|
|
|
|
bool colorFromHexString(byte* rgb, const char* in);
|
2021-10-16 13:13:30 +00:00
|
|
|
uint32_t colorBalanceFromKelvin(uint16_t kelvin, uint32_t rgb);
|
2021-11-24 10:02:25 +00:00
|
|
|
uint16_t approximateKelvinFromRGB(uint32_t rgb);
|
2022-02-20 21:24:11 +00:00
|
|
|
void setRandomColor(byte* rgb);
|
|
|
|
|
2021-03-25 19:00:08 +00:00
|
|
|
//dmx.cpp
|
|
|
|
void initDMX();
|
|
|
|
void handleDMX();
|
|
|
|
|
|
|
|
//e131.cpp
|
|
|
|
void handleE131Packet(e131_packet_t* p, IPAddress clientIP, byte protocol);
|
2022-09-22 18:34:46 +00:00
|
|
|
void handleArtnetPollReply(IPAddress ipAddress);
|
|
|
|
void prepareArtnetPollReply(ArtPollReply* reply);
|
|
|
|
void sendArtnetPollReply(ArtPollReply* reply, IPAddress ipAddress, uint16_t portAddress);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//file.cpp
|
|
|
|
bool handleFileRead(AsyncWebServerRequest*, String path);
|
|
|
|
bool writeObjectToFileUsingId(const char* file, uint16_t id, JsonDocument* content);
|
|
|
|
bool writeObjectToFile(const char* file, const char* key, JsonDocument* content);
|
|
|
|
bool readObjectFromFileUsingId(const char* file, uint16_t id, JsonDocument* dest);
|
|
|
|
bool readObjectFromFile(const char* file, const char* key, JsonDocument* dest);
|
|
|
|
void updateFSInfo();
|
|
|
|
void closeFile();
|
2024-02-25 16:08:01 +00:00
|
|
|
inline bool writeObjectToFileUsingId(const String &file, uint16_t id, JsonDocument* content) { return writeObjectToFileUsingId(file.c_str(), id, content); };
|
|
|
|
inline bool writeObjectToFile(const String &file, const char* key, JsonDocument* content) { return writeObjectToFile(file.c_str(), key, content); };
|
|
|
|
inline bool readObjectFromFileUsingId(const String &file, uint16_t id, JsonDocument* dest) { return readObjectFromFileUsingId(file.c_str(), id, dest); };
|
|
|
|
inline bool readObjectFromFile(const String &file, const char* key, JsonDocument* dest) { return readObjectFromFile(file.c_str(), key, dest); };
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//hue.cpp
|
|
|
|
void handleHue();
|
|
|
|
void reconnectHue();
|
|
|
|
void onHueError(void* arg, AsyncClient* client, int8_t error);
|
|
|
|
void onHueConnect(void* arg, AsyncClient* client);
|
|
|
|
void sendHuePoll();
|
|
|
|
void onHueData(void* arg, AsyncClient* client, void *data, size_t len);
|
|
|
|
|
2021-11-16 22:20:26 +00:00
|
|
|
//improv.cpp
|
2023-06-26 23:51:44 +00:00
|
|
|
enum ImprovRPCType {
|
|
|
|
Command_Wifi = 0x01,
|
|
|
|
Request_State = 0x02,
|
|
|
|
Request_Info = 0x03,
|
|
|
|
Request_Scan = 0x04
|
|
|
|
};
|
|
|
|
|
2021-11-16 22:20:26 +00:00
|
|
|
void handleImprovPacket();
|
2023-06-26 23:51:44 +00:00
|
|
|
void sendImprovRPCResult(ImprovRPCType type, uint8_t n_strings = 0, const char **strings = nullptr);
|
2021-11-16 22:20:26 +00:00
|
|
|
void sendImprovStateResponse(uint8_t state, bool error = false);
|
|
|
|
void sendImprovInfoResponse();
|
2023-06-26 23:51:44 +00:00
|
|
|
void startImprovWifiScan();
|
|
|
|
void handleImprovWifiScan();
|
|
|
|
void sendImprovIPRPCResult(ImprovRPCType type);
|
2021-11-16 22:20:26 +00:00
|
|
|
|
2021-03-25 19:00:08 +00:00
|
|
|
//ir.cpp
|
|
|
|
void initIR();
|
2024-05-03 13:45:15 +00:00
|
|
|
void deInitIR();
|
2021-03-25 19:00:08 +00:00
|
|
|
void handleIR();
|
|
|
|
|
|
|
|
//json.cpp
|
|
|
|
#include "ESPAsyncWebServer.h"
|
|
|
|
#include "src/dependencies/json/ArduinoJson-v6.h"
|
|
|
|
#include "src/dependencies/json/AsyncJson-v6.h"
|
2024-03-09 19:00:41 +00:00
|
|
|
#include "FX.h"
|
2021-03-25 19:00:08 +00:00
|
|
|
|
2023-03-10 14:20:50 +00:00
|
|
|
bool deserializeSegment(JsonObject elem, byte it, byte presetId = 0);
|
2021-07-09 16:54:28 +00:00
|
|
|
bool deserializeState(JsonObject root, byte callMode = CALL_MODE_DIRECT_CHANGE, byte presetId = 0);
|
2022-07-06 11:13:54 +00:00
|
|
|
void serializeSegment(JsonObject& root, Segment& seg, byte id, bool forPreset = false, bool segmentBounds = true);
|
2022-10-08 16:25:51 +00:00
|
|
|
void serializeState(JsonObject root, bool forPreset = false, bool includeBri = true, bool segmentBounds = true, bool selectedSegmentsOnly = false);
|
2021-03-25 19:00:08 +00:00
|
|
|
void serializeInfo(JsonObject root);
|
2023-05-30 17:36:14 +00:00
|
|
|
void serializeModeNames(JsonArray root);
|
|
|
|
void serializeModeData(JsonArray root);
|
2021-06-19 21:16:40 +00:00
|
|
|
void serveJson(AsyncWebServerRequest* request);
|
2022-02-01 11:02:04 +00:00
|
|
|
#ifdef WLED_ENABLE_JSONLIVE
|
2021-03-25 19:00:08 +00:00
|
|
|
bool serveLiveLeds(AsyncWebServerRequest* request, uint32_t wsClient = 0);
|
2022-02-01 11:02:04 +00:00
|
|
|
#endif
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//led.cpp
|
2022-02-25 09:24:00 +00:00
|
|
|
void setValuesFromSegment(uint8_t s);
|
|
|
|
void setValuesFromMainSeg();
|
2022-02-23 18:20:07 +00:00
|
|
|
void setValuesFromFirstSelectedSeg();
|
2021-03-25 19:00:08 +00:00
|
|
|
void toggleOnOff();
|
2022-02-20 21:24:11 +00:00
|
|
|
void applyBri();
|
|
|
|
void applyFinalBri();
|
2022-02-21 18:44:12 +00:00
|
|
|
void applyValuesToSelectedSegs();
|
2022-02-20 21:24:11 +00:00
|
|
|
void colorUpdated(byte callMode);
|
|
|
|
void stateUpdated(byte callMode);
|
2021-03-25 19:00:08 +00:00
|
|
|
void updateInterfaces(uint8_t callMode);
|
|
|
|
void handleTransitions();
|
|
|
|
void handleNightlight();
|
|
|
|
byte scaledBri(byte in);
|
|
|
|
|
2023-05-30 17:36:14 +00:00
|
|
|
#ifdef WLED_ENABLE_LOXONE
|
2021-03-25 19:00:08 +00:00
|
|
|
//lx_parser.cpp
|
|
|
|
bool parseLx(int lxValue, byte* rgbw);
|
|
|
|
void parseLxJson(int lxValue, byte segId, bool secondary);
|
2023-05-30 17:36:14 +00:00
|
|
|
#endif
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//mqtt.cpp
|
|
|
|
bool initMqtt();
|
|
|
|
void publishMqtt();
|
|
|
|
|
|
|
|
//ntp.cpp
|
2021-05-26 22:09:52 +00:00
|
|
|
void handleTime();
|
2021-03-25 19:00:08 +00:00
|
|
|
void handleNetworkTime();
|
|
|
|
void sendNTPPacket();
|
2023-01-06 08:24:29 +00:00
|
|
|
bool checkNTPResponse();
|
2021-03-25 19:00:08 +00:00
|
|
|
void updateLocalTime();
|
|
|
|
void getTimeString(char* out);
|
|
|
|
bool checkCountdown();
|
|
|
|
void setCountdown();
|
|
|
|
byte weekdayMondayFirst();
|
|
|
|
void checkTimers();
|
|
|
|
void calculateSunriseAndSunset();
|
2021-05-27 00:02:02 +00:00
|
|
|
void setTimeFromAPI(uint32_t timein);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//overlay.cpp
|
|
|
|
void handleOverlayDraw();
|
|
|
|
void _overlayAnalogCountdown();
|
|
|
|
void _overlayAnalogClock();
|
|
|
|
|
|
|
|
//playlist.cpp
|
|
|
|
void shufflePlaylist();
|
|
|
|
void unloadPlaylist();
|
2021-07-11 00:38:31 +00:00
|
|
|
int16_t loadPlaylist(JsonObject playlistObject, byte presetId = 0);
|
2021-03-25 19:00:08 +00:00
|
|
|
void handlePlaylist();
|
2022-10-08 16:25:51 +00:00
|
|
|
void serializePlaylist(JsonObject obj);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//presets.cpp
|
2024-02-25 16:08:01 +00:00
|
|
|
const char *getPresetsFileName(bool persistent = true);
|
2022-11-16 19:55:21 +00:00
|
|
|
void initPresetsFile();
|
2022-09-20 19:17:44 +00:00
|
|
|
void handlePresets();
|
|
|
|
bool applyPreset(byte index, byte callMode = CALL_MODE_DIRECT_CHANGE);
|
2024-03-22 19:49:13 +00:00
|
|
|
bool applyPresetFromPlaylist(byte index);
|
2023-06-22 08:06:19 +00:00
|
|
|
void applyPresetWithFallback(uint8_t presetID, uint8_t callMode, uint8_t effectID = 0, uint8_t paletteID = 0);
|
2021-10-04 18:22:04 +00:00
|
|
|
inline bool applyTemporaryPreset() {return applyPreset(255);};
|
2022-03-15 08:55:23 +00:00
|
|
|
void savePreset(byte index, const char* pname = nullptr, JsonObject saveobj = JsonObject());
|
|
|
|
inline void saveTemporaryPreset() {savePreset(255);};
|
2021-03-25 19:00:08 +00:00
|
|
|
void deletePreset(byte index);
|
2023-01-06 08:24:29 +00:00
|
|
|
bool getPresetName(byte index, String& name);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
2023-06-22 08:06:19 +00:00
|
|
|
//remote.cpp
|
2023-09-10 16:52:14 +00:00
|
|
|
void handleRemote(uint8_t *data, size_t len);
|
2023-06-22 08:06:19 +00:00
|
|
|
|
2021-03-25 19:00:08 +00:00
|
|
|
//set.cpp
|
|
|
|
bool isAsterisksOnly(const char* str, byte maxLen);
|
|
|
|
void handleSettingsSet(AsyncWebServerRequest *request, byte subPage);
|
2024-11-10 23:18:24 +00:00
|
|
|
bool handleHttpApi(AsyncWebServerRequest *request, const String& req, bool apply=true);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//udp.cpp
|
|
|
|
void notify(byte callMode, bool followUp=false);
|
2023-05-30 17:36:14 +00:00
|
|
|
uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, uint8_t *buffer, uint8_t bri=255, bool isRGBW=false);
|
2021-03-25 19:00:08 +00:00
|
|
|
void realtimeLock(uint32_t timeoutMs, byte md = REALTIME_MODE_GENERIC);
|
2022-03-31 22:59:19 +00:00
|
|
|
void exitRealtime();
|
2021-03-25 19:00:08 +00:00
|
|
|
void handleNotifications();
|
|
|
|
void setRealtimePixel(uint16_t i, byte r, byte g, byte b, byte w);
|
|
|
|
void refreshNodeList();
|
|
|
|
void sendSysInfoUDP();
|
2023-09-10 16:52:14 +00:00
|
|
|
#ifndef WLED_DISABLE_ESPNOW
|
2024-02-15 19:40:55 +00:00
|
|
|
void espNowSentCB(uint8_t* address, uint8_t status);
|
2023-09-10 16:52:14 +00:00
|
|
|
void espNowReceiveCB(uint8_t* address, uint8_t* data, uint8_t len, signed int rssi, bool broadcast);
|
|
|
|
#endif
|
2021-03-25 19:00:08 +00:00
|
|
|
|
2021-10-11 12:13:34 +00:00
|
|
|
//network.cpp
|
|
|
|
int getSignalQuality(int rssi);
|
|
|
|
void WiFiEvent(WiFiEvent_t event);
|
|
|
|
|
2021-03-25 19:00:08 +00:00
|
|
|
//um_manager.cpp
|
2022-06-10 22:50:29 +00:00
|
|
|
typedef enum UM_Data_Types {
|
|
|
|
UMT_BYTE = 0,
|
|
|
|
UMT_UINT16,
|
|
|
|
UMT_INT16,
|
|
|
|
UMT_UINT32,
|
|
|
|
UMT_INT32,
|
|
|
|
UMT_FLOAT,
|
|
|
|
UMT_DOUBLE,
|
|
|
|
UMT_BYTE_ARR,
|
|
|
|
UMT_UINT16_ARR,
|
|
|
|
UMT_INT16_ARR,
|
|
|
|
UMT_UINT32_ARR,
|
|
|
|
UMT_INT32_ARR,
|
|
|
|
UMT_FLOAT_ARR,
|
|
|
|
UMT_DOUBLE_ARR
|
|
|
|
} um_types_t;
|
2022-06-08 19:14:01 +00:00
|
|
|
typedef struct UM_Exchange_Data {
|
2022-06-10 14:37:55 +00:00
|
|
|
// should just use: size_t arr_size, void **arr_ptr, byte *ptr_type
|
2022-06-10 22:50:29 +00:00
|
|
|
size_t u_size; // size of u_data array
|
|
|
|
um_types_t *u_type; // array of data types
|
|
|
|
void **u_data; // array of pointers to data
|
2022-06-08 19:14:01 +00:00
|
|
|
UM_Exchange_Data() {
|
2022-06-10 22:50:29 +00:00
|
|
|
u_size = 0;
|
|
|
|
u_type = nullptr;
|
|
|
|
u_data = nullptr;
|
2022-06-08 19:14:01 +00:00
|
|
|
}
|
|
|
|
~UM_Exchange_Data() {
|
2022-06-10 22:50:29 +00:00
|
|
|
if (u_type) delete[] u_type;
|
|
|
|
if (u_data) delete[] u_data;
|
2022-06-08 19:14:01 +00:00
|
|
|
}
|
|
|
|
} um_data_t;
|
2022-06-10 22:50:29 +00:00
|
|
|
const unsigned int um_data_size = sizeof(um_data_t); // 12 bytes
|
2022-06-08 19:14:01 +00:00
|
|
|
|
2021-03-25 19:00:08 +00:00
|
|
|
class Usermod {
|
2022-06-08 19:14:01 +00:00
|
|
|
protected:
|
|
|
|
um_data_t *um_data; // um_data should be allocated using new in (derived) Usermod's setup() or constructor
|
2021-03-25 19:00:08 +00:00
|
|
|
public:
|
2022-06-08 19:14:01 +00:00
|
|
|
Usermod() { um_data = nullptr; }
|
|
|
|
virtual ~Usermod() { if (um_data) delete um_data; }
|
2022-06-11 16:55:23 +00:00
|
|
|
virtual void setup() = 0; // pure virtual, has to be overriden
|
|
|
|
virtual void loop() = 0; // pure virtual, has to be overriden
|
2023-02-05 11:23:05 +00:00
|
|
|
virtual void handleOverlayDraw() {} // called after all effects have been processed, just before strip.show()
|
|
|
|
virtual bool handleButton(uint8_t b) { return false; } // button overrides are possible here
|
|
|
|
virtual bool getUMData(um_data_t **data) { if (data) *data = nullptr; return false; }; // usermod data exchange [see examples for audio effects]
|
|
|
|
virtual void connected() {} // called when WiFi is (re)connected
|
2024-09-18 23:19:40 +00:00
|
|
|
virtual void appendConfigData(Print& settingsScript); // helper function called from usermod settings page to add metadata for entry fields
|
2023-02-05 11:23:05 +00:00
|
|
|
virtual void addToJsonState(JsonObject& obj) {} // add JSON objects for WLED state
|
|
|
|
virtual void addToJsonInfo(JsonObject& obj) {} // add JSON objects for UI Info page
|
|
|
|
virtual void readFromJsonState(JsonObject& obj) {} // process JSON messages received from web server
|
|
|
|
virtual void addToConfig(JsonObject& obj) {} // add JSON entries that go to cfg.json
|
2021-07-05 21:14:57 +00:00
|
|
|
virtual bool readFromConfig(JsonObject& obj) { return true; } // Note as of 2021-06 readFromConfig() now needs to return a bool, see usermod_v2_example.h
|
2023-02-05 11:23:05 +00:00
|
|
|
virtual void onMqttConnect(bool sessionPresent) {} // fired when MQTT connection is established (so usermod can subscribe)
|
|
|
|
virtual bool onMqttMessage(char* topic, char* payload) { return false; } // fired upon MQTT message received (wled topic)
|
2024-05-25 18:01:38 +00:00
|
|
|
virtual bool onEspNowMessage(uint8_t* sender, uint8_t* payload, uint8_t len) { return false; } // fired upon ESP-NOW message received
|
2023-02-05 11:23:05 +00:00
|
|
|
virtual void onUpdateBegin(bool) {} // fired prior to and after unsuccessful firmware update
|
|
|
|
virtual void onStateChange(uint8_t mode) {} // fired upon WLED state change
|
2021-03-25 19:00:08 +00:00
|
|
|
virtual uint16_t getId() {return USERMOD_ID_UNSPECIFIED;}
|
2024-09-17 22:26:46 +00:00
|
|
|
|
|
|
|
// API shims
|
|
|
|
private:
|
|
|
|
static Print* oappend_shim;
|
|
|
|
// old form of appendConfigData; called by default appendConfigData(Print&) with oappend_shim set up
|
|
|
|
// private so it is not accidentally invoked except via Usermod::appendConfigData(Print&)
|
|
|
|
virtual void appendConfigData() {}
|
|
|
|
protected:
|
|
|
|
// Shim for oappend(), which used to exist in utils.cpp
|
|
|
|
template<typename T> static inline void oappend(const T& t) { oappend_shim->print(t); };
|
2024-09-28 02:33:20 +00:00
|
|
|
#ifdef ESP8266
|
|
|
|
// Handle print(PSTR()) without crashing by detecting PROGMEM strings
|
|
|
|
static void oappend(const char* c) { if ((intptr_t) c >= 0x40000000) oappend_shim->print(FPSTR(c)); else oappend_shim->print(c); };
|
|
|
|
#endif
|
2021-03-25 19:00:08 +00:00
|
|
|
};
|
|
|
|
|
2021-05-13 14:19:53 +00:00
|
|
|
class UsermodManager {
|
2021-03-25 19:00:08 +00:00
|
|
|
private:
|
2024-09-19 19:44:11 +00:00
|
|
|
static Usermod* ums[WLED_MAX_USERMODS];
|
|
|
|
static byte numMods;
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
public:
|
2024-09-19 19:44:11 +00:00
|
|
|
static void loop();
|
|
|
|
static void handleOverlayDraw();
|
|
|
|
static bool handleButton(uint8_t b);
|
|
|
|
static bool getUMData(um_data_t **um_data, uint8_t mod_id = USERMOD_ID_RESERVED); // USERMOD_ID_RESERVED will poll all usermods
|
|
|
|
static void setup();
|
|
|
|
static void connected();
|
2024-09-24 19:43:47 +00:00
|
|
|
static void appendConfigData(Print&);
|
2024-09-19 19:44:11 +00:00
|
|
|
static void addToJsonState(JsonObject& obj);
|
|
|
|
static void addToJsonInfo(JsonObject& obj);
|
|
|
|
static void readFromJsonState(JsonObject& obj);
|
|
|
|
static void addToConfig(JsonObject& obj);
|
|
|
|
static bool readFromConfig(JsonObject& obj);
|
2024-05-25 18:01:38 +00:00
|
|
|
#ifndef WLED_DISABLE_MQTT
|
2024-09-19 19:44:11 +00:00
|
|
|
static void onMqttConnect(bool sessionPresent);
|
|
|
|
static bool onMqttMessage(char* topic, char* payload);
|
2024-05-25 18:01:38 +00:00
|
|
|
#endif
|
|
|
|
#ifndef WLED_DISABLE_ESPNOW
|
2024-09-19 19:44:11 +00:00
|
|
|
static bool onEspNowMessage(uint8_t* sender, uint8_t* payload, uint8_t len);
|
2024-05-25 18:01:38 +00:00
|
|
|
#endif
|
2024-09-19 19:44:11 +00:00
|
|
|
static void onUpdateBegin(bool);
|
|
|
|
static void onStateChange(uint8_t);
|
|
|
|
static bool add(Usermod* um);
|
|
|
|
static Usermod* lookup(uint16_t mod_id);
|
|
|
|
static inline byte getModCount() {return numMods;};
|
2021-03-25 19:00:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
//usermods_list.cpp
|
|
|
|
void registerUsermods();
|
|
|
|
|
|
|
|
//usermod.cpp
|
|
|
|
void userSetup();
|
|
|
|
void userConnected();
|
|
|
|
void userLoop();
|
|
|
|
|
2021-10-11 12:13:34 +00:00
|
|
|
//util.cpp
|
2021-11-23 19:20:19 +00:00
|
|
|
int getNumVal(const String* req, uint16_t pos);
|
|
|
|
void parseNumber(const char* str, byte* val, byte minv=0, byte maxv=255);
|
|
|
|
bool getVal(JsonVariant elem, byte* val, byte minv=0, byte maxv=255);
|
2023-11-06 19:08:45 +00:00
|
|
|
bool getBoolVal(JsonVariant elem, bool dflt);
|
2022-06-03 20:22:18 +00:00
|
|
|
bool updateVal(const char* req, const char* key, byte* val, byte minv=0, byte maxv=255);
|
2024-09-18 23:19:40 +00:00
|
|
|
size_t printSetFormCheckbox(Print& settingsScript, const char* key, int val);
|
|
|
|
size_t printSetFormValue(Print& settingsScript, const char* key, int val);
|
|
|
|
size_t printSetFormValue(Print& settingsScript, const char* key, const char* val);
|
|
|
|
size_t printSetFormIndex(Print& settingsScript, const char* key, int index);
|
|
|
|
size_t printSetClassElementHTML(Print& settingsScript, const char* key, const int index, const char* val);
|
2021-10-11 12:13:34 +00:00
|
|
|
void prepareHostname(char* hostname);
|
|
|
|
bool isAsterisksOnly(const char* str, byte maxLen);
|
2021-11-14 15:56:34 +00:00
|
|
|
bool requestJSONBufferLock(uint8_t module=255);
|
2021-11-12 22:33:10 +00:00
|
|
|
void releaseJSONBufferLock();
|
2022-01-31 19:35:11 +00:00
|
|
|
uint8_t extractModeName(uint8_t mode, const char *src, char *dest, uint8_t maxLen);
|
2022-07-23 20:00:19 +00:00
|
|
|
uint8_t extractModeSlider(uint8_t mode, uint8_t slider, char *dest, uint8_t maxLen, uint8_t *var = nullptr);
|
|
|
|
int16_t extractModeDefaults(uint8_t mode, const char *segVar);
|
2023-06-14 09:53:39 +00:00
|
|
|
void checkSettingsPIN(const char *pin);
|
2022-05-20 12:48:40 +00:00
|
|
|
uint16_t crc16(const unsigned char* data_p, size_t length);
|
2022-07-25 19:31:50 +00:00
|
|
|
um_data_t* simulateSound(uint8_t simulationId);
|
2022-08-10 18:53:11 +00:00
|
|
|
void enumerateLedmaps();
|
2023-09-10 16:52:14 +00:00
|
|
|
uint8_t get_random_wheel_index(uint8_t pos);
|
2024-09-15 09:04:02 +00:00
|
|
|
float mapf(float x, float in_min, float in_max, float out_min, float out_max);
|
2021-10-11 12:13:34 +00:00
|
|
|
|
2024-01-06 15:05:18 +00:00
|
|
|
// RAII guard class for the JSON Buffer lock
|
|
|
|
// Modeled after std::lock_guard
|
|
|
|
class JSONBufferGuard {
|
|
|
|
bool holding_lock;
|
|
|
|
public:
|
|
|
|
inline JSONBufferGuard(uint8_t module=255) : holding_lock(requestJSONBufferLock(module)) {};
|
|
|
|
inline ~JSONBufferGuard() { if (holding_lock) releaseJSONBufferLock(); };
|
|
|
|
inline JSONBufferGuard(const JSONBufferGuard&) = delete; // Noncopyable
|
|
|
|
inline JSONBufferGuard& operator=(const JSONBufferGuard&) = delete;
|
|
|
|
inline JSONBufferGuard(JSONBufferGuard&& r) : holding_lock(r.holding_lock) { r.holding_lock = false; }; // but movable
|
|
|
|
inline JSONBufferGuard& operator=(JSONBufferGuard&& r) { holding_lock |= r.holding_lock; r.holding_lock = false; return *this; };
|
|
|
|
inline bool owns_lock() const { return holding_lock; }
|
|
|
|
explicit inline operator bool() const { return owns_lock(); };
|
|
|
|
inline void release() { if (holding_lock) releaseJSONBufferLock(); holding_lock = false; }
|
|
|
|
};
|
|
|
|
|
2022-07-30 09:04:04 +00:00
|
|
|
#ifdef WLED_ADD_EEPROM_SUPPORT
|
2021-03-25 19:00:08 +00:00
|
|
|
//wled_eeprom.cpp
|
|
|
|
void applyMacro(byte index);
|
|
|
|
void deEEP();
|
|
|
|
void deEEPSettings();
|
|
|
|
void clearEEPROM();
|
2022-07-30 09:04:04 +00:00
|
|
|
#endif
|
2021-03-25 19:00:08 +00:00
|
|
|
|
2021-10-11 12:13:34 +00:00
|
|
|
//wled_math.cpp
|
2024-05-10 13:59:11 +00:00
|
|
|
#if defined(ESP8266) && !defined(WLED_USE_REAL_MATH)
|
2022-02-10 15:09:16 +00:00
|
|
|
template <typename T> T atan_t(T x);
|
2021-10-11 12:13:34 +00:00
|
|
|
float cos_t(float phi);
|
|
|
|
float sin_t(float x);
|
|
|
|
float tan_t(float x);
|
|
|
|
float acos_t(float x);
|
|
|
|
float asin_t(float x);
|
|
|
|
float floor_t(float x);
|
|
|
|
float fmod_t(float num, float denom);
|
|
|
|
#else
|
|
|
|
#include <math.h>
|
2024-05-09 21:58:58 +00:00
|
|
|
#define sin_t sinf
|
|
|
|
#define cos_t cosf
|
|
|
|
#define tan_t tanf
|
|
|
|
#define asin_t asinf
|
|
|
|
#define acos_t acosf
|
|
|
|
#define atan_t atanf
|
|
|
|
#define fmod_t fmodf
|
|
|
|
#define floor_t floorf
|
2021-10-11 12:13:34 +00:00
|
|
|
#endif
|
|
|
|
|
2021-03-25 19:00:08 +00:00
|
|
|
//wled_serial.cpp
|
|
|
|
void handleSerial();
|
2022-02-01 19:02:46 +00:00
|
|
|
void updateBaudRate(uint32_t rate);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//wled_server.cpp
|
2022-03-01 22:37:28 +00:00
|
|
|
void createEditHandler(bool enable);
|
2021-03-25 19:00:08 +00:00
|
|
|
void initServer();
|
|
|
|
void serveMessage(AsyncWebServerRequest* request, uint16_t code, const String& headl, const String& subl="", byte optionT=255);
|
2023-12-12 14:45:57 +00:00
|
|
|
void serveJsonError(AsyncWebServerRequest* request, uint16_t code, uint16_t error);
|
2021-03-25 19:00:08 +00:00
|
|
|
void serveSettings(AsyncWebServerRequest* request, bool post = false);
|
2021-12-06 19:53:09 +00:00
|
|
|
void serveSettingsJS(AsyncWebServerRequest* request);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
//ws.cpp
|
|
|
|
void handleWs();
|
|
|
|
void wsEvent(AsyncWebSocket * server, AsyncWebSocketClient * client, AwsEventType type, void * arg, uint8_t *data, size_t len);
|
|
|
|
void sendDataWs(AsyncWebSocketClient * client = nullptr);
|
|
|
|
|
|
|
|
//xml.cpp
|
2024-09-06 01:13:55 +00:00
|
|
|
void XML_response(Print& dest);
|
|
|
|
void getSettingsJS(byte subPage, Print& dest);
|
2021-03-25 19:00:08 +00:00
|
|
|
|
|
|
|
#endif
|