mirror of https://github.com/arendst/Tasmota.git
universal display driver
This commit is contained in:
parent
16e397dbd7
commit
0e7e956a7b
|
@ -65,7 +65,9 @@ keywords if then else endif, or, and are better readable for beginners (others m
|
|||
#define SCRIPT_MAXPERM (PMEM_SIZE)-4/sizeof(float)
|
||||
#define MAX_SCRIPT_SIZE MAX_RULE_SIZE*MAX_RULE_SETS
|
||||
|
||||
#ifndef MAX_SARRAY_NUM
|
||||
#define MAX_SARRAY_NUM 32
|
||||
#endif
|
||||
|
||||
uint32_t EncodeLightId(uint8_t relay_id);
|
||||
uint32_t DecodeLightId(uint32_t hue_id);
|
||||
|
@ -1413,38 +1415,42 @@ uint32_t match_vars(char *dvnam, float **fp, char **sp, uint32_t *ind) {
|
|||
}
|
||||
#endif //USE_SCRIPT_GLOBVARS
|
||||
|
||||
#ifndef SCRIPT_IS_STRING_MAXSIZE
|
||||
#define SCRIPT_IS_STRING_MAXSIZE 256
|
||||
#endif
|
||||
|
||||
char *isargs(char *lp, uint32_t isind) {
|
||||
float fvar;
|
||||
lp = GetNumericArgument(lp, OPER_EQU, &fvar, 0);
|
||||
SCRIPT_SKIP_SPACES
|
||||
if (*lp!='"') {
|
||||
if (*lp != '"') {
|
||||
return lp;
|
||||
}
|
||||
lp++;
|
||||
|
||||
if (glob_script_mem.si_num[isind]>0 && glob_script_mem.last_index_string[isind]) {
|
||||
if (glob_script_mem.si_num[isind] > 0 && glob_script_mem.last_index_string[isind]) {
|
||||
free(glob_script_mem.last_index_string[isind]);
|
||||
}
|
||||
char *sstart = lp;
|
||||
uint8_t slen = 0;
|
||||
for (uint32_t cnt = 0; cnt<256; cnt++) {
|
||||
if (*lp=='\n' || *lp=='"' || *lp==0) {
|
||||
for (uint32_t cnt = 0; cnt < SCRIPT_IS_STRING_MAXSIZE; cnt++) {
|
||||
if (*lp == '\n' || *lp == '"' || *lp == 0) {
|
||||
lp++;
|
||||
if (cnt>0 && !slen) {
|
||||
if (cnt > 0 && !slen) {
|
||||
slen++;
|
||||
}
|
||||
glob_script_mem.siro_num[isind] = slen;
|
||||
break;
|
||||
}
|
||||
if (*lp=='|') {
|
||||
if (*lp == '|') {
|
||||
slen++;
|
||||
}
|
||||
lp++;
|
||||
}
|
||||
|
||||
glob_script_mem.si_num[isind] = fvar;
|
||||
if (glob_script_mem.si_num[isind]>0) {
|
||||
if (glob_script_mem.si_num[isind]>MAX_SARRAY_NUM) {
|
||||
if (glob_script_mem.si_num[isind] > 0) {
|
||||
if (glob_script_mem.si_num[isind] > MAX_SARRAY_NUM) {
|
||||
glob_script_mem.si_num[isind] = MAX_SARRAY_NUM;
|
||||
}
|
||||
|
||||
|
@ -1468,17 +1474,17 @@ float fvar;
|
|||
char str[SCRIPT_MAXSSIZE];
|
||||
str[0] = 0;
|
||||
uint8_t index = fvar;
|
||||
if (index<1) index = 1;
|
||||
if (index < 1) index = 1;
|
||||
index--;
|
||||
if (gv) gv->strind = index;
|
||||
glob_script_mem.sind_num = isind;
|
||||
if (glob_script_mem.last_index_string[isind]) {
|
||||
if (!glob_script_mem.si_num[isind]) {
|
||||
if (index<=glob_script_mem.siro_num[isind]) {
|
||||
if (index <= glob_script_mem.siro_num[isind]) {
|
||||
GetTextIndexed(str, sizeof(str), index , glob_script_mem.last_index_string[isind]);
|
||||
}
|
||||
} else {
|
||||
if (index>glob_script_mem.si_num[isind]) {
|
||||
if (index > glob_script_mem.si_num[isind]) {
|
||||
index = glob_script_mem.si_num[isind];
|
||||
}
|
||||
strlcpy(str,glob_script_mem.last_index_string[isind] + (index * glob_script_mem.max_ssize), glob_script_mem.max_ssize);
|
||||
|
@ -1866,6 +1872,15 @@ chknext:
|
|||
}
|
||||
#endif //USE_SCRIPT_TASK
|
||||
#endif //ESP32
|
||||
#ifdef USE_ANGLE_FUNC
|
||||
if (!strncmp(vname, "cos(", 4)) {
|
||||
lp = GetNumericArgument(lp + 4, OPER_EQU, &fvar, gv);
|
||||
fvar = cosf(fvar);
|
||||
lp++;
|
||||
len = 0;
|
||||
goto exit;
|
||||
}
|
||||
#endif
|
||||
break;
|
||||
case 'd':
|
||||
if (!strncmp(vname, "day", 3)) {
|
||||
|
@ -2196,6 +2211,22 @@ chknext:
|
|||
len = 0;
|
||||
goto exit;
|
||||
}
|
||||
if (!strncmp(vname, "fmt(", 4)) {
|
||||
lp = GetNumericArgument(lp + 4, OPER_EQU, &fvar, gv);
|
||||
if (!fvar) {
|
||||
#ifdef ESP8266
|
||||
LittleFS.format();
|
||||
#endif
|
||||
#ifdef ESP32
|
||||
LITTLEFS.format();
|
||||
#endif
|
||||
} else {
|
||||
//SD.format();
|
||||
}
|
||||
lp++;
|
||||
len = 0;
|
||||
goto exit;
|
||||
}
|
||||
if (!strncmp(vname, "frd(", 4)) {
|
||||
char str[glob_script_mem.max_ssize + 1];
|
||||
lp = GetStringArgument(lp + 4, OPER_EQU, str, 0);
|
||||
|
|
|
@ -80,6 +80,7 @@ const uint8_t DISPLAY_LOG_ROWS = 32; // Number of lines in display log
|
|||
#define D_CMND_DISP_CLOCK "Clock"
|
||||
#define D_CMND_DISP_TEXTNC "TextNC" // NC - "No Clear"
|
||||
#define D_CMND_DISP_SCROLLTEXT "ScrollText"
|
||||
#define D_CMND_DISP_REINIT "reinit"
|
||||
|
||||
enum XdspFunctions { FUNC_DISPLAY_INIT_DRIVER, FUNC_DISPLAY_INIT, FUNC_DISPLAY_EVERY_50_MSECOND, FUNC_DISPLAY_EVERY_SECOND,
|
||||
FUNC_DISPLAY_MODEL, FUNC_DISPLAY_MODE, FUNC_DISPLAY_POWER,
|
||||
|
@ -108,7 +109,7 @@ const char kDisplayCommands[] PROGMEM = D_PRFX_DISPLAY "|" // Prefix
|
|||
#endif
|
||||
D_CMND_DISP_CLEAR "|" D_CMND_DISP_NUMBER "|" D_CMND_DISP_FLOAT "|" D_CMND_DISP_NUMBERNC "|" D_CMND_DISP_FLOATNC "|"
|
||||
D_CMND_DISP_RAW "|" D_CMND_DISP_LEVEL "|" D_CMND_DISP_SEVENSEG_TEXT "|" D_CMND_DISP_SEVENSEG_TEXTNC "|"
|
||||
D_CMND_DISP_SCROLLDELAY "|" D_CMND_DISP_CLOCK "|" D_CMND_DISP_TEXTNC "|" D_CMND_DISP_SCROLLTEXT
|
||||
D_CMND_DISP_SCROLLDELAY "|" D_CMND_DISP_CLOCK "|" D_CMND_DISP_TEXTNC "|" D_CMND_DISP_SCROLLTEXT "|" D_CMND_DISP_REINIT
|
||||
;
|
||||
|
||||
void (* const DisplayCommand[])(void) PROGMEM = {
|
||||
|
@ -120,7 +121,7 @@ void (* const DisplayCommand[])(void) PROGMEM = {
|
|||
#endif
|
||||
&CmndDisplayClear, &CmndDisplayNumber, &CmndDisplayFloat, &CmndDisplayNumberNC, &CmndDisplayFloatNC,
|
||||
&CmndDisplayRaw, &CmndDisplayLevel, &CmndDisplaySevensegText, &CmndDisplaySevensegTextNC,
|
||||
&CmndDisplayScrollDelay, &CmndDisplayClock, &CmndDisplayTextNC, &CmndDisplayScrollText
|
||||
&CmndDisplayScrollDelay, &CmndDisplayClock, &CmndDisplayTextNC, &CmndDisplayScrollText,&DisplayReInitDriver
|
||||
};
|
||||
|
||||
#ifdef USE_GRAPH
|
||||
|
@ -989,7 +990,7 @@ void Display_Text_From_File(const char *file) {
|
|||
File fp;
|
||||
if (!ufsp) return;
|
||||
fp = ufsp->open(file, FS_FILE_READ);
|
||||
if (fp >= 0) {
|
||||
if (fp > 0) {
|
||||
char *savptr = XdrvMailbox.data;
|
||||
char linebuff[128];
|
||||
while (fp.available()) {
|
||||
|
@ -1021,7 +1022,7 @@ void Display_Text_From_File(const char *file) {
|
|||
fp.close();
|
||||
}
|
||||
}
|
||||
#endif
|
||||
#endif // USE_UFILESYS
|
||||
|
||||
|
||||
#ifdef USE_DT_VARS
|
||||
|
@ -2001,6 +2002,11 @@ void CmndDisplayScrollText(void) {
|
|||
if(result) ResponseCmndChar(XdrvMailbox.data);
|
||||
}
|
||||
|
||||
void DisplayReInitDriver(void) {
|
||||
XdspCall(FUNC_DISPLAY_INIT_DRIVER);
|
||||
ResponseCmndDone();
|
||||
}
|
||||
|
||||
/*********************************************************************************************\
|
||||
* Optional drivers
|
||||
\*********************************************************************************************/
|
||||
|
|
|
@ -0,0 +1,332 @@
|
|||
/*
|
||||
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/>.
|
||||
*/
|
||||
|
||||
|
||||
#ifdef USE_DISPLAY
|
||||
#ifdef USE_UNIVERSAL_DISPLAY
|
||||
|
||||
#define XDSP_17 17
|
||||
|
||||
#include <uDisplay.h>
|
||||
|
||||
uDisplay *udisp;
|
||||
bool udisp_init_done = false;
|
||||
extern uint8_t color_type;
|
||||
extern uint16_t fg_color;
|
||||
extern uint16_t bg_color;
|
||||
|
||||
#ifdef USE_UFILESYS
|
||||
extern FS *ufsp;
|
||||
#endif
|
||||
|
||||
#define DISPDESC_SIZE 1000
|
||||
|
||||
#define DSP_ROM_DESC
|
||||
|
||||
/*********************************************************************************************/
|
||||
#ifdef DSP_ROM_DESC
|
||||
/* sample descriptor */
|
||||
const char DSP_SAMPLE_DESC[] PROGMEM =
|
||||
// name,xs,ys,bpp,interface, (HEX) address, scl,sda,reset
|
||||
// '*' means take pin number from tasmota
|
||||
":H\n"
|
||||
"SH1106,128,64,1,I2C,3c,*,*,*\n"
|
||||
// splash settings, font, size, fgcol, bgcol, x,y
|
||||
":S\n"
|
||||
"0,1,1,0,40,20\n"
|
||||
// init register settings, must be in HEX
|
||||
":I\n"
|
||||
"AE\n"
|
||||
"D5,80\n"
|
||||
"A8,3f\n"
|
||||
"D3,00\n"
|
||||
"40\n"
|
||||
"8D,14\n"
|
||||
"20,00\n"
|
||||
"A1\n"
|
||||
"C8\n"
|
||||
"DA,12\n"
|
||||
"81,CF\n"
|
||||
"D9F1\n"
|
||||
"DB,40\n"
|
||||
"A4\n"
|
||||
"A6\n"
|
||||
"AF\n"
|
||||
// switch display off
|
||||
":o\n"
|
||||
"AE\n"
|
||||
// switch display on
|
||||
":O\n"
|
||||
"AF\n"
|
||||
"#\n";
|
||||
|
||||
#endif // DSP_ROM_DESC
|
||||
/*********************************************************************************************/
|
||||
|
||||
void Init_uDisp(void) {
|
||||
char *ddesc = 0;
|
||||
char *fbuff;
|
||||
|
||||
if (1) {
|
||||
Settings.display_model = XDSP_17;
|
||||
|
||||
fg_color = 1;
|
||||
bg_color = 0;
|
||||
color_type = COLOR_BW;
|
||||
|
||||
fbuff = (char*)calloc(DISPDESC_SIZE, 1);
|
||||
if (!fbuff) return;
|
||||
|
||||
#ifdef USE_UFILESYS
|
||||
if (ufsp && !TasmotaGlobal.no_autoexec) {
|
||||
File fp;
|
||||
fp = ufsp->open("/dispdesc.txt", "r");
|
||||
if (fp > 0) {
|
||||
uint32_t size = fp.size();
|
||||
fp.read((uint8_t*)fbuff, size);
|
||||
fp.close();
|
||||
ddesc = fbuff;
|
||||
AddLog(LOG_LEVEL_INFO, PSTR("DSP: File descriptor used"));
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
#ifdef USE_SCRIPT
|
||||
if (bitRead(Settings.rule_enabled, 0) && !ddesc) {
|
||||
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;
|
||||
AddLog(LOG_LEVEL_INFO, PSTR("DSP: Script descriptor used"));
|
||||
}
|
||||
}
|
||||
#endif // USE_SCRIPT
|
||||
|
||||
|
||||
#ifdef DSP_ROM_DESC
|
||||
if (!ddesc) {
|
||||
memcpy_P(fbuff, DSP_SAMPLE_DESC, sizeof(DSP_SAMPLE_DESC));
|
||||
ddesc = fbuff;
|
||||
AddLog(LOG_LEVEL_INFO, PSTR("DSP: Flash descriptor used"));
|
||||
}
|
||||
#endif // DSP_ROM_DESC
|
||||
|
||||
if (!ddesc) {
|
||||
AddLog(LOG_LEVEL_INFO, PSTR("DSP: No valid descriptor found"));
|
||||
if (fbuff) free(fbuff);
|
||||
return;
|
||||
}
|
||||
// now replace tasmota vars before passing to driver
|
||||
char *cp = strstr(ddesc, "I2C");
|
||||
if (cp) {
|
||||
cp += 4;
|
||||
//,3c,22,21,-1
|
||||
// i2c addr
|
||||
//if (*cp == '*') {
|
||||
// Settings.display_address
|
||||
//}
|
||||
uint8_t i2caddr = strtol(cp, 0, 16);
|
||||
if (I2cSetDevice(i2caddr)) {
|
||||
I2cSetActiveFound(i2caddr, "DSP-I2C");
|
||||
}
|
||||
cp+=3;
|
||||
//replacepin(&cp, Settings.display_address);
|
||||
replacepin(&cp, Pin(GPIO_I2C_SCL));
|
||||
replacepin(&cp, Pin(GPIO_I2C_SDA));
|
||||
replacepin(&cp, Pin(GPIO_OLED_RESET));
|
||||
}
|
||||
|
||||
cp = strstr(ddesc, "SPI");
|
||||
if (cp) {
|
||||
cp += 4;
|
||||
//; 7 params nr,cs,sclk,mosi,dc,bl,reset,miso
|
||||
//SPI,*,*,*,*,*,*,*
|
||||
if (*cp == '1') {
|
||||
cp+=2;
|
||||
replacepin(&cp, Pin(GPIO_SPI_CS));
|
||||
replacepin(&cp, Pin(GPIO_SPI_CLK));
|
||||
replacepin(&cp, Pin(GPIO_SPI_MOSI));
|
||||
replacepin(&cp, Pin(GPIO_SPI_DC));
|
||||
replacepin(&cp, Pin(GPIO_BACKLIGHT));
|
||||
replacepin(&cp, Pin(GPIO_OLED_RESET));
|
||||
replacepin(&cp, Pin(GPIO_SPI_MISO));
|
||||
} else {
|
||||
// soft spi pins
|
||||
cp+=2;
|
||||
replacepin(&cp, Pin(GPIO_SSPI_CS));
|
||||
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));
|
||||
}
|
||||
}
|
||||
|
||||
// init renderer
|
||||
if (udisp) delete udisp;
|
||||
udisp = new uDisplay(ddesc);
|
||||
|
||||
/*
|
||||
File fp;
|
||||
fp = ufsp->open("/dump.txt", "w");
|
||||
fp.write(ddesc, DISPDESC_SIZE);
|
||||
fp.close();
|
||||
*/
|
||||
// release desc buffer
|
||||
if (fbuff) free(fbuff);
|
||||
|
||||
renderer = udisp->Init();
|
||||
if (!renderer) return;
|
||||
|
||||
Settings.display_width = renderer->width();
|
||||
Settings.display_height = renderer->height();
|
||||
fg_color = udisp->fgcol();
|
||||
bg_color = udisp->bgcol();
|
||||
|
||||
renderer->DisplayInit(DISPLAY_INIT_MODE, Settings.display_size, Settings.display_rotate, Settings.display_font);
|
||||
|
||||
|
||||
#ifdef SHOW_SPLASH
|
||||
udisp->Splash();
|
||||
#endif
|
||||
|
||||
udisp_init_done = true;
|
||||
AddLog(LOG_LEVEL_INFO, PSTR("DSP: %s!"), udisp->devname());
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/*********************************************************************************************/
|
||||
|
||||
void replacepin(char **cp, uint16_t pin) {
|
||||
char *lp = *cp;
|
||||
if (*lp == ',') lp++;
|
||||
if (*lp == '*') {
|
||||
char val[8];
|
||||
itoa(pin, val, 10);
|
||||
uint16_t slen = strlen(val);
|
||||
//AddLog(LOG_LEVEL_INFO, PSTR("replace pin: %d"), pin);
|
||||
memmove(lp + slen, lp + 1, strlen(lp));
|
||||
memmove(lp, val, slen);
|
||||
}
|
||||
char *np = strchr(lp, ',');
|
||||
if (np) {
|
||||
*cp = np + 1;
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef USE_DISPLAY_MODES1TO5
|
||||
|
||||
void UDISP_PrintLog(void)
|
||||
{
|
||||
disp_refresh--;
|
||||
if (!disp_refresh) {
|
||||
disp_refresh = Settings.display_refresh;
|
||||
if (!disp_screen_buffer_cols) { DisplayAllocScreenBuffer(); }
|
||||
|
||||
char* txt = DisplayLogBuffer('\370');
|
||||
if (txt != NULL) {
|
||||
uint8_t last_row = Settings.display_rows -1;
|
||||
|
||||
renderer->clearDisplay();
|
||||
renderer->setTextSize(Settings.display_size);
|
||||
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);
|
||||
|
||||
AddLog(LOG_LEVEL_DEBUG, PSTR(D_LOG_DEBUG "[%s]"), disp_screen_buffer[last_row]);
|
||||
|
||||
renderer->println(disp_screen_buffer[last_row]);
|
||||
renderer->Updateframe();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void UDISP_Time(void)
|
||||
{
|
||||
char line[12];
|
||||
|
||||
renderer->clearDisplay();
|
||||
renderer->setTextSize(Settings.display_size);
|
||||
renderer->setTextFont(Settings.display_font);
|
||||
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();
|
||||
}
|
||||
|
||||
void UDISP_Refresh(void) // Every second
|
||||
{
|
||||
if (!renderer) return;
|
||||
if (Settings.display_mode) { // Mode 0 is User text
|
||||
switch (Settings.display_mode) {
|
||||
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
|
||||
\*********************************************************************************************/
|
||||
|
||||
bool Xdsp17(uint8_t function)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
if (FUNC_DISPLAY_INIT_DRIVER == function) {
|
||||
Init_uDisp();
|
||||
}
|
||||
else if (udisp_init_done && (XDSP_17 == Settings.display_model)) {
|
||||
switch (function) {
|
||||
case FUNC_DISPLAY_MODEL:
|
||||
result = true;
|
||||
break;
|
||||
#ifdef USE_DISPLAY_MODES1TO5
|
||||
case FUNC_DISPLAY_EVERY_SECOND:
|
||||
UDISP_Refresh();
|
||||
break;
|
||||
#endif // USE_DISPLAY_MODES1TO5
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
#endif // USE_UNIVERSAL_DISPLAY
|
||||
#endif // USE_DISPLAY
|
Loading…
Reference in New Issue