forked from btclock/btclock_v3
386 lines
11 KiB
C++
386 lines
11 KiB
C++
#include "epd.hpp"
|
|
|
|
Native_Pin EPD_CS[NUM_SCREENS] = {
|
|
Native_Pin(2),
|
|
Native_Pin(4),
|
|
Native_Pin(6),
|
|
Native_Pin(10),
|
|
Native_Pin(33),
|
|
Native_Pin(21),
|
|
Native_Pin(17),
|
|
#if NUM_SCREENS == 9
|
|
// MCP23X17_Pin(mcp2, 7),
|
|
Native_Pin(-1),
|
|
Native_Pin(-1),
|
|
#endif
|
|
};
|
|
Native_Pin EPD_BUSY[NUM_SCREENS] = {
|
|
Native_Pin(3),
|
|
Native_Pin(5),
|
|
Native_Pin(7),
|
|
Native_Pin(9),
|
|
Native_Pin(37),
|
|
Native_Pin(18),
|
|
Native_Pin(16),
|
|
};
|
|
MCP23X17_Pin EPD_RESET_MPD[NUM_SCREENS] = {
|
|
MCP23X17_Pin(mcp, 8),
|
|
MCP23X17_Pin(mcp, 9),
|
|
MCP23X17_Pin(mcp, 10),
|
|
MCP23X17_Pin(mcp, 11),
|
|
MCP23X17_Pin(mcp, 12),
|
|
MCP23X17_Pin(mcp, 13),
|
|
MCP23X17_Pin(mcp, 14),
|
|
};
|
|
|
|
Native_Pin EPD_DC = Native_Pin(14);
|
|
|
|
GxEPD2_BW<GxEPD2_213_B74, GxEPD2_213_B74::HEIGHT> displays[NUM_SCREENS] = {
|
|
GxEPD2_213_B74(&EPD_CS[0], &EPD_DC, &EPD_RESET_MPD[0], &EPD_BUSY[0]),
|
|
GxEPD2_213_B74(&EPD_CS[1], &EPD_DC, &EPD_RESET_MPD[1], &EPD_BUSY[1]),
|
|
GxEPD2_213_B74(&EPD_CS[2], &EPD_DC, &EPD_RESET_MPD[2], &EPD_BUSY[2]),
|
|
GxEPD2_213_B74(&EPD_CS[3], &EPD_DC, &EPD_RESET_MPD[3], &EPD_BUSY[3]),
|
|
GxEPD2_213_B74(&EPD_CS[4], &EPD_DC, &EPD_RESET_MPD[4], &EPD_BUSY[4]),
|
|
GxEPD2_213_B74(&EPD_CS[5], &EPD_DC, &EPD_RESET_MPD[5], &EPD_BUSY[5]),
|
|
GxEPD2_213_B74(&EPD_CS[6], &EPD_DC, &EPD_RESET_MPD[6], &EPD_BUSY[6]),
|
|
#if NUM_SCREENS == 9
|
|
GxEPD2_213_B74(&EPD8_CS, &EPD_DC, &EPD_RESET_MPD[7], &EPD8_BUSY),
|
|
GxEPD2_213_B74(&EPD9_CS, &EPD_DC, &EPD_RESET_MPD[8], &EPD9_BUSY),
|
|
#endif
|
|
};
|
|
|
|
std::array<String, NUM_SCREENS> currentEpdContent;
|
|
std::array<String, NUM_SCREENS> epdContent;
|
|
uint32_t lastFullRefresh[NUM_SCREENS];
|
|
TaskHandle_t tasks[NUM_SCREENS];
|
|
TaskHandle_t epdTaskHandle = NULL;
|
|
|
|
SemaphoreHandle_t epdUpdateSemaphore[NUM_SCREENS];
|
|
|
|
int fgColor = GxEPD_WHITE;
|
|
int bgColor = GxEPD_BLACK;
|
|
|
|
#define FONT_SMALL Antonio_SemiBold20pt7b
|
|
#define FONT_BIG Antonio_SemiBold90pt7b
|
|
|
|
void setupDisplays()
|
|
{
|
|
for (uint i = 0; i < NUM_SCREENS; i++)
|
|
{
|
|
displays[i].init();
|
|
}
|
|
|
|
for (uint i = 0; i < NUM_SCREENS; i++)
|
|
{
|
|
epdUpdateSemaphore[i] = xSemaphoreCreateBinary();
|
|
xSemaphoreGive(epdUpdateSemaphore[i]);
|
|
|
|
int *taskParam = new int;
|
|
*taskParam = i;
|
|
|
|
xTaskCreate(updateDisplay, ("EpdUpd" + String(i)).c_str(), 4096, taskParam, tskIDLE_PRIORITY, &tasks[i]); // create task
|
|
}
|
|
|
|
xTaskCreate(taskEpd, "epd_task", 2048, NULL, tskIDLE_PRIORITY, &epdTaskHandle);
|
|
|
|
epdContent = {"B",
|
|
"T",
|
|
"C",
|
|
"L",
|
|
"O",
|
|
"C",
|
|
"K"};
|
|
|
|
setEpdContent(epdContent);
|
|
// for (uint i = 0; i < NUM_SCREENS; i++)
|
|
// {
|
|
// xTaskNotifyGive(tasks[i]);
|
|
// }
|
|
}
|
|
|
|
void taskEpd(void *pvParameters)
|
|
{
|
|
while (1)
|
|
{
|
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
|
|
|
bool updatedThisCycle = false;
|
|
|
|
for (uint i = 0; i < NUM_SCREENS; i++)
|
|
{
|
|
if (epdContent[i].compareTo(currentEpdContent[i]) != 0)
|
|
{
|
|
if (!updatedThisCycle)
|
|
{
|
|
updatedThisCycle = true;
|
|
}
|
|
|
|
if (xSemaphoreTake(epdUpdateSemaphore[i], pdMS_TO_TICKS(5000)) == pdTRUE)
|
|
{
|
|
xTaskNotifyGive(tasks[i]);
|
|
}
|
|
else
|
|
{
|
|
Serial.println("Couldnt get screen" + String(i));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void setEpdContent(std::array<String, NUM_SCREENS> newEpdContent)
|
|
{
|
|
epdContent = newEpdContent;
|
|
if (epdTaskHandle != NULL)
|
|
xTaskNotifyGive(epdTaskHandle);
|
|
if (eventSourceTaskHandle != NULL)
|
|
xTaskNotifyGive(eventSourceTaskHandle);
|
|
}
|
|
|
|
extern "C" void updateDisplay(void *pvParameters) noexcept
|
|
{
|
|
const int epdIndex = *(int *)pvParameters;
|
|
delete (int *)pvParameters;
|
|
|
|
for (;;)
|
|
{
|
|
// Wait for the task notification
|
|
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
|
|
|
if (epdContent[epdIndex].compareTo(currentEpdContent[epdIndex]) != 0)
|
|
{
|
|
|
|
displays[epdIndex].init(0, false); // Little longer reset duration because of MCP
|
|
uint count = 0;
|
|
while (EPD_BUSY[epdIndex].digitalRead() == HIGH || count < 10)
|
|
{
|
|
vTaskDelay(pdMS_TO_TICKS(100));
|
|
if (count >= 9)
|
|
{
|
|
displays[epdIndex].init(0, false);
|
|
}
|
|
count++;
|
|
}
|
|
|
|
bool updatePartial = true;
|
|
|
|
// Full Refresh every half hour
|
|
if (!lastFullRefresh[epdIndex] || (millis() - lastFullRefresh[epdIndex]) > (preferences.getUInt("fullRefreshMin", 30) * 60 * 1000))
|
|
{
|
|
updatePartial = false;
|
|
lastFullRefresh[epdIndex] = millis();
|
|
}
|
|
|
|
// if (updatePartial)
|
|
// {
|
|
// displays[epdIndex].setPartialWindow(0, 0, displays[i].width(), display[i].height());
|
|
// }
|
|
|
|
if (strstr(epdContent[epdIndex].c_str(), "/") != NULL)
|
|
{
|
|
String top = epdContent[epdIndex].substring(0, epdContent[epdIndex].indexOf("/"));
|
|
String bottom = epdContent[epdIndex].substring(epdContent[epdIndex].indexOf("/") + 1);
|
|
#ifdef PAGED_WRITE
|
|
splitTextPaged(epdIndex, top, bottom, updatePartial);
|
|
#else
|
|
splitText(epdIndex, top, bottom, updatePartial);
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
|
|
#ifdef PAGED_WRITE
|
|
showDigitPaged(epdIndex, epdContent[epdIndex].c_str()[0], updatePartial, &FONT_BIG);
|
|
#else
|
|
showDigit(epdIndex, epdContent[epdIndex].c_str()[0], updatePartial, &FONT_BIG);
|
|
#endif
|
|
}
|
|
|
|
#ifdef PAGED_WRITE
|
|
currentEpdContent[epdIndex] = epdContent[epdIndex];
|
|
#else
|
|
char tries = 0;
|
|
while (tries < 3)
|
|
{
|
|
if (displays[epdIndex].displayWithReturn(updatePartial))
|
|
{
|
|
displays[epdIndex].hibernate();
|
|
break;
|
|
}
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(100));
|
|
tries++;
|
|
}
|
|
currentEpdContent[epdIndex] = epdContent[epdIndex];
|
|
|
|
#endif
|
|
}
|
|
xSemaphoreGive(epdUpdateSemaphore[epdIndex]);
|
|
}
|
|
}
|
|
|
|
void updateDisplayAlt(int epdIndex)
|
|
{
|
|
}
|
|
|
|
void splitText(const uint dispNum, String top, String bottom, bool partial)
|
|
{
|
|
displays[dispNum].setRotation(2);
|
|
displays[dispNum].setFont(&FONT_SMALL);
|
|
displays[dispNum].setTextColor(getFgColor());
|
|
|
|
// Top text
|
|
int16_t ttbx, ttby;
|
|
uint16_t ttbw, ttbh;
|
|
displays[dispNum].getTextBounds(top, 0, 0, &ttbx, &ttby, &ttbw, &ttbh);
|
|
uint16_t tx = ((displays[dispNum].width() - ttbw) / 2) - ttbx;
|
|
uint16_t ty = ((displays[dispNum].height() - ttbh) / 2) - ttby - ttbh / 2 - 12;
|
|
|
|
// Bottom text
|
|
int16_t tbbx, tbby;
|
|
uint16_t tbbw, tbbh;
|
|
displays[dispNum].getTextBounds(bottom, 0, 0, &tbbx, &tbby, &tbbw, &tbbh);
|
|
uint16_t bx = ((displays[dispNum].width() - tbbw) / 2) - tbbx;
|
|
uint16_t by = ((displays[dispNum].height() - tbbh) / 2) - tbby + tbbh / 2 + 12;
|
|
|
|
// Make separator as wide as the shortest text.
|
|
uint16_t lineWidth, lineX;
|
|
if (tbbw < ttbh)
|
|
lineWidth = tbbw;
|
|
else
|
|
lineWidth = ttbw;
|
|
lineX = round((displays[dispNum].width() - lineWidth) / 2);
|
|
|
|
displays[dispNum].fillScreen(getBgColor());
|
|
displays[dispNum].setCursor(tx, ty);
|
|
displays[dispNum].print(top);
|
|
displays[dispNum].fillRoundRect(lineX, displays[dispNum].height() / 2 - 3, lineWidth, 6, 3, getFgColor());
|
|
displays[dispNum].setCursor(bx, by);
|
|
displays[dispNum].print(bottom);
|
|
}
|
|
|
|
void splitTextPaged(const uint dispNum, String top, String bottom, bool partial)
|
|
{
|
|
displays[dispNum].setRotation(2);
|
|
displays[dispNum].setFont(&FONT_SMALL);
|
|
displays[dispNum].setTextColor(getFgColor());
|
|
|
|
// Top text
|
|
int16_t ttbx, ttby;
|
|
uint16_t ttbw, ttbh;
|
|
displays[dispNum].getTextBounds(top, 0, 0, &ttbx, &ttby, &ttbw, &ttbh);
|
|
uint16_t tx = ((displays[dispNum].width() - ttbw) / 2) - ttbx;
|
|
uint16_t ty = ((displays[dispNum].height() - ttbh) / 2) - ttby - ttbh / 2 - 12;
|
|
|
|
// Bottom text
|
|
int16_t tbbx, tbby;
|
|
uint16_t tbbw, tbbh;
|
|
displays[dispNum].getTextBounds(bottom, 0, 0, &tbbx, &tbby, &tbbw, &tbbh);
|
|
uint16_t bx = ((displays[dispNum].width() - tbbw) / 2) - tbbx;
|
|
uint16_t by = ((displays[dispNum].height() - tbbh) / 2) - tbby + tbbh / 2 + 12;
|
|
|
|
// Make separator as wide as the shortest text.
|
|
uint16_t lineWidth, lineX;
|
|
if (tbbw < ttbh)
|
|
lineWidth = tbbw;
|
|
else
|
|
lineWidth = ttbw;
|
|
lineX = round((displays[dispNum].width() - lineWidth) / 2);
|
|
|
|
if (partial)
|
|
{
|
|
displays[dispNum].setPartialWindow(0, 0, displays[dispNum].width(), displays[dispNum].height());
|
|
}
|
|
else
|
|
{
|
|
displays[dispNum].setFullWindow();
|
|
}
|
|
displays[dispNum].firstPage();
|
|
|
|
do
|
|
{
|
|
displays[dispNum].fillScreen(getBgColor());
|
|
displays[dispNum].setCursor(tx, ty);
|
|
displays[dispNum].print(top);
|
|
displays[dispNum].fillRoundRect(lineX, displays[dispNum].height() / 2 - 3, lineWidth, 6, 3, getFgColor());
|
|
displays[dispNum].setCursor(bx, by);
|
|
displays[dispNum].print(bottom);
|
|
} while (displays[dispNum].nextPage());
|
|
}
|
|
|
|
void showDigit(const uint dispNum, char chr, bool partial, const GFXfont *font)
|
|
{
|
|
String str(chr);
|
|
displays[dispNum].setRotation(2);
|
|
displays[dispNum].setFont(font);
|
|
displays[dispNum].setTextColor(getFgColor());
|
|
int16_t tbx, tby;
|
|
uint16_t tbw, tbh;
|
|
displays[dispNum].getTextBounds(str, 0, 0, &tbx, &tby, &tbw, &tbh);
|
|
// center the bounding box by transposition of the origin:
|
|
uint16_t x = ((displays[dispNum].width() - tbw) / 2) - tbx;
|
|
uint16_t y = ((displays[dispNum].height() - tbh) / 2) - tby;
|
|
displays[dispNum].fillScreen(getBgColor());
|
|
displays[dispNum].setCursor(x, y);
|
|
displays[dispNum].print(str);
|
|
}
|
|
|
|
void showDigitPaged(const uint dispNum, char chr, bool partial, const GFXfont *font)
|
|
{
|
|
String str(chr);
|
|
displays[dispNum].setRotation(2);
|
|
displays[dispNum].setFont(font);
|
|
displays[dispNum].setTextColor(getFgColor());
|
|
int16_t tbx, tby;
|
|
uint16_t tbw, tbh;
|
|
displays[dispNum].getTextBounds(str, 0, 0, &tbx, &tby, &tbw, &tbh);
|
|
// center the bounding box by transposition of the origin:
|
|
uint16_t x = ((displays[dispNum].width() - tbw) / 2) - tbx;
|
|
uint16_t y = ((displays[dispNum].height() - tbh) / 2) - tby;
|
|
if (partial)
|
|
{
|
|
displays[dispNum].setPartialWindow(0, 0, displays[dispNum].width(), displays[dispNum].height());
|
|
}
|
|
else
|
|
{
|
|
displays[dispNum].setFullWindow();
|
|
}
|
|
displays[dispNum].firstPage();
|
|
|
|
do
|
|
{
|
|
displays[dispNum].fillScreen(getBgColor());
|
|
displays[dispNum].setCursor(x, y);
|
|
displays[dispNum].print(str);
|
|
} while (displays[dispNum].nextPage());
|
|
}
|
|
|
|
int getBgColor()
|
|
{
|
|
return bgColor;
|
|
}
|
|
|
|
int getFgColor()
|
|
{
|
|
return fgColor;
|
|
}
|
|
|
|
void setBgColor(int color)
|
|
{
|
|
bgColor = color;
|
|
}
|
|
|
|
void setFgColor(int color)
|
|
{
|
|
fgColor = color;
|
|
}
|
|
|
|
std::array<String, NUM_SCREENS> getCurrentEpdContent()
|
|
{
|
|
// Serial.println("currentEpdContent");
|
|
|
|
// for (int i = 0; i < NUM_SCREENS; i++) {
|
|
// Serial.printf("%d = %s", i, currentEpdContent[i]);
|
|
// }
|
|
return currentEpdContent;
|
|
}
|