2021-04-11 11:32:02 +01:00
|
|
|
|
/*
|
|
|
|
|
xdsp_17_universal.ino - universal display driver support for Tasmota
|
|
|
|
|
|
|
|
|
|
Copyright (C) 2021 Gerhard Mutz and Theo Arends
|
|
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
#ifdef USE_DISPLAY
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#ifdef USE_UNIVERSAL_DISPLAY
|
|
|
|
|
|
|
|
|
|
#define XDSP_17 17
|
|
|
|
|
|
|
|
|
|
#include <uDisplay.h>
|
|
|
|
|
|
|
|
|
|
bool udisp_init_done = false;
|
2021-04-14 13:26:59 +01:00
|
|
|
|
uint8_t ctouch_counter;
|
2021-04-30 14:26:41 +01:00
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#ifdef USE_UFILESYS
|
2021-04-14 13:26:59 +01:00
|
|
|
|
extern FS *ffsp;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#endif
|
|
|
|
|
|
2022-12-18 13:06:04 +00:00
|
|
|
|
#undef GT911_address
|
|
|
|
|
#define GT911_address 0x5D
|
2023-12-18 10:03:18 +00:00
|
|
|
|
#undef CST816S_address
|
|
|
|
|
#define CST816S_address 0x15
|
2022-10-03 11:24:27 +01:00
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
enum { GPIO_DP_RES = GPIO_SENSOR_END-1,GPIO_DP_CS,GPIO_DP_RS,GPIO_DP_WR,GPIO_DP_RD,GPIO_DPAR0,GPIO_DPAR1,GPIO_DPAR2,GPIO_DPAR3,GPIO_DPAR4,GPIO_DPAR5,GPIO_DPAR6,GPIO_DPAR7,GPIO_DPAR8,GPIO_DPAR9,GPIO_DPAR10,GPIO_DPAR11,GPIO_DPAR12,GPIO_DPAR13,GPIO_DPAR14,GPIO_DPAR15};
|
2022-10-03 11:24:27 +01:00
|
|
|
|
|
2021-04-30 14:26:41 +01:00
|
|
|
|
#ifndef USE_DISPLAY
|
|
|
|
|
uint8_t color_type;
|
|
|
|
|
uint16_t fg_color;
|
|
|
|
|
uint16_t bg_color;
|
|
|
|
|
Renderer *renderer;
|
|
|
|
|
#else
|
|
|
|
|
extern uint8_t color_type;
|
|
|
|
|
extern uint16_t fg_color;
|
|
|
|
|
extern uint16_t bg_color;
|
|
|
|
|
#endif
|
|
|
|
|
|
2023-01-17 09:19:06 +00:00
|
|
|
|
#ifndef DISPDESC_SIZE
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#define DISPDESC_SIZE 1000
|
2023-01-17 09:19:06 +00:00
|
|
|
|
#endif
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
2021-04-25 14:14:50 +01:00
|
|
|
|
void Core2DisplayPower(uint8_t on);
|
|
|
|
|
void Core2DisplayDim(uint8_t dim);
|
|
|
|
|
|
2021-04-14 13:26:59 +01:00
|
|
|
|
//#define DSP_ROM_DESC
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
2021-05-01 08:54:05 +01:00
|
|
|
|
#ifndef DISP_DESC_FILE
|
|
|
|
|
//#define DISP_DESC_FILE "/dispdesc.txt"
|
|
|
|
|
#define DISP_DESC_FILE "/display.ini"
|
2021-12-08 15:32:02 +00:00
|
|
|
|
#endif // DISP_DESC_FILE
|
2021-05-01 08:54:05 +01:00
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
/*********************************************************************************************/
|
2021-04-21 16:11:19 +01:00
|
|
|
|
#ifdef DSP_ROM_DESC
|
2022-04-01 16:52:43 +01:00
|
|
|
|
const char DSP_SAMPLE_DESC[] PROGMEM = DSP_ROM_DESC;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#endif // DSP_ROM_DESC
|
|
|
|
|
/*********************************************************************************************/
|
2021-11-17 21:48:48 +00:00
|
|
|
|
Renderer *Init_uDisplay(const char *desc) {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
char *ddesc = 0;
|
|
|
|
|
char *fbuff;
|
|
|
|
|
uDisplay *udisp;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
2021-04-21 16:11:19 +01:00
|
|
|
|
if (TasmotaGlobal.gpio_optiona.udisplay_driver || desc) {
|
|
|
|
|
|
2021-06-11 17:14:12 +01:00
|
|
|
|
Settings->display_model = XDSP_17;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
2021-05-01 08:54:05 +01:00
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
fbuff = (char*)calloc(DISPDESC_SIZE, 1);
|
2021-04-21 10:01:40 +01:00
|
|
|
|
if (!fbuff) return 0;
|
|
|
|
|
|
|
|
|
|
if (desc) {
|
2021-04-21 16:11:19 +01:00
|
|
|
|
memcpy_P(fbuff, desc, DISPDESC_SIZE - 1);
|
2021-04-21 10:01:40 +01:00
|
|
|
|
ddesc = fbuff;
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("DSP: const char descriptor used"));
|
2021-04-21 10:01:40 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#ifdef USE_UFILESYS
|
2021-04-21 10:01:40 +01:00
|
|
|
|
if (ffsp && !TasmotaGlobal.no_autoexec && !ddesc) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
File fp;
|
2021-05-01 08:54:05 +01:00
|
|
|
|
fp = ffsp->open(DISP_DESC_FILE, "r");
|
2021-04-11 11:32:02 +01:00
|
|
|
|
if (fp > 0) {
|
|
|
|
|
uint32_t size = fp.size();
|
2023-01-17 09:19:06 +00:00
|
|
|
|
if (size > DISPDESC_SIZE - 50) {
|
|
|
|
|
free(fbuff);
|
|
|
|
|
fbuff = (char*)calloc(size + 50, 1);
|
|
|
|
|
if (!fbuff) return 0;
|
|
|
|
|
}
|
2021-04-11 11:32:02 +01:00
|
|
|
|
fp.read((uint8_t*)fbuff, size);
|
|
|
|
|
fp.close();
|
|
|
|
|
ddesc = fbuff;
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("DSP: File descriptor used"));
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-08 15:32:02 +00:00
|
|
|
|
#endif // USE_UFILESYS
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
|
|
|
|
#ifdef USE_SCRIPT
|
2021-06-11 17:14:12 +01:00
|
|
|
|
if (bitRead(Settings->rule_enabled, 0) && !ddesc) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
uint8_t dfound = Run_Scripter(">d",-2,0);
|
|
|
|
|
if (dfound == 99) {
|
|
|
|
|
char *lp = glob_script_mem.section_ptr + 2;
|
|
|
|
|
while (*lp != '\n') lp++;
|
|
|
|
|
memcpy(fbuff, lp + 1, DISPDESC_SIZE - 1);
|
|
|
|
|
ddesc = fbuff;
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("DSP: Script descriptor used"));
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif // USE_SCRIPT
|
|
|
|
|
|
2021-04-16 18:36:45 +01:00
|
|
|
|
#ifdef USE_RULES
|
2021-06-11 17:14:12 +01:00
|
|
|
|
if (!bitRead(Settings->rule_enabled, 2) && !ddesc) {
|
2021-04-16 18:36:45 +01:00
|
|
|
|
// only if rule3 is not enabled for rules
|
2021-06-11 17:14:12 +01:00
|
|
|
|
char *cp = Settings->rules[2];
|
2021-04-16 18:36:45 +01:00
|
|
|
|
while (*cp == ' ') cp++;
|
|
|
|
|
memcpy(fbuff, cp, DISPDESC_SIZE - 1);
|
|
|
|
|
if (fbuff[0] == ':' && fbuff[1] == 'H') {
|
|
|
|
|
// assume display descriptor, replace space with line feed
|
|
|
|
|
for (uint32_t cnt = 0; cnt < DISPDESC_SIZE; cnt++) {
|
|
|
|
|
if (fbuff[cnt] == ' ') fbuff[cnt] = '\n';
|
|
|
|
|
}
|
|
|
|
|
ddesc = fbuff;
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("DSP: Rule 3 descriptor used"));
|
2021-04-16 18:36:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
#endif // USE_RULES
|
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#ifdef DSP_ROM_DESC
|
|
|
|
|
if (!ddesc) {
|
|
|
|
|
memcpy_P(fbuff, DSP_SAMPLE_DESC, sizeof(DSP_SAMPLE_DESC));
|
|
|
|
|
ddesc = fbuff;
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("DSP: Flash descriptor used"));
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
|
|
|
|
#endif // DSP_ROM_DESC
|
|
|
|
|
|
|
|
|
|
if (!ddesc) {
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("DSP: No valid descriptor found"));
|
2021-04-11 11:32:02 +01:00
|
|
|
|
if (fbuff) free(fbuff);
|
2021-04-21 10:01:40 +01:00
|
|
|
|
return 0;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
2023-01-17 09:19:06 +00:00
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// Replace tasmota vars before passing to driver uDisplay.cpp
|
|
|
|
|
char *cp = strstr(ddesc, "I2C");
|
2024-07-12 15:36:04 +01:00
|
|
|
|
if (cp) {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// I2C,3c,22,21,*
|
|
|
|
|
// I2C1,3c,22,21,*
|
|
|
|
|
// I2C2,3c,22,21,*
|
2024-07-12 15:36:04 +01:00
|
|
|
|
cp += 3;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// ,3c,22,21,*
|
|
|
|
|
// 1,3c,22,21,*
|
|
|
|
|
// 2,3c,22,21,*
|
|
|
|
|
uint8_t wire_n = 0;
|
2021-04-21 10:01:40 +01:00
|
|
|
|
if (*cp == '1' || *cp == '2') {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
wire_n = (*cp & 3) - 1;
|
2021-04-21 10:01:40 +01:00
|
|
|
|
cp += 2;
|
|
|
|
|
} else {
|
|
|
|
|
cp++;
|
|
|
|
|
}
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 3c,22,21,-1
|
2021-04-14 13:26:59 +01:00
|
|
|
|
uint8_t i2caddr = strtol(cp, &cp, 16);
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// ,22,21,-1
|
2024-07-12 13:57:27 +01:00
|
|
|
|
cp++;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 22,21,-1
|
|
|
|
|
int8_t scl = replacepin(&cp, Pin(GPIO_I2C_SCL, wire_n));
|
|
|
|
|
int8_t sda = replacepin(&cp, Pin(GPIO_I2C_SDA, wire_n));
|
2021-04-14 13:26:59 +01:00
|
|
|
|
replacepin(&cp, Pin(GPIO_OLED_RESET));
|
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
if (wire_n == 0) {
|
2023-05-07 13:48:49 +01:00
|
|
|
|
if (!TasmotaGlobal.i2c_enabled) {
|
|
|
|
|
I2cBegin(sda, scl);
|
|
|
|
|
}
|
2021-04-21 10:01:40 +01:00
|
|
|
|
}
|
|
|
|
|
#ifdef ESP32
|
2024-07-13 13:58:24 +01:00
|
|
|
|
if (wire_n == 1) {
|
2023-05-07 13:48:49 +01:00
|
|
|
|
if (!TasmotaGlobal.i2c_enabled_2) {
|
|
|
|
|
I2c2Begin(sda, scl);
|
|
|
|
|
}
|
2021-04-21 10:01:40 +01:00
|
|
|
|
}
|
2024-07-12 15:36:04 +01:00
|
|
|
|
#endif // ESP32
|
2024-07-13 13:58:24 +01:00
|
|
|
|
if (I2cSetDevice(i2caddr, wire_n)) {
|
|
|
|
|
char display_name[16] = { 0 }; // Loosly related to dname in uDisplay.h
|
|
|
|
|
strcpy_P(display_name, PSTR("Display"));
|
|
|
|
|
cp = strstr(ddesc, ":H,");
|
|
|
|
|
if (cp) {
|
|
|
|
|
// :H,SH1106,128,64,1,I2C,3c,22,21,*
|
|
|
|
|
cp += 3;
|
|
|
|
|
char *lp = strchr(cp, ',');
|
|
|
|
|
if (lp) {
|
|
|
|
|
uint32_t len = lp - cp +1;
|
|
|
|
|
if (len < sizeof(display_name)) {
|
|
|
|
|
strlcpy(display_name, cp, len);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
I2cSetActiveFound(i2caddr, display_name, wire_n);
|
2021-04-21 10:01:40 +01:00
|
|
|
|
}
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-14 13:26:59 +01:00
|
|
|
|
cp = strstr(ddesc, "SPI,");
|
2021-04-11 11:32:02 +01:00
|
|
|
|
if (cp) {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 9 params nr,cs,sclk,mosi,dc,bl,reset,miso,mhz
|
|
|
|
|
// SPI,1,*,*,*,*,*,*,*,80
|
|
|
|
|
// SPI,*,*,*,*,*,*,*,*,40
|
2021-04-11 11:32:02 +01:00
|
|
|
|
cp += 4;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 1,*,*,*,*,*,*,*,80
|
|
|
|
|
uint32_t spi_type = 10; // SPI,* = Software SPI
|
|
|
|
|
if (isdigit(*cp) && (*cp != '0')) {
|
|
|
|
|
spi_type = *cp - '1'; // SPI,1 = 0, SPI,2 = 1, SPI,3 = 2
|
2021-04-21 16:11:19 +01:00
|
|
|
|
}
|
2024-07-13 13:58:24 +01:00
|
|
|
|
cp += 2;
|
|
|
|
|
// *,*,*,*,*,*,*,80
|
|
|
|
|
if (spi_type < 10) {
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SPI_CS, spi_type));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SPI_CLK, spi_type));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SPI_MOSI, spi_type));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SPI_DC, spi_type));
|
2021-12-08 15:32:02 +00:00
|
|
|
|
replacepin(&cp, Pin(GPIO_BACKLIGHT));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_OLED_RESET));
|
2024-07-13 13:58:24 +01:00
|
|
|
|
replacepin(&cp, Pin(GPIO_SPI_MISO, spi_type));
|
2021-04-11 11:32:02 +01:00
|
|
|
|
} else {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// Soft spi pins
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SSPI_CS));
|
2021-04-11 11:32:02 +01:00
|
|
|
|
replacepin(&cp, Pin(GPIO_SSPI_SCLK));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SSPI_MOSI));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SSPI_DC));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_BACKLIGHT));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_OLED_RESET));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_SSPI_MISO));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
cp = strstr(ddesc, ":UTI,");
|
2024-03-25 21:04:50 +00:00
|
|
|
|
if (cp) {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// :UTI,FT3267,I1,38,-1,-1
|
|
|
|
|
// :UTI,FT5206,I1,38,-1,-1
|
|
|
|
|
// :UTI,FT5206,I2,38,*,*
|
|
|
|
|
// :UTI,FT6236,I1,38,-1,-1
|
|
|
|
|
// :UTI,FT6336U,I2,38,-1,-1
|
|
|
|
|
// :UTI,FT6336U,I1,38,*,*
|
|
|
|
|
// :UTI,GT911,I1,5d,-1,-1
|
|
|
|
|
// :UTI,XPT2046,S1,*,*,*
|
|
|
|
|
// :UTI,SRES,R
|
|
|
|
|
cp += 5; // Skip ":UTI,"
|
|
|
|
|
// FT5206,I1,38,-1,-1
|
|
|
|
|
cp = strchr(cp, ','); // Skip name
|
|
|
|
|
// ,I1,38,-1,-1
|
2024-03-25 21:04:50 +00:00
|
|
|
|
cp++;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// I1,38,-1,-1
|
|
|
|
|
// S1,*,*,*
|
|
|
|
|
if (*cp == 'I') { // I = I2C
|
|
|
|
|
cp += 3; // Skip interface type and bus
|
|
|
|
|
// 38,-1,-1
|
|
|
|
|
cp = strchr(cp, ','); // Skip I2C address
|
|
|
|
|
// ,-1,-1
|
2024-03-25 21:04:50 +00:00
|
|
|
|
cp++;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// -1,-1
|
2024-03-25 21:04:50 +00:00
|
|
|
|
replacepin(&cp, Pin(GPIO_TS_RST));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_TS_IRQ));
|
2024-07-13 13:58:24 +01:00
|
|
|
|
} else if (*cp == 'S') { // S = SPI
|
|
|
|
|
cp += 3; // Skip interface type and bus
|
|
|
|
|
// *,*,*
|
2024-03-25 21:04:50 +00:00
|
|
|
|
replacepin(&cp, Pin(GPIO_TS_SPI_CS));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_TS_RST));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_TS_IRQ));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-18 13:06:04 +00:00
|
|
|
|
uint16_t xs, ys;
|
|
|
|
|
// we need screen size for gt911 touch controler
|
|
|
|
|
cp = strstr(ddesc, ":H,");
|
|
|
|
|
if (cp) {
|
|
|
|
|
cp += 3;
|
|
|
|
|
cp = strchr(cp, ',');
|
|
|
|
|
cp++;
|
|
|
|
|
xs = strtol(cp, &cp, 10);
|
|
|
|
|
cp++;
|
|
|
|
|
ys = strtol(cp, &cp, 10);
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-03 11:24:27 +01:00
|
|
|
|
#ifdef CONFIG_IDF_TARGET_ESP32S3
|
|
|
|
|
int8_t xp, xm, yp, ym;
|
2022-12-18 13:06:04 +00:00
|
|
|
|
|
2022-10-03 11:24:27 +01:00
|
|
|
|
cp = strstr(ddesc, "PAR,");
|
|
|
|
|
if (cp) {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// PAR,8,41,40,39,38,42,-1,8,9,10,11,12,13,14,15,20
|
|
|
|
|
// PAR,16,-1,37,36,35,48,45,47,21,14,13,12,11,10,9,3,8,16,15,7,6,5,4,20
|
2022-10-03 11:24:27 +01:00
|
|
|
|
cp += 4;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 8,41,40,39,38,42,-1,8,9,10,11,12,13,14,15,20
|
2022-10-03 11:24:27 +01:00
|
|
|
|
// 8 or 16 bus
|
|
|
|
|
uint8_t mode = strtol(cp, &cp, 10);
|
|
|
|
|
cp++;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 41,40,39,38,42,-1,8,9,10,11,12,13,14,15,20
|
2022-10-03 11:24:27 +01:00
|
|
|
|
replacepin(&cp, Pin(GPIO_DP_RES));
|
|
|
|
|
xm = replacepin(&cp, Pin(GPIO_DP_CS));
|
|
|
|
|
yp = replacepin(&cp, Pin(GPIO_DP_RS));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DP_WR));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DP_RD));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_BACKLIGHT));
|
|
|
|
|
|
|
|
|
|
ym = replacepin(&cp, Pin(GPIO_DPAR0));
|
|
|
|
|
xp = replacepin(&cp, Pin(GPIO_DPAR1));
|
|
|
|
|
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR2));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR3));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR4));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR5));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR6));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR7));
|
|
|
|
|
|
|
|
|
|
if (mode == 16) {
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR8));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR9));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR10));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR11));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR12));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR13));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR14));
|
|
|
|
|
replacepin(&cp, Pin(GPIO_DPAR15));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif // CONFIG_IDF_TARGET_ESP32S3
|
2021-04-11 11:32:02 +01:00
|
|
|
|
/*
|
|
|
|
|
File fp;
|
2021-04-14 13:26:59 +01:00
|
|
|
|
fp = ffsp->open("/dump.txt", "w");
|
|
|
|
|
fp.write((uint8_t*)ddesc, DISPDESC_SIZE);
|
2021-04-11 11:32:02 +01:00
|
|
|
|
fp.close();
|
2021-04-16 18:36:45 +01:00
|
|
|
|
*/
|
2021-04-14 13:26:59 +01:00
|
|
|
|
|
2021-04-16 18:36:45 +01:00
|
|
|
|
// init renderer
|
2021-04-21 10:01:40 +01:00
|
|
|
|
if (renderer) {
|
|
|
|
|
delete renderer;
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("DSP: reinit"));
|
2021-04-21 10:01:40 +01:00
|
|
|
|
}
|
2023-01-17 09:19:06 +00:00
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
udisp = new uDisplay(ddesc);
|
2021-04-14 13:26:59 +01:00
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// check for touch option TI1 or TI2
|
2023-12-18 10:03:18 +00:00
|
|
|
|
#if defined (USE_CST816S) || defined(USE_FT5206) || defined(USE_GT911)
|
2021-04-14 13:26:59 +01:00
|
|
|
|
cp = strstr(ddesc, ":TI");
|
|
|
|
|
if (cp) {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// :TI1,38,39,38
|
|
|
|
|
// :TI2,38,22,21
|
2021-04-19 16:01:33 +01:00
|
|
|
|
cp += 3;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 1,38,39,38
|
|
|
|
|
uint8_t wire_n = (*cp & 3) - 1;
|
2021-04-19 16:01:33 +01:00
|
|
|
|
cp += 2;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 38,39,38
|
2021-04-14 13:26:59 +01:00
|
|
|
|
uint8_t i2caddr = strtol(cp, &cp, 16);
|
2024-07-12 13:57:27 +01:00
|
|
|
|
cp++;
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// 39,38
|
2024-07-12 13:57:27 +01:00
|
|
|
|
int8_t scl = replacepin(&cp, Pin(GPIO_I2C_SCL, wire_n));
|
|
|
|
|
int8_t sda = replacepin(&cp, Pin(GPIO_I2C_SDA, wire_n));
|
|
|
|
|
int8_t irq = -1;
|
2022-12-18 13:06:04 +00:00
|
|
|
|
if (*(cp - 1) == ',') {
|
|
|
|
|
irq = strtol(cp, &cp, 10);
|
|
|
|
|
}
|
2024-07-12 13:57:27 +01:00
|
|
|
|
int8_t rst = -1;
|
2022-12-18 13:06:04 +00:00
|
|
|
|
if (*cp == ',') {
|
|
|
|
|
cp++;
|
|
|
|
|
rst = strtol(cp, &cp, 10);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-14 13:26:59 +01:00
|
|
|
|
if (wire_n == 0) {
|
2023-05-07 13:48:49 +01:00
|
|
|
|
if (!TasmotaGlobal.i2c_enabled) {
|
|
|
|
|
I2cBegin(sda, scl);
|
|
|
|
|
}
|
2021-04-14 13:26:59 +01:00
|
|
|
|
}
|
|
|
|
|
#ifdef ESP32
|
|
|
|
|
if (wire_n == 1) {
|
2023-05-07 13:48:49 +01:00
|
|
|
|
if (!TasmotaGlobal.i2c_enabled_2) {
|
|
|
|
|
I2c2Begin(sda, scl, 400000);
|
|
|
|
|
}
|
2021-04-14 13:26:59 +01:00
|
|
|
|
}
|
2024-07-13 13:58:24 +01:00
|
|
|
|
#endif // ESP32
|
2021-04-14 13:26:59 +01:00
|
|
|
|
if (I2cSetDevice(i2caddr, wire_n)) {
|
2022-12-18 13:06:04 +00:00
|
|
|
|
if (i2caddr == GT911_address) {
|
|
|
|
|
I2cSetActiveFound(i2caddr, "GT911", wire_n);
|
2023-12-18 10:03:18 +00:00
|
|
|
|
} else if (i2caddr == CST816S_address) {
|
|
|
|
|
I2cSetActiveFound(i2caddr, "CST816S", wire_n);
|
2022-12-18 13:06:04 +00:00
|
|
|
|
} else {
|
|
|
|
|
I2cSetActiveFound(i2caddr, "FT5206", wire_n);
|
|
|
|
|
}
|
2021-04-14 13:26:59 +01:00
|
|
|
|
}
|
2024-01-18 09:23:21 +00:00
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// Start digitizer
|
2022-12-18 13:06:04 +00:00
|
|
|
|
if (i2caddr == GT911_address) {
|
|
|
|
|
#ifdef USE_GT911
|
2024-07-13 13:58:24 +01:00
|
|
|
|
if (!wire_n) {
|
|
|
|
|
GT911_Touch_Init(&Wire, irq, rst, xs, ys);
|
|
|
|
|
}
|
|
|
|
|
#ifdef ESP32
|
|
|
|
|
else {
|
|
|
|
|
GT911_Touch_Init(&Wire1, irq, rst, xs, ys);
|
|
|
|
|
}
|
|
|
|
|
#endif // ESP32
|
|
|
|
|
#endif // USE_GT911
|
|
|
|
|
}
|
|
|
|
|
else if (i2caddr == CST816S_address) {
|
2023-12-18 10:03:18 +00:00
|
|
|
|
#ifdef USE_CST816S
|
|
|
|
|
CST816S_Touch_Init(wire_n, irq, rst);
|
2024-07-13 13:58:24 +01:00
|
|
|
|
#endif // USE_CST816S
|
|
|
|
|
}
|
|
|
|
|
else {
|
2022-12-18 13:06:04 +00:00
|
|
|
|
#ifdef USE_FT5206
|
2024-07-13 13:58:24 +01:00
|
|
|
|
if (!wire_n) {
|
|
|
|
|
FT5206_Touch_Init(Wire);
|
|
|
|
|
}
|
|
|
|
|
#ifdef ESP32
|
|
|
|
|
else {
|
|
|
|
|
FT5206_Touch_Init(Wire1);
|
|
|
|
|
}
|
|
|
|
|
#endif // ESP32
|
|
|
|
|
#endif // USE_FT5206
|
2022-12-18 13:06:04 +00:00
|
|
|
|
}
|
2021-04-14 13:26:59 +01:00
|
|
|
|
}
|
2022-12-18 13:06:04 +00:00
|
|
|
|
#endif // USE_FT5206 || GT911
|
2021-04-14 13:26:59 +01:00
|
|
|
|
|
|
|
|
|
#ifdef USE_XPT2046
|
|
|
|
|
cp = strstr(ddesc, ":TS,");
|
|
|
|
|
if (cp) {
|
2024-07-13 13:58:24 +01:00
|
|
|
|
// :TS,16
|
2023-04-23 09:19:31 +01:00
|
|
|
|
cp += 4;
|
2021-04-14 13:26:59 +01:00
|
|
|
|
uint8_t touch_cs = replacepin(&cp, Pin(GPIO_XPT2046_CS));
|
2023-04-23 09:19:31 +01:00
|
|
|
|
int8_t irqpin = -1;
|
|
|
|
|
if (*(cp - 1) == ',') {
|
|
|
|
|
irqpin = strtol(cp, &cp, 10);
|
|
|
|
|
}
|
|
|
|
|
uint8_t bus = 1;
|
|
|
|
|
if (*cp == ',') {
|
|
|
|
|
cp++;
|
|
|
|
|
bus = strtol(cp, &cp, 10);
|
|
|
|
|
if (bus < 1) bus = 1;
|
|
|
|
|
}
|
|
|
|
|
XPT2046_Touch_Init(touch_cs, irqpin, bus - 1);
|
2021-04-14 13:26:59 +01:00
|
|
|
|
}
|
2021-12-08 15:32:02 +00:00
|
|
|
|
#endif // USE_XPT2046
|
2021-04-14 13:26:59 +01:00
|
|
|
|
|
2022-10-03 11:24:27 +01:00
|
|
|
|
#ifdef CONFIG_IDF_TARGET_ESP32S3
|
|
|
|
|
#ifdef SIMPLE_RES_TOUCH
|
|
|
|
|
cp = strstr(ddesc, ":TR,");
|
|
|
|
|
if (cp) {
|
|
|
|
|
cp += 4;
|
|
|
|
|
Simple_ResTouch_Init(xp, xm, yp, ym);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-10-04 12:04:48 +01:00
|
|
|
|
uint8_t inirot = Settings->display_rotate;
|
|
|
|
|
|
|
|
|
|
cp = strstr(ddesc, ":r,");
|
|
|
|
|
if (cp) {
|
|
|
|
|
cp+=3;
|
|
|
|
|
inirot = strtol(cp, &cp, 10);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
// release desc buffer
|
|
|
|
|
if (fbuff) free(fbuff);
|
|
|
|
|
|
|
|
|
|
renderer = udisp->Init();
|
2021-04-21 10:01:40 +01:00
|
|
|
|
if (!renderer) return 0;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
2021-04-25 14:14:50 +01:00
|
|
|
|
fg_color = renderer->fgcol();
|
|
|
|
|
bg_color = renderer->bgcol();
|
|
|
|
|
color_type = renderer->color_type();
|
|
|
|
|
|
2021-10-04 12:04:48 +01:00
|
|
|
|
renderer->DisplayInit(DISPLAY_INIT_MODE, Settings->display_size, inirot, Settings->display_font);
|
2024-06-08 14:33:44 +01:00
|
|
|
|
renderer->clearDisplay();
|
2021-10-04 12:04:48 +01:00
|
|
|
|
|
|
|
|
|
Settings->display_width = renderer->width();
|
|
|
|
|
Settings->display_height = renderer->height();
|
2021-12-08 15:32:02 +00:00
|
|
|
|
|
2024-01-18 09:23:21 +00:00
|
|
|
|
#ifdef USE_UNIVERSAL_TOUCH
|
|
|
|
|
utouch_Touch_Init();
|
|
|
|
|
#endif
|
|
|
|
|
|
2023-08-18 10:34:01 +01:00
|
|
|
|
bool iniinv = Settings->display_options.invert;
|
2024-01-18 09:23:21 +00:00
|
|
|
|
/*
|
2023-08-18 10:34:01 +01:00
|
|
|
|
cp = strstr(ddesc, ":n,");
|
|
|
|
|
if (cp) {
|
|
|
|
|
cp+=3;
|
|
|
|
|
iniinv = strtol(cp, &cp, 10);
|
|
|
|
|
Settings->display_options.invert = iniinv;
|
2024-01-18 09:23:21 +00:00
|
|
|
|
}*/
|
2023-08-18 10:34:01 +01:00
|
|
|
|
renderer->invertDisplay(iniinv);
|
|
|
|
|
|
2023-02-05 11:52:21 +00:00
|
|
|
|
ApplyDisplayDimmer();
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
|
|
|
|
#ifdef SHOW_SPLASH
|
2024-06-08 14:33:44 +01:00
|
|
|
|
if (!Settings->flag5.display_no_splash) { // SetOption135 - (Display & LVGL) force disabling default splash screen
|
2022-03-05 21:56:24 +00:00
|
|
|
|
renderer->Splash();
|
|
|
|
|
}
|
2021-12-08 15:32:02 +00:00
|
|
|
|
#endif // SHOW_SPLASH
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
|
|
|
|
udisp_init_done = true;
|
2022-01-13 18:20:10 +00:00
|
|
|
|
AddLog(LOG_LEVEL_INFO, PSTR("DSP: Configured display '%s'"), renderer->devname());
|
2021-04-25 14:14:50 +01:00
|
|
|
|
|
2021-04-21 10:01:40 +01:00
|
|
|
|
return renderer;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
2021-04-21 10:01:40 +01:00
|
|
|
|
return 0;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*********************************************************************************************/
|
|
|
|
|
|
2024-03-25 21:04:50 +00:00
|
|
|
|
int8_t replacepin(char **cp, int16_t pin) {
|
2024-07-12 13:57:27 +01:00
|
|
|
|
// cp = 6,*,* and pin = 4 => lp = 6,*,* and cp = *,*
|
|
|
|
|
// cp = *,*,* and pin = 4 => lp = 4,*,* and cp = *,*
|
|
|
|
|
// cp = ,*,* and pin = 4 => lp = 4,*,* and cp = *,*
|
2024-07-13 13:58:24 +01:00
|
|
|
|
char *lp = *cp;
|
2024-07-12 13:57:27 +01:00
|
|
|
|
if ((*lp == '*') || (*lp == ',')) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
char val[8];
|
|
|
|
|
itoa(pin, val, 10);
|
|
|
|
|
uint16_t slen = strlen(val);
|
|
|
|
|
//AddLog(LOG_LEVEL_INFO, PSTR("replace pin: %d"), pin);
|
2024-07-12 13:57:27 +01:00
|
|
|
|
uint32_t idx = 0;
|
|
|
|
|
if (*lp == '*') { idx++; }
|
|
|
|
|
memmove(lp + slen, lp + idx, strlen(lp));
|
2021-04-11 11:32:02 +01:00
|
|
|
|
memmove(lp, val, slen);
|
|
|
|
|
}
|
2024-07-13 13:58:24 +01:00
|
|
|
|
int8_t res = strtol(lp, 0, 10);
|
2021-04-11 11:32:02 +01:00
|
|
|
|
char *np = strchr(lp, ',');
|
|
|
|
|
if (np) {
|
|
|
|
|
*cp = np + 1;
|
|
|
|
|
}
|
2021-04-14 13:26:59 +01:00
|
|
|
|
return res;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef USE_DISPLAY_MODES1TO5
|
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
void UDISP_PrintLog(void) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
disp_refresh--;
|
|
|
|
|
if (!disp_refresh) {
|
2021-06-11 17:14:12 +01:00
|
|
|
|
disp_refresh = Settings->display_refresh;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
if (!disp_screen_buffer_cols) { DisplayAllocScreenBuffer(); }
|
|
|
|
|
|
|
|
|
|
char* txt = DisplayLogBuffer('\370');
|
2024-06-11 10:07:30 +01:00
|
|
|
|
if (txt != nullptr) {
|
2021-06-11 17:14:12 +01:00
|
|
|
|
uint8_t last_row = Settings->display_rows -1;
|
2021-04-11 11:32:02 +01:00
|
|
|
|
|
2024-06-08 16:22:19 +01:00
|
|
|
|
// renderer->clearDisplay();
|
2021-06-11 17:14:12 +01:00
|
|
|
|
renderer->setTextSize(Settings->display_size);
|
2021-04-11 11:32:02 +01:00
|
|
|
|
renderer->setCursor(0,0);
|
|
|
|
|
for (byte i = 0; i < last_row; i++) {
|
|
|
|
|
strlcpy(disp_screen_buffer[i], disp_screen_buffer[i +1], disp_screen_buffer_cols);
|
|
|
|
|
renderer->println(disp_screen_buffer[i]);
|
|
|
|
|
}
|
|
|
|
|
strlcpy(disp_screen_buffer[last_row], txt, disp_screen_buffer_cols);
|
|
|
|
|
DisplayFillScreen(last_row);
|
|
|
|
|
renderer->println(disp_screen_buffer[last_row]);
|
|
|
|
|
renderer->Updateframe();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
void UDISP_Time(void) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
char line[12];
|
|
|
|
|
|
2024-06-08 16:22:19 +01:00
|
|
|
|
// renderer->clearDisplay();
|
2021-06-11 17:14:12 +01:00
|
|
|
|
renderer->setTextSize(Settings->display_size);
|
|
|
|
|
renderer->setTextFont(Settings->display_font);
|
2021-04-11 11:32:02 +01:00
|
|
|
|
renderer->setCursor(0, 0);
|
|
|
|
|
snprintf_P(line, sizeof(line), PSTR(" %02d" D_HOUR_MINUTE_SEPARATOR "%02d" D_MINUTE_SECOND_SEPARATOR "%02d"), RtcTime.hour, RtcTime.minute, RtcTime.second); // [ 12:34:56 ]
|
|
|
|
|
renderer->println(line);
|
|
|
|
|
renderer->println();
|
|
|
|
|
snprintf_P(line, sizeof(line), PSTR("%02d" D_MONTH_DAY_SEPARATOR "%02d" D_YEAR_MONTH_SEPARATOR "%04d"), RtcTime.day_of_month, RtcTime.month, RtcTime.year); // [01-02-2018]
|
|
|
|
|
renderer->println(line);
|
|
|
|
|
renderer->Updateframe();
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 13:58:24 +01:00
|
|
|
|
void UDISP_Refresh(void) { // Every second
|
2021-04-11 11:32:02 +01:00
|
|
|
|
if (!renderer) return;
|
2021-06-11 17:14:12 +01:00
|
|
|
|
if (Settings->display_mode) { // Mode 0 is User text
|
|
|
|
|
switch (Settings->display_mode) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
case 1: // Time
|
|
|
|
|
UDISP_Time();
|
|
|
|
|
break;
|
|
|
|
|
case 2: // Local
|
|
|
|
|
case 3: // Local
|
|
|
|
|
case 4: // Mqtt
|
|
|
|
|
case 5: // Mqtt
|
|
|
|
|
UDISP_PrintLog();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif // USE_DISPLAY_MODES1TO5
|
|
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
|
* Interface
|
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
2022-11-11 09:44:56 +00:00
|
|
|
|
bool Xdsp17(uint32_t function) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
bool result = false;
|
|
|
|
|
|
|
|
|
|
if (FUNC_DISPLAY_INIT_DRIVER == function) {
|
2021-11-13 19:45:27 +00:00
|
|
|
|
Init_uDisplay(nullptr);
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
2021-06-11 17:14:12 +01:00
|
|
|
|
else if (udisp_init_done && (XDSP_17 == Settings->display_model)) {
|
2021-04-11 11:32:02 +01:00
|
|
|
|
switch (function) {
|
|
|
|
|
case FUNC_DISPLAY_MODEL:
|
|
|
|
|
result = true;
|
|
|
|
|
break;
|
2021-04-14 13:26:59 +01:00
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
#ifdef USE_DISPLAY_MODES1TO5
|
|
|
|
|
case FUNC_DISPLAY_EVERY_SECOND:
|
|
|
|
|
UDISP_Refresh();
|
|
|
|
|
break;
|
|
|
|
|
#endif // USE_DISPLAY_MODES1TO5
|
2021-04-14 13:26:59 +01:00
|
|
|
|
|
2021-04-11 11:32:02 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif // USE_UNIVERSAL_DISPLAY
|
|
|
|
|
#endif // USE_DISPLAY
|