2023-12-28 19:36:18 +00:00
|
|
|
#include <ESPMegaDisplay.hpp>
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Receives and processes serial commands from the display adapter.
|
|
|
|
* @param process Flag indicating whether to process the received commands.
|
|
|
|
* @return True if data is received, false otherwise.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
bool ESPMegaDisplay::recieveSerialCommand(bool process){
|
|
|
|
bool dataRecieved = false;
|
2023-12-28 19:36:18 +00:00
|
|
|
// Read the serial buffer if available
|
|
|
|
while(displayAdapter->available()) {
|
|
|
|
rx_buffer[rx_buffer_index] = displayAdapter->read();
|
|
|
|
rx_buffer_index++;
|
|
|
|
// Check for overflow
|
|
|
|
if(rx_buffer_index>=256){
|
|
|
|
rx_buffer_index = 0;
|
|
|
|
}
|
2023-12-29 04:55:59 +00:00
|
|
|
if(process) this-> processSerialCommand();
|
|
|
|
dataRecieved = true;
|
2023-12-28 19:36:18 +00:00
|
|
|
}
|
2023-12-29 04:55:59 +00:00
|
|
|
return dataRecieved;
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Receives and processes serial commands from the display adapter.
|
|
|
|
* @return True if data is received, false otherwise.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
bool ESPMegaDisplay::recieveSerialCommand(){
|
|
|
|
return recieveSerialCommand(true);
|
2023-12-28 19:36:18 +00:00
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Processes the received serial command.
|
|
|
|
*/
|
2023-12-28 19:36:18 +00:00
|
|
|
void ESPMegaDisplay::processSerialCommand(){
|
|
|
|
// Check if the rx buffer ended with stop bytes (0xFF 0xFF 0xFF)
|
2023-12-29 04:55:59 +00:00
|
|
|
if(!payloadIsValid()) return;
|
2023-12-28 19:36:18 +00:00
|
|
|
// The rx buffer ended with stop bytes
|
|
|
|
// Check if the rx buffer is a push payload
|
|
|
|
// The payload type is the first byte of the payload
|
|
|
|
// 0x00 is invalid instruction
|
|
|
|
// 0x01 is success
|
|
|
|
// 0x65 is a touch event
|
|
|
|
// 0x66 is a page report event
|
|
|
|
if(rx_buffer[0]==0x65){
|
|
|
|
processTouchPayload();
|
|
|
|
}
|
|
|
|
else if(rx_buffer[0]==0x66){
|
|
|
|
processPageReportPayload();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Processes the touch event payload.
|
|
|
|
*/
|
2023-12-28 19:36:18 +00:00
|
|
|
void ESPMegaDisplay::processTouchPayload() {
|
|
|
|
if (rx_buffer_index != 4) return;
|
|
|
|
// The second byte of the payload is the page number
|
|
|
|
uint8_t page = rx_buffer[1];
|
|
|
|
// The third byte of the payload is the component id
|
|
|
|
uint8_t component = rx_buffer[2];
|
|
|
|
// The fourth byte of the payload is the event
|
|
|
|
uint8_t event = rx_buffer[3];
|
|
|
|
// 0x01 is press, 0x00 is release
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Processes the page report event payload.
|
|
|
|
*/
|
2023-12-28 19:36:18 +00:00
|
|
|
void ESPMegaDisplay::processPageReportPayload() {
|
|
|
|
// The second byte of the payload is the page number
|
|
|
|
this->currentPage = rx_buffer[1];
|
|
|
|
if(pageChangeCallback!=NULL){
|
|
|
|
pageChangeCallback(this->currentPage);
|
|
|
|
}
|
|
|
|
rx_buffer_index = 0;
|
2023-12-29 04:55:59 +00:00
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Sends stop bytes to the display adapter.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
void ESPMegaDisplay::sendStopBytes() {
|
|
|
|
displayAdapter->write(0xFF);
|
|
|
|
displayAdapter->write(0xFF);
|
|
|
|
displayAdapter->write(0xFF);
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Sends a command to the display adapter.
|
|
|
|
* @param command The command to send.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
void ESPMegaDisplay::sendCommand(char* command) {
|
|
|
|
displayAdapter->print(command);
|
|
|
|
sendStopBytes();
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Jumps to the specified page on the display.
|
|
|
|
* @param page The page number to jump to.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
void ESPMegaDisplay::jumpToPage(int page) {
|
|
|
|
this->displayAdapter->print("page ");
|
|
|
|
this->displayAdapter->print(page);
|
|
|
|
sendStopBytes();
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Sets the value of a number component on the display.
|
|
|
|
* @param component The component name.
|
|
|
|
* @param value The value to set.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
void ESPMegaDisplay::setNumber(char* component, int value) {
|
|
|
|
this->displayAdapter->print(component);
|
|
|
|
this->displayAdapter->print("=");
|
|
|
|
this->displayAdapter->print(value);
|
|
|
|
sendStopBytes();
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Sets the value of a string component on the display.
|
|
|
|
* @param component The component name.
|
|
|
|
* @param value The value to set.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
void ESPMegaDisplay::setString(char* component, char* value) {
|
|
|
|
this->displayAdapter->print(component);
|
|
|
|
this->displayAdapter->print("=\"");
|
|
|
|
this->displayAdapter->print(value);
|
|
|
|
this->displayAdapter->print("\"");
|
|
|
|
sendStopBytes();
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Gets the value of a number component from the display.
|
|
|
|
* @param component The component name.
|
|
|
|
* @return The value of the number component.
|
|
|
|
*/
|
|
|
|
uint32_t ESPMegaDisplay::getNumber(char* component) {
|
|
|
|
uint32_t start = millis();
|
|
|
|
// Send the get command
|
|
|
|
this->displayAdapter->print("get ");
|
|
|
|
this->displayAdapter->print(component);
|
|
|
|
sendStopBytes();
|
|
|
|
// Wait for the response
|
|
|
|
if(!waitForValidPayload(DISPLAY_FETCH_TIMEOUT)) return 0;
|
|
|
|
// The rx buffer is valid
|
|
|
|
// The expected payload is type 0x71
|
|
|
|
if(rx_buffer[0]!=0x71) return 0;
|
|
|
|
// The 2nd to 5th byte of the payload is the value
|
|
|
|
// It's a 4 byte 32-bit value in little endian order.
|
|
|
|
// Convert the 4 bytes to a 32-bit value
|
|
|
|
uint32_t value = 0;
|
|
|
|
value |= rx_buffer[1];
|
|
|
|
value |= rx_buffer[2]<<8;
|
|
|
|
value |= rx_buffer[3]<<16;
|
|
|
|
value |= rx_buffer[4]<<24;
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Gets the value of a string component from the display.
|
|
|
|
* @param component The component name.
|
|
|
|
* @return The value of the string component.
|
|
|
|
* @note The returned char array must be freed after use.
|
|
|
|
*/
|
|
|
|
char* ESPMegaDisplay::getString(char* component) {
|
2023-12-29 04:55:59 +00:00
|
|
|
uint32_t start = millis();
|
|
|
|
// Send the get command
|
|
|
|
this->displayAdapter->print("get ");
|
|
|
|
this->displayAdapter->print(component);
|
|
|
|
sendStopBytes();
|
|
|
|
// Wait for the response
|
2023-12-29 13:04:25 +00:00
|
|
|
if(!waitForValidPayload(DISPLAY_FETCH_TIMEOUT)) return "";
|
2023-12-29 04:55:59 +00:00
|
|
|
// The rx buffer is valid
|
2023-12-29 13:04:25 +00:00
|
|
|
// The expected payload is type 0x70
|
|
|
|
if(rx_buffer[0]!=0x70) return "";
|
|
|
|
// The 2nd bytes onwards before the stop bytes is the string
|
|
|
|
// The length of the string is the length of the payload minus 4
|
|
|
|
uint8_t length = rx_buffer_index-4;
|
|
|
|
// First we malloc a char array with the length of the string
|
|
|
|
char* value = (char*)malloc(length+1);
|
|
|
|
// Copy the string from the rx buffer to the char array
|
|
|
|
memcpy(value, rx_buffer+1, length);
|
|
|
|
// Add the null terminator
|
|
|
|
value[length] = '\0';
|
|
|
|
// Return the char array
|
|
|
|
return value;
|
2023-12-29 04:55:59 +00:00
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
bool ESPMegaDisplay::getStringToBuffer(char* component, char* buffer, uint8_t buffer_size) {
|
|
|
|
uint32_t start = millis();
|
|
|
|
// Send the get command
|
|
|
|
this->displayAdapter->print("get ");
|
|
|
|
this->displayAdapter->print(component);
|
|
|
|
sendStopBytes();
|
|
|
|
// Wait for the response
|
|
|
|
if(!waitForValidPayload(DISPLAY_FETCH_TIMEOUT)) return false;
|
|
|
|
// The rx buffer is valid
|
|
|
|
// The expected payload is type 0x70
|
|
|
|
if(rx_buffer[0]!=0x70) return false;
|
|
|
|
// The 2nd bytes onwards before the stop bytes is the string
|
|
|
|
// The length of the string is the length of the payload minus 4
|
|
|
|
uint8_t length = rx_buffer_index-4;
|
|
|
|
// Check if the buffer is large enough to hold the string
|
|
|
|
if(length>buffer_size) return false;
|
|
|
|
// Copy the string from the rx buffer to the char array
|
|
|
|
memcpy(buffer, rx_buffer+1, length);
|
|
|
|
// Add the null terminator
|
|
|
|
buffer[length] = '\0';
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Waits for a valid payload from the display adapter.
|
|
|
|
* @param timeout The timeout value in milliseconds.
|
|
|
|
* @return True if a valid payload is received, false otherwise.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
bool ESPMegaDisplay::waitForValidPayload(uint32_t timeout) {
|
|
|
|
uint32_t start = millis();
|
|
|
|
// If the payload is not valid, keep reading the serial buffer until timeout
|
|
|
|
while(!this->payloadIsValid()){
|
|
|
|
if(millis()-start>timeout){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
recieveSerialCommand(false);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-12-29 13:04:25 +00:00
|
|
|
/**
|
|
|
|
* @brief Checks if the received payload is valid.
|
|
|
|
* @return True if the payload is valid, false otherwise.
|
|
|
|
*/
|
2023-12-29 04:55:59 +00:00
|
|
|
bool ESPMegaDisplay::payloadIsValid() {
|
|
|
|
// Check if the rx buffer ended with stop bytes (0xFF 0xFF 0xFF)
|
|
|
|
if(rx_buffer_index<3) return false;
|
|
|
|
if(rx_buffer[rx_buffer_index-1]!=0xFF) return false;
|
|
|
|
if(rx_buffer[rx_buffer_index-2]!=0xFF) return false;
|
|
|
|
if(rx_buffer[rx_buffer_index-3]!=0xFF) return false;
|
|
|
|
return true;
|
2023-12-29 13:04:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Sets the brightness of the display.
|
|
|
|
* @param value The brightness value.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::setBrightness(int value) {
|
|
|
|
this->displayAdapter->print("dim=");
|
|
|
|
this->displayAdapter->print(value);
|
|
|
|
sendStopBytes();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Sets the volume of the display.
|
|
|
|
* @param value The volume value.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::setVolume(int value) {
|
|
|
|
this->displayAdapter->print("vol=");
|
|
|
|
this->displayAdapter->print(value);
|
|
|
|
sendStopBytes();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Restarts the display.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::reset() {
|
|
|
|
// First we send a stop bytes to clear the serial buffer
|
|
|
|
// This ensures that the display is ready to receive the reset command
|
|
|
|
sendStopBytes();
|
|
|
|
this->displayAdapter->print("rest");
|
|
|
|
sendStopBytes();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Registers a callback function that will be called when a push event is received.
|
|
|
|
* @param callback The callback function.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::registerPushCallback(std::function<void(uint8_t, uint8_t)> callback) {
|
|
|
|
this->pushCallback = callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Registers a callback function that will be called when a pop event is received.
|
|
|
|
* @param callback The callback function.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::registerPopCallback(std::function<void(uint8_t, uint8_t)> callback) {
|
|
|
|
this->popCallback = callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Registers a callback function that will be called when a page change event is received.
|
|
|
|
* @param callback The callback function.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::registerPageChangeCallback(std::function<void(uint8_t)> callback) {
|
|
|
|
this->pageChangeCallback = callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Unregisters the push callback function.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::unregisterPushCallback() {
|
|
|
|
this->pushCallback = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Unregisters the pop callback function.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::unregisterPopCallback() {
|
|
|
|
this->popCallback = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Unregisters the page change callback function.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::unregisterPageChangeCallback() {
|
|
|
|
this->pageChangeCallback = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Constructor for the ESPMegaDisplay class.
|
|
|
|
* @param displayAdapter The serial adapter connected to the display.
|
|
|
|
*/
|
|
|
|
ESPMegaDisplay::ESPMegaDisplay(HardwareSerial *displayAdapter) {
|
|
|
|
this->displayAdapter = displayAdapter;
|
|
|
|
this->currentPage = 0;
|
|
|
|
this->rx_buffer_index = 0;
|
|
|
|
this->pushCallback = NULL;
|
|
|
|
this->popCallback = NULL;
|
|
|
|
this->pageChangeCallback = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initializes the display.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::begin() {
|
|
|
|
this->displayAdapter->begin(115200);
|
|
|
|
this->displayAdapter->setTimeout(100);
|
|
|
|
this->displayAdapter->flush();
|
|
|
|
this->reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The main loop function of the display.
|
|
|
|
*/
|
|
|
|
void ESPMegaDisplay::loop() {
|
|
|
|
// Check if there is data in the serial buffer
|
|
|
|
// If there is data, process the data
|
|
|
|
recieveSerialCommand();
|
2023-12-28 19:36:18 +00:00
|
|
|
}
|