ESP32-PaxCounter/src/ledmatrixdisplay.cpp

198 lines
5.7 KiB
C++
Raw Normal View History

#ifdef HAS_MATRIX_DISPLAY
#include "globals.h"
2019-08-17 18:30:45 +02:00
#define NUMROWS 16
#define NUMCOLS 96
2019-05-31 13:20:11 +02:00
#define MATRIX_DISPLAY_PAGES (2) // number of display pages
2019-08-17 18:45:00 +02:00
#define LINE_DIAGRAM_DIVIDER (2) // scales pax numbers to led rows
// local Tag for logging
static const char TAG[] = __FILE__;
uint8_t MatrixDisplayIsOn = 0;
2019-05-07 21:17:06 +02:00
static unsigned long ulLastNumMacs = 0;
2019-05-31 13:20:11 +02:00
static time_t ulLastTime = myTZ.toLocal(now());
LEDMatrix matrix(LED_MATRIX_LA_74138, LED_MATRIX_LB_74138, LED_MATRIX_LC_74138,
LED_MATRIX_LD_74138, LED_MATRIX_EN_74138, LED_MATRIX_DATA_R1,
LED_MATRIX_LATCHPIN, LED_MATRIX_CLOCKPIN);
// Display Buffer 128 = 64 * 16 / 8
2019-08-17 18:30:45 +02:00
uint8_t displaybuf[LED_MATRIX_WIDTH * LED_MATRIX_HEIGHT / 8];
2019-05-31 13:20:11 +02:00
// --- SELECT YOUR FONT HERE ---
const FONT_INFO *ActiveFontInfo = &digital7_18ptFontInfo;
// const FONT_INFO *ActiveFontInfo = &arialNarrow_17ptFontInfo;
// const FONT_INFO *ActiveFontInfo = &gillSansMTCondensed_18ptFontInfo;
// const FONT_INFO *ActiveFontInfo = &gillSansMTCondensed_16ptFontInfo;
2019-05-31 13:20:11 +02:00
const uint8_t *iaActiveFont = ActiveFontInfo->Bitmap;
const FONT_CHAR_INFO *ActiveFontCharInfo = ActiveFontInfo->Descriptors;
2019-05-30 13:00:24 +02:00
void init_matrix_display(bool reverse) {
ESP_LOGI(TAG, "Initializing LED Matrix display");
matrix.begin(displaybuf, LED_MATRIX_WIDTH, LED_MATRIX_HEIGHT);
2019-05-30 13:00:24 +02:00
if (reverse)
matrix.reverse();
matrix.clear();
2019-08-17 18:30:45 +02:00
matrix.drawPoint(0, NUMROWS - 1, 1);
} // init_display
2019-05-31 13:20:11 +02:00
void refreshTheMatrixDisplay(bool nextPage) {
2019-08-17 18:30:45 +02:00
static uint8_t DisplayPage = 0, col = 0, row = 0;
2019-05-31 13:20:11 +02:00
char buff[16];
2019-05-07 21:17:06 +02:00
// if Matrixdisplay is switched off we don't refresh it to relax cpu
if (!MatrixDisplayIsOn && (MatrixDisplayIsOn == cfg.screenon))
return;
// set display on/off according to current device configuration
if (MatrixDisplayIsOn != cfg.screenon) {
MatrixDisplayIsOn = cfg.screenon;
}
2019-05-31 13:20:11 +02:00
if (nextPage) {
DisplayPage =
(DisplayPage >= MATRIX_DISPLAY_PAGES - 1) ? 0 : (DisplayPage + 1);
matrix.clear();
2019-08-17 18:30:45 +02:00
col = 0;
}
2019-05-31 13:20:11 +02:00
switch (DisplayPage % MATRIX_DISPLAY_PAGES) {
2019-08-17 18:30:45 +02:00
// page 0: number of current pax OR footfall line diagram
// page 1: time of day
2019-05-31 13:20:11 +02:00
case 0:
2019-08-17 18:30:45 +02:00
if (cfg.countermode == 1)
{ // cumulative counter mode -> display total number of pax
if (ulLastNumMacs != macs.size()) {
ulLastNumMacs = macs.size();
matrix.clear();
DrawNumber(String(ulLastNumMacs));
}
}
else { // cyclic counter mode -> plot a line diagram
if (ulLastNumMacs != macs.size()) {
// next count cycle?
if (macs.size() == 0) {
col++;
// display full?
if (col >= NUMCOLS) {
col = 0;
matrix.clear();
}
} else {
// clear previous dot
matrix.drawPoint(col, row, 0);
}
// set current dot
ulLastNumMacs = macs.size();
2019-08-17 18:45:00 +02:00
row = NUMROWS - 1 - ((ulLastNumMacs / LINE_DIAGRAM_DIVIDER) % NUMROWS);
2019-08-17 18:30:45 +02:00
matrix.drawPoint(col, row, 1);
}
2019-05-31 13:20:11 +02:00
}
break;
2019-05-31 13:20:11 +02:00
case 1:
const time_t t = myTZ.toLocal(now());
if (ulLastTime != t) {
ulLastTime = t;
matrix.clear();
snprintf(buff, sizeof(buff), "%02d:%02d:%02d", hour(t), minute(t),
second(t));
DrawNumber(String(buff));
}
break;
2019-05-31 13:20:11 +02:00
} // switch page
matrix.scan();
}
// (x, y) top-left position, x should be multiple of 8
void DrawChar(uint16_t x, uint16_t y, char cChar) {
// Get address of char in font char descriptor from font descriptor
auto CharDescAddress = (cChar - ActiveFontInfo->StartChar);
// Get offset of char into font bitmap
uint16_t FontBitmapOffset = ActiveFontCharInfo[CharDescAddress].offset;
// Check font height, if it's less than matrix height we need to
// add some empty lines to font does not stick to the top
if (ActiveFontInfo->CharHeight < (LED_MATRIX_HEIGHT - y)) {
uint8_t FillerLines = (LED_MATRIX_HEIGHT - y) - ActiveFontInfo->CharHeight;
if (FillerLines % 2 == 0) {
// Use floor (round down) to get heighest possible divider
// for missing lines
y = floor(FillerLines / 2);
}
}
int iDst = (x / 8) + (y * 8);
int Shift = x % 8;
for (uint8_t i = 0; i < ActiveFontCharInfo[CharDescAddress].height; i++) {
int iDigitA = iaActiveFont[FontBitmapOffset];
int Left = iDigitA >> Shift;
int Right = iDigitA << (8 - Shift);
displaybuf[iDst] |= Left;
displaybuf[iDst + 1] |= Right;
if (ActiveFontCharInfo[CharDescAddress].width > 8) {
int iDigitB = iaActiveFont[FontBitmapOffset + 1];
Left = iDigitB >> Shift;
Right = iDigitB << (8 - Shift);
displaybuf[iDst + 1] |= Left;
displaybuf[iDst + 2] |= Right;
FontBitmapOffset++;
}
FontBitmapOffset++;
iDst += 8;
}
}
void DrawNumber(String strNum, uint8_t iDotPos) {
uint8_t iNumLength = strNum.length();
uint8_t iDigitPos = 0;
for (int i = 0; i < iNumLength; i++) {
DrawChar(iDigitPos, 0, strNum.charAt(i));
if (i + 1 == iDotPos) {
iDigitPos = iDigitPos + GetCharWidth(strNum.charAt(i)) +
ActiveFontInfo->SpaceWidth;
DrawChar(iDigitPos, 0, '.');
iDigitPos = iDigitPos + GetCharWidth('.') + ActiveFontInfo->SpaceWidth;
} else {
iDigitPos = iDigitPos + GetCharWidth(strNum.charAt(i)) +
ActiveFontInfo->SpaceWidth;
}
}
}
uint8_t GetCharFromFont(char cChar) {
auto cStartChar = ActiveFontInfo->StartChar;
auto iCharLocation = cChar - cStartChar;
auto iCharBitmap = iaActiveFont[iCharLocation];
return iCharBitmap;
}
uint8_t GetCharWidth(char cChar) {
// Get address of char in font char descriptor from font descriptor
auto CharDescAddress = (cChar - ActiveFontInfo->StartChar);
// Get offset of char into font bitmap
auto CharDescriptor = ActiveFontCharInfo[CharDescAddress];
return CharDescriptor.width;
}
#endif // HAS_MATRIX_DISPLAY