196 lines
5.6 KiB
C++
196 lines
5.6 KiB
C++
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
#include "Display_Graphic.h"
|
|
#include <Hoymiles.h>
|
|
#include <NetworkSettings.h>
|
|
#include <map>
|
|
#include <time.h>
|
|
|
|
std::map<DisplayType_t, std::function<U8G2*(uint8_t, uint8_t, uint8_t, uint8_t)>> display_types = {
|
|
{ DisplayType_t::PCD8544, [](uint8_t reset, uint8_t clock, uint8_t data, uint8_t cs) { return new U8G2_PCD8544_84X48_F_4W_HW_SPI(U8G2_R0, cs, data, reset); } },
|
|
{ DisplayType_t::SSD1306, [](uint8_t reset, uint8_t clock, uint8_t data, uint8_t cs) { return new U8G2_SSD1306_128X64_NONAME_F_HW_I2C(U8G2_R0, reset, clock, data); } },
|
|
{ DisplayType_t::SH1106, [](uint8_t reset, uint8_t clock, uint8_t data, uint8_t cs) { return new U8G2_SH1106_128X64_NONAME_F_HW_I2C(U8G2_R0, reset, clock, data); } },
|
|
};
|
|
|
|
DisplayGraphicClass::DisplayGraphicClass()
|
|
{
|
|
}
|
|
|
|
DisplayGraphicClass::~DisplayGraphicClass()
|
|
{
|
|
delete _display;
|
|
}
|
|
|
|
void DisplayGraphicClass::init(DisplayType_t type, uint8_t data, uint8_t clk, uint8_t cs, uint8_t reset)
|
|
{
|
|
_display_type = type;
|
|
if (_display_type > DisplayType_t::None) {
|
|
auto constructor = display_types[_display_type];
|
|
_display = constructor(reset, clk, data, cs);
|
|
_display->begin();
|
|
setContrast(DISPLAY_CONTRAST);
|
|
}
|
|
}
|
|
|
|
void DisplayGraphicClass::calcLineHeights()
|
|
{
|
|
uint8_t yOff = 0;
|
|
for (uint8_t i = 0; i < 4; i++) {
|
|
setFont(i);
|
|
yOff += (_display->getMaxCharHeight());
|
|
_lineOffsets[i] = yOff;
|
|
}
|
|
}
|
|
|
|
void DisplayGraphicClass::setFont(uint8_t line)
|
|
{
|
|
switch (line) {
|
|
case 0:
|
|
_display->setFont((_isLarge) ? u8g2_font_ncenB14_tr : u8g2_font_logisoso16_tr);
|
|
break;
|
|
case 3:
|
|
_display->setFont(u8g2_font_5x8_tr);
|
|
break;
|
|
default:
|
|
_display->setFont((_isLarge) ? u8g2_font_ncenB10_tr : u8g2_font_5x8_tr);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void DisplayGraphicClass::printText(const char* text, uint8_t line)
|
|
{
|
|
uint8_t dispX;
|
|
if (!_isLarge) {
|
|
dispX = (line == 0) ? 5 : 0;
|
|
} else {
|
|
dispX = (line == 0) ? 20 : 5;
|
|
}
|
|
setFont(line);
|
|
|
|
dispX += enableScreensaver ? (_mExtra % 7) : 0;
|
|
_display->drawStr(dispX, _lineOffsets[line], text);
|
|
}
|
|
|
|
void DisplayGraphicClass::setOrientation(uint8_t rotation)
|
|
{
|
|
if (_display_type == DisplayType_t::None) {
|
|
return;
|
|
}
|
|
|
|
switch (rotation) {
|
|
case 0:
|
|
_display->setDisplayRotation(U8G2_R0);
|
|
break;
|
|
case 1:
|
|
_display->setDisplayRotation(U8G2_R1);
|
|
break;
|
|
case 2:
|
|
_display->setDisplayRotation(U8G2_R2);
|
|
break;
|
|
case 3:
|
|
_display->setDisplayRotation(U8G2_R3);
|
|
break;
|
|
}
|
|
|
|
_isLarge = (_display->getWidth() > 100);
|
|
calcLineHeights();
|
|
}
|
|
|
|
void DisplayGraphicClass::setStartupDisplay()
|
|
{
|
|
if (_display_type == DisplayType_t::None) {
|
|
return;
|
|
}
|
|
|
|
_display->clearBuffer();
|
|
printText("OpenDTU!", 0);
|
|
_display->sendBuffer();
|
|
}
|
|
|
|
void DisplayGraphicClass::loop()
|
|
{
|
|
if (_display_type == DisplayType_t::None) {
|
|
return;
|
|
}
|
|
|
|
if ((millis() - _lastDisplayUpdate) > _period) {
|
|
float totalPower = 0;
|
|
float totalYieldDay = 0;
|
|
float totalYieldTotal = 0;
|
|
|
|
uint8_t isprod = 0;
|
|
|
|
for (uint8_t i = 0; i < Hoymiles.getNumInverters(); i++) {
|
|
auto inv = Hoymiles.getInverterByPos(i);
|
|
if (inv == nullptr) {
|
|
continue;
|
|
}
|
|
|
|
if (inv->isProducing()) {
|
|
isprod++;
|
|
}
|
|
|
|
for (auto& c : inv->Statistics()->getChannelsByType(TYPE_AC)) {
|
|
totalPower += inv->Statistics()->getChannelFieldValue(TYPE_AC, c, FLD_PAC);
|
|
totalYieldDay += inv->Statistics()->getChannelFieldValue(TYPE_AC, c, FLD_YD);
|
|
totalYieldTotal += inv->Statistics()->getChannelFieldValue(TYPE_AC, c, FLD_YT);
|
|
}
|
|
}
|
|
|
|
_display->clearBuffer();
|
|
|
|
//=====> Actual Production ==========
|
|
if ((totalPower > 0) && (isprod > 0)) {
|
|
_display->setPowerSave(false);
|
|
if (totalPower > 999) {
|
|
snprintf(_fmtText, sizeof(_fmtText), "%2.1f kW", (totalPower / 1000));
|
|
} else {
|
|
snprintf(_fmtText, sizeof(_fmtText), "%3.0f W", totalPower);
|
|
}
|
|
printText(_fmtText, 0);
|
|
_previousMillis = millis();
|
|
}
|
|
//<=======================
|
|
|
|
//=====> Offline ===========
|
|
else {
|
|
printText("offline", 0);
|
|
// check if it's time to enter power saving mode
|
|
if (millis() - _previousMillis >= (_interval * 2)) {
|
|
_display->setPowerSave(enablePowerSafe);
|
|
}
|
|
}
|
|
//<=======================
|
|
|
|
//=====> Today & Total Production =======
|
|
snprintf(_fmtText, sizeof(_fmtText), "today: %4.0f Wh", totalYieldDay);
|
|
printText(_fmtText, 1);
|
|
|
|
snprintf(_fmtText, sizeof(_fmtText), "total: %.1f kWh", totalYieldTotal);
|
|
printText(_fmtText, 2);
|
|
//<=======================
|
|
|
|
//=====> IP or Date-Time ========
|
|
if (!(_mExtra % 10) && NetworkSettings.localIP()) {
|
|
printText(NetworkSettings.localIP().toString().c_str(), 3);
|
|
} else {
|
|
// Get current time
|
|
time_t now = time(nullptr);
|
|
strftime(_fmtText, sizeof(_fmtText), "%a %d.%m.%Y %H:%M", localtime(&now));
|
|
printText(_fmtText, 3);
|
|
}
|
|
_display->sendBuffer();
|
|
|
|
_mExtra++;
|
|
_lastDisplayUpdate = millis();
|
|
}
|
|
}
|
|
|
|
void DisplayGraphicClass::setContrast(uint8_t contrast)
|
|
{
|
|
if (_display_type == DisplayType_t::None) {
|
|
return;
|
|
}
|
|
_display->setContrast(contrast * 2.55f);
|
|
}
|
|
|
|
DisplayGraphicClass Display; |