Adafruit Library
dispDrvThinkInkGrayscale4Eaamfgn.h
Go to the documentation of this file.
1 
16 #ifndef WS_DRV_THINKINK_GRAYSCALE4_EAAMFGN_H
17 #define WS_DRV_THINKINK_GRAYSCALE4_EAAMFGN_H
18 
19 #include "dispDrvBase.h"
20 
21 #define STATUS_BAR_HEIGHT \
22  20
23 #define STATUS_BAR_BORDER 1
24 #define STATUS_BAR_ICON_SZ 16
25 #define STATUS_BAR_ICON_SPACING \
26  4
27 #define STATUS_BAR_ICON_MARGIN \
28  5
29 
30 
34 public:
49  drvDispThinkInkGrayscale4Eaamfgn(int16_t dc, int16_t rst, int16_t cs,
50  int16_t sram_cs = -1, int16_t busy = -1)
51  : dispDrvBase(dc, rst, cs, sram_cs, busy), _display(nullptr) {}
52 
54  if (_display) {
55  // Clear the display buffer before deleting
56  _display->clearBuffer();
57  _display->display();
58  delete _display;
59  _display = nullptr;
60  }
61  }
62 
72  bool begin(thinkinkmode_t mode, bool reset = true) override {
73  _display = new ThinkInk_290_Grayscale4_EAAMFGN(_pin_dc, _pin_rst, _pin_cs,
75  if (!_display)
76  return false; // Allocation failed
77 
78  // Initialize the display
79  _display->begin(mode);
80  // Configure display settings
81  _display->setTextSize(_text_sz);
82  _display->setTextColor(EPD_BLACK);
83  _display->setTextWrap(false);
84  _height = _display->height();
85  _width = _display->width();
86  _display->clearBuffer();
87  return true;
88  }
89 
93  virtual void showSplash() override {
94  if (!_display)
95  return;
96  _display->drawBitmap(0, 0, epd_bitmap_ws_logo_296128, 296, 128, EPD_BLACK);
97  _display->display();
98  }
99 
105  virtual void drawStatusBar(const char *io_username) override {
106  if (!_display)
107  return;
108 
109  // Clear the entire display buffer to remove splash screen
110  _display->clearBuffer();
111 
112  // Draw status bar
113  _display->fillRect(0, 0, _display->width(), STATUS_BAR_HEIGHT, EPD_BLACK);
114  _display->fillRect(STATUS_BAR_BORDER, STATUS_BAR_BORDER,
115  _display->width() - (2 * STATUS_BAR_BORDER),
116  STATUS_BAR_HEIGHT - (2 * STATUS_BAR_BORDER), EPD_WHITE);
117 
118  // Draw username on left side of the status bar
119  _display->setTextSize(1);
120  _display->setTextColor(EPD_BLACK);
121  _display->setCursor(5, 6);
122  _display->print(io_username);
123 
124  // Calculate status bar icon positions and center vertically
129  _display->width() - STATUS_BAR_ICON_SZ - STATUS_BAR_ICON_MARGIN;
134  // Draw icons on right side of the status bar
135  _display->drawBitmap(_statusbar_icon_cloud_x, _statusbar_icons_y,
137  STATUS_BAR_ICON_SZ, EPD_BLACK);
138  _display->drawBitmap(_statusbar_icon_wifi_x, _statusbar_icons_y,
140  STATUS_BAR_ICON_SZ, EPD_BLACK);
141  _display->drawBitmap(_statusbar_icon_battery_x, _statusbar_icons_y,
143  STATUS_BAR_ICON_SZ, EPD_BLACK);
144 
145  _display->display();
146  }
147 
158  void updateStatusBar(int8_t rssi, uint8_t bat, bool mqtt_status) override {
159  if (!_display)
160  return;
161 
162  // Only update wifi icon if the RSSI has changed significantly (+/- 5dB)
163  bool update_rssi = abs(rssi - _statusbar_rssi) >= 5;
164  // Only update cloud icon if MQTT status has changed
165  bool update_mqtt = mqtt_status != _statusbar_mqtt_connected;
166 
167  // No need to update if nothing has changed
168  if (!update_rssi && !update_mqtt)
169  return;
170 
171  if (update_mqtt) {
172  // updating the RSSI occurs too frequently to be practical
173  _display->fillRect(_statusbar_icon_cloud_x, _statusbar_icons_y,
175  if (mqtt_status) {
176  _display->drawBitmap(_statusbar_icon_cloud_x, _statusbar_icons_y,
178  STATUS_BAR_ICON_SZ, EPD_BLACK);
179  } else {
180  _display->drawBitmap(_statusbar_icon_cloud_x, _statusbar_icons_y,
181  epd_bmp_cloud_offline, STATUS_BAR_ICON_SZ,
182  STATUS_BAR_ICON_SZ, EPD_BLACK);
183  }
184  _statusbar_mqtt_connected = mqtt_status;
185  }
186 
187  // Update WiFi icon only if RSSI has changed significantly (+/-3dB)
188  if (update_rssi) {
189  const unsigned char *wifi_icon = epd_bmp_wifi_no_signal;
190  if (rssi >= -50) {
191  wifi_icon = epd_bmp_wifi_full;
192  } else if (rssi < -50 && rssi >= -60) {
193  wifi_icon = epd_bmp_wifi_fair;
194  } else if (rssi < -60 && rssi >= -70) {
195  wifi_icon = epd_bmp_wifi_weak;
196  } else if (rssi < -70 && rssi >= -80) {
197  wifi_icon = epd_bmp_wifi_no_signal;
198  } else {
199  wifi_icon = epd_bmp_wifi_no_signal;
200  }
201  // Clear and draw the new WiFi icon, based on RSSI
202  _display->fillRect(_statusbar_icon_wifi_x, _statusbar_icons_y,
204  _display->drawBitmap(_statusbar_icon_wifi_x, _statusbar_icons_y,
206  EPD_BLACK);
207  _statusbar_rssi = rssi;
208  }
209 
210  _display->display();
211  }
212 
220  virtual void writeMessage(const char *message) override {
221  if (_display == nullptr)
222  return;
223 
224  // Clear only the area below the status bar
225  _display->fillRect(0, STATUS_BAR_HEIGHT, _display->width(),
226  _display->height() - STATUS_BAR_HEIGHT, EPD_WHITE);
227  // Add padding between status bar and text content
228  int16_t y_idx = STATUS_BAR_HEIGHT + 4;
229  _display->setCursor(0, y_idx);
230 
231  // Calculate the line height based on the text size (NOTE: base height is
232  // 8px)
233  int16_t line_height = 8 * _text_sz;
234  uint16_t c_idx = 0;
235  size_t msg_size = strlen(message);
236 
237  // Reset the text size to the configured value before we write
238  _display->setTextSize(_text_sz);
239 
240  for (size_t i = 0; i < msg_size && c_idx < msg_size; i++) {
241  if (y_idx + line_height > _height)
242  break;
243  if (message[i] == '\\' && i + 1 < msg_size &&
244  (message[i + 1] == 'n' || message[i + 1] == 'r')) {
245  // Handle \r\n sequence as a single newline
246  if (message[i + 1] == 'r' && i + 3 < msg_size &&
247  message[i + 2] == '\\' && message[i + 3] == 'n') {
248  // Skip to the next line
249  if (y_idx + line_height > _height)
250  break;
251  y_idx += line_height;
252  _display->setCursor(0, y_idx);
253  i += 3;
254  } else if (message[i + 1] == 'n') {
255  // Skip to the next line
256  if (y_idx + line_height > _height)
257  break;
258  y_idx += line_height;
259  _display->setCursor(0, y_idx);
260  i++;
261  }
262  } else if (message[i] == 0xC2 && message[i + 1] == 0xB0) {
263  // Degree symbol
264  _display->write(char(247));
265  i++;
266  } else {
267  _display->print(message[i]);
268  }
269  }
270  _display->display();
271  }
272 
273 private:
274  ThinkInk_290_Grayscale4_EAAMFGN *_display;
275 };
276 
277 #endif // WS_DRV_THINKINK_GRAYSCALE4_EAAMFGN_H
const unsigned char epd_bmp_bat_full[]
battery-full-thin-full icon from FontAwesome (16x16px)
Definition: icons.h:70
int _statusbar_icons_y
Y position of status bar icons.
Definition: dispDrvBase.h:178
#define STATUS_BAR_ICON_MARGIN
Margin from edge of display to status bar icons in pixels.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:27
const unsigned char epd_bmp_wifi_fair[]
wifi-fair-thin-full icon from FontAwesome (16x16px)
Definition: icons.h:46
drvDispThinkInkGrayscale4Eaamfgn(int16_t dc, int16_t rst, int16_t cs, int16_t sram_cs=-1, int16_t busy=-1)
Constructor for the ThinkInk Grayscale 4-level EAAMFGN display driver.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:49
int _statusbar_icon_battery_x
X position of battery icon.
Definition: dispDrvBase.h:179
virtual void drawStatusBar(const char *io_username) override
Draws a status bar at the top of the display.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:105
const unsigned char epd_bmp_wifi_no_signal[]
wifi-slash-thin-full icon from FontAwesome (16x16px)
Definition: icons.h:62
uint8_t _text_sz
Text size for displaying a message.
Definition: dispDrvBase.h:173
bool begin(thinkinkmode_t mode, bool reset=true) override
Attempts to initialize the ThinkInk Grayscale 4-level EAAMFGN display driver.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:72
int8_t _statusbar_rssi
RSSI value for status bar.
Definition: dispDrvBase.h:182
#define STATUS_BAR_ICON_SZ
Size of status bar icons in pixels.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:24
#define STATUS_BAR_BORDER
Border around the status bar in pixels.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:23
int16_t _pin_sram_cs
Optional EPD SRAM chip select pin.
Definition: dispDrvBase.h:169
Driver for a ThinkInk 2.9" Grayscale 4-level EAAMFGN display.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:33
virtual void writeMessage(const char *message) override
Writes a message to the display.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:220
int _statusbar_icon_cloud_x
X position of cloud icon.
Definition: dispDrvBase.h:181
int16_t _pin_rst
Reset pin.
Definition: dispDrvBase.h:166
int16_t _width
Width of the display.
Definition: dispDrvBase.h:175
#define STATUS_BAR_ICON_SPACING
Spacing between status bar icons in pixels.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:25
const unsigned char epd_bmp_wifi_full[]
wifi-thin-full icon from FontAwesome (16x16px)
Definition: icons.h:38
virtual void showSplash() override
Displays a splash screen.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:93
#define STATUS_BAR_HEIGHT
Height of the status bar in pixels, assumes 16px icons.
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:21
const unsigned char epd_bmp_cloud_online[]
cloud-thin-full icon from FontAwesome (16x16px)
Definition: icons.h:22
int16_t _height
Height of the display.
Definition: dispDrvBase.h:174
bool _statusbar_mqtt_connected
MQTT connection status for the status bar.
Definition: dispDrvBase.h:185
void updateStatusBar(int8_t rssi, uint8_t bat, bool mqtt_status) override
Updates the status bar with current information (battery level, connectivity status, etc).
Definition: dispDrvThinkInkGrayscale4Eaamfgn.h:158
int16_t _pin_busy
Optional Busy pin.
Definition: dispDrvBase.h:168
Abstract base class for display drivers. This class provides a common interface for all display drive...
Definition: dispDrvBase.h:28
int16_t _pin_cs
Chip Select pin.
Definition: dispDrvBase.h:167
const unsigned char epd_bmp_wifi_weak[]
wifi-weak-thin-full icon from FontAwesome (16x16px)
Definition: icons.h:54
int16_t _pin_dc
Data/Command pin.
Definition: dispDrvBase.h:165
int _statusbar_icon_wifi_x
X position of WiFi icon.
Definition: dispDrvBase.h:180