2018-08-28 17:13:14 +01:00
|
|
|
/*
|
2019-10-27 10:13:24 +00:00
|
|
|
xdsp_02_ssd1306.ino - Display Oled SSD1306 support for Tasmota
|
2018-08-28 17:13:14 +01:00
|
|
|
|
2019-08-19 15:49:20 +01:00
|
|
|
Copyright (C) 2019 Theo Arends and Adafruit
|
2018-08-28 17:13:14 +01:00
|
|
|
|
|
|
|
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_I2C
|
|
|
|
#ifdef USE_DISPLAY
|
|
|
|
#ifdef USE_DISPLAY_SSD1306
|
|
|
|
|
2019-08-19 17:17:44 +01:00
|
|
|
#define XDSP_02 2
|
2019-11-03 16:54:39 +00:00
|
|
|
#define XI2C_04 4 // See I2CDEVICES.md
|
2019-08-19 17:17:44 +01:00
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
#define OLED_RESET 4
|
|
|
|
|
|
|
|
#define SPRINT(A) char str[32];sprintf(str,"val: %d ",A);Serial.println((char*)str);
|
|
|
|
|
2018-08-28 17:13:14 +01:00
|
|
|
#define OLED_ADDRESS1 0x3C // Oled 128x32 I2C address
|
|
|
|
#define OLED_ADDRESS2 0x3D // Oled 128x64 I2C address
|
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
#define OLED_BUFFER_COLS 40 // Max number of columns in display shadow buffer
|
|
|
|
#define OLED_BUFFER_ROWS 16 // Max number of lines in display shadow buffer
|
|
|
|
|
2018-08-28 17:13:14 +01:00
|
|
|
#define OLED_FONT_WIDTH 6
|
|
|
|
#define OLED_FONT_HEIGTH 8
|
|
|
|
|
|
|
|
#include <Wire.h>
|
2019-08-19 12:21:54 +01:00
|
|
|
#include <renderer.h>
|
2018-08-28 17:13:14 +01:00
|
|
|
#include <Adafruit_SSD1306.h>
|
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
Adafruit_SSD1306 *oled1306;
|
2018-08-28 17:13:14 +01:00
|
|
|
|
2019-08-19 17:17:44 +01:00
|
|
|
extern uint8_t *buffer;
|
2018-08-28 17:13:14 +01:00
|
|
|
|
2019-08-19 17:17:44 +01:00
|
|
|
/*********************************************************************************************/
|
2018-08-28 17:13:14 +01:00
|
|
|
|
2019-11-20 19:53:12 +00:00
|
|
|
void SSD1306InitDriver(void)
|
2018-08-28 17:13:14 +01:00
|
|
|
{
|
|
|
|
if (!Settings.display_model) {
|
2019-11-06 16:48:38 +00:00
|
|
|
if (I2cSetDevice(OLED_ADDRESS1)) {
|
2018-08-28 17:13:14 +01:00
|
|
|
Settings.display_address[0] = OLED_ADDRESS1;
|
|
|
|
Settings.display_model = XDSP_02;
|
|
|
|
}
|
2019-11-06 16:48:38 +00:00
|
|
|
else if (I2cSetDevice(OLED_ADDRESS2)) {
|
2018-08-28 17:13:14 +01:00
|
|
|
Settings.display_address[0] = OLED_ADDRESS2;
|
|
|
|
Settings.display_model = XDSP_02;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (XDSP_02 == Settings.display_model) {
|
2019-11-09 17:34:22 +00:00
|
|
|
I2cSetActiveFound(Settings.display_address[0], "SSD1306");
|
2019-08-19 12:21:54 +01:00
|
|
|
|
2019-10-30 09:14:16 +00:00
|
|
|
if ((Settings.display_width != 64) && (Settings.display_width != 96) && (Settings.display_width != 128)) {
|
2019-07-23 13:05:42 +01:00
|
|
|
Settings.display_width = 128;
|
|
|
|
}
|
2019-10-30 09:14:16 +00:00
|
|
|
if ((Settings.display_height != 16) && (Settings.display_height != 32) && (Settings.display_height != 48) && (Settings.display_height != 64)) {
|
2019-07-23 13:05:42 +01:00
|
|
|
Settings.display_height = 64;
|
|
|
|
}
|
2019-08-19 12:21:54 +01:00
|
|
|
|
|
|
|
uint8_t reset_pin = -1;
|
2019-07-23 13:05:42 +01:00
|
|
|
if (pin[GPIO_OLED_RESET] < 99) {
|
|
|
|
reset_pin = pin[GPIO_OLED_RESET];
|
|
|
|
}
|
2018-08-28 17:13:14 +01:00
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
// allocate screen buffer
|
2019-08-19 17:17:44 +01:00
|
|
|
if (buffer) { free(buffer); }
|
|
|
|
buffer = (unsigned char*)calloc((Settings.display_width * Settings.display_height) / 8,1);
|
|
|
|
if (!buffer) { return; }
|
2019-08-19 12:21:54 +01:00
|
|
|
|
|
|
|
// init renderer
|
2019-08-19 17:17:44 +01:00
|
|
|
// oled1306 = new Adafruit_SSD1306(SSD1306_LCDWIDTH,SSD1306_LCDHEIGHT);
|
2019-08-19 12:21:54 +01:00
|
|
|
oled1306 = new Adafruit_SSD1306(Settings.display_width, Settings.display_height, &Wire, reset_pin);
|
2019-08-19 17:17:44 +01:00
|
|
|
oled1306->begin(SSD1306_SWITCHCAPVCC, Settings.display_address[0], 0);
|
|
|
|
renderer = oled1306;
|
|
|
|
renderer->DisplayInit(DISPLAY_INIT_MODE, Settings.display_size, Settings.display_rotate, Settings.display_font);
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->setTextColor(1,0);
|
|
|
|
|
|
|
|
#ifdef SHOW_SPLASH
|
|
|
|
renderer->setTextFont(0);
|
|
|
|
renderer->setTextSize(2);
|
|
|
|
renderer->setCursor(20,20);
|
|
|
|
renderer->println(F("SSD1306"));
|
|
|
|
renderer->Updateframe();
|
|
|
|
renderer->DisplayOnff(1);
|
|
|
|
#endif
|
2018-08-28 17:13:14 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************************************/
|
|
|
|
#ifdef USE_DISPLAY_MODES1TO5
|
|
|
|
|
2018-11-14 13:32:09 +00:00
|
|
|
void Ssd1306PrintLog(void)
|
2018-08-28 17:13:14 +01:00
|
|
|
{
|
|
|
|
disp_refresh--;
|
|
|
|
if (!disp_refresh) {
|
|
|
|
disp_refresh = Settings.display_refresh;
|
2018-08-31 11:17:09 +01:00
|
|
|
if (!disp_screen_buffer_cols) { DisplayAllocScreenBuffer(); }
|
|
|
|
|
|
|
|
char* txt = DisplayLogBuffer('\370');
|
2019-08-19 12:21:54 +01:00
|
|
|
if (txt != NULL) {
|
2018-08-31 11:17:09 +01:00
|
|
|
uint8_t last_row = Settings.display_rows -1;
|
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->clearDisplay();
|
|
|
|
renderer->setTextSize(Settings.display_size);
|
|
|
|
renderer->setCursor(0,0);
|
|
|
|
for (byte i = 0; i < last_row; i++) {
|
2018-08-31 11:17:09 +01:00
|
|
|
strlcpy(disp_screen_buffer[i], disp_screen_buffer[i +1], disp_screen_buffer_cols);
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->println(disp_screen_buffer[i]);
|
2018-08-31 11:17:09 +01:00
|
|
|
}
|
|
|
|
strlcpy(disp_screen_buffer[last_row], txt, disp_screen_buffer_cols);
|
|
|
|
DisplayFillScreen(last_row);
|
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
snprintf_P(log_data, sizeof(log_data), PSTR(D_LOG_DEBUG "[%s]"), disp_screen_buffer[last_row]);
|
|
|
|
AddLog(LOG_LEVEL_DEBUG);
|
2018-08-31 11:17:09 +01:00
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->println(disp_screen_buffer[last_row]);
|
|
|
|
renderer->Updateframe();
|
2018-08-28 17:13:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-14 13:32:09 +00:00
|
|
|
void Ssd1306Time(void)
|
2018-08-28 17:13:14 +01:00
|
|
|
{
|
|
|
|
char line[12];
|
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->clearDisplay();
|
2019-09-11 16:04:39 +01:00
|
|
|
renderer->setTextSize(Settings.display_size);
|
|
|
|
renderer->setTextFont(Settings.display_font);
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->setCursor(0, 0);
|
2018-08-28 17:13:14 +01:00
|
|
|
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 ]
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->println(line);
|
2018-08-28 17:13:14 +01:00
|
|
|
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]
|
2019-08-19 12:21:54 +01:00
|
|
|
renderer->println(line);
|
|
|
|
renderer->Updateframe();
|
2018-08-28 17:13:14 +01:00
|
|
|
}
|
|
|
|
|
2018-11-14 13:32:09 +00:00
|
|
|
void Ssd1306Refresh(void) // Every second
|
2018-08-28 17:13:14 +01:00
|
|
|
{
|
2019-09-08 19:48:47 +01:00
|
|
|
if (!renderer) return;
|
|
|
|
|
2018-08-28 17:13:14 +01:00
|
|
|
if (Settings.display_mode) { // Mode 0 is User text
|
|
|
|
switch (Settings.display_mode) {
|
|
|
|
case 1: // Time
|
|
|
|
Ssd1306Time();
|
|
|
|
break;
|
|
|
|
case 2: // Local
|
|
|
|
case 3: // Local
|
|
|
|
case 4: // Mqtt
|
|
|
|
case 5: // Mqtt
|
|
|
|
Ssd1306PrintLog();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // USE_DISPLAY_MODES1TO5
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
* Interface
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
2019-08-19 12:21:54 +01:00
|
|
|
bool Xdsp02(byte function)
|
2018-08-28 17:13:14 +01:00
|
|
|
{
|
2019-11-04 09:38:05 +00:00
|
|
|
if (!I2cEnabled(XI2C_04)) { return false; }
|
2019-11-03 16:54:39 +00:00
|
|
|
|
2019-01-28 13:08:33 +00:00
|
|
|
bool result = false;
|
2018-08-28 17:13:14 +01:00
|
|
|
|
2019-11-04 09:38:05 +00:00
|
|
|
if (FUNC_DISPLAY_INIT_DRIVER == function) {
|
|
|
|
SSD1306InitDriver();
|
|
|
|
}
|
|
|
|
else if (XDSP_02 == Settings.display_model) {
|
|
|
|
switch (function) {
|
2018-08-28 17:13:14 +01:00
|
|
|
#ifdef USE_DISPLAY_MODES1TO5
|
2019-11-04 09:38:05 +00:00
|
|
|
case FUNC_DISPLAY_EVERY_SECOND:
|
|
|
|
Ssd1306Refresh();
|
|
|
|
break;
|
2018-08-28 17:13:14 +01:00
|
|
|
#endif // USE_DISPLAY_MODES1TO5
|
2019-11-04 09:38:05 +00:00
|
|
|
case FUNC_DISPLAY_MODEL:
|
|
|
|
result = true;
|
|
|
|
break;
|
2018-08-28 17:13:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // USE_DISPLAY_SSD1306
|
|
|
|
#endif // USE_DISPLAY
|
|
|
|
#endif // USE_I2C
|