Paperd.Ink Library 0.0.5
Library for interacting with Paperd.Ink devices.
Loading...
Searching...
No Matches
GxEPD2_213_BN.cpp
Go to the documentation of this file.
1// Display Library for SPI e-paper panels from Dalian Good Display and boards from Waveshare.
2// Requires HW SPI and Adafruit_GFX. Caution: the e-paper panels require 3.3V supply AND data lines!
3//
4// Panel: DEPG0213BN : https://www.dke.top/products/213-eiink-display
5// Controller : SSD1680 : https://www.good-display.com/companyfile/101.html
6// Display: LILYGO® T5 V2.3.1 2.13 inch : https://www.aliexpress.com/item/32869729970.html
7//
8// Author: Jean-Marc Zingg/
9//
10// Version: see library.properties
11//
12// Library: https://github.com/ZinggJM/GxEPD2
13
14#include "GxEPD2_213_BN.h"
15
16GxEPD2_213_BN::GxEPD2_213_BN(int16_t cs, int16_t dc, int16_t rst, int16_t busy) :
17 GxEPD2_EPD(cs, dc, rst, busy, HIGH, 10000000, WIDTH, HEIGHT, panel, hasColor, hasPartialUpdate, hasFastPartialUpdate)
18{
19}
20
21void GxEPD2_213_BN::clearScreen(uint8_t value)
22{
23 writeScreenBuffer(value);
24 refresh(true);
26}
27
29{
30 if (!_using_partial_mode) _Init_Part();
31 if (_initial_write) _writeScreenBuffer(0x26, value); // set previous
32 _writeScreenBuffer(0x24, value); // set current
33 _initial_write = false; // initial full screen buffer clean done
34}
35
37{
38 if (!_using_partial_mode) _Init_Part();
39 _writeScreenBuffer(0x26, value); // set previous
40 _writeScreenBuffer(0x24, value); // set current
41}
42
43void GxEPD2_213_BN::_writeScreenBuffer(uint8_t command, uint8_t value)
44{
45 _writeCommand(command);
46 for (uint32_t i = 0; i < uint32_t(WIDTH) * uint32_t(HEIGHT) / 8; i++)
47 {
48 _writeData(value);
49 }
50}
51
52void GxEPD2_213_BN::writeImage(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
53{
54 _writeImage(0x24, bitmap, x, y, w, h, invert, mirror_y, pgm);
55}
56
57void GxEPD2_213_BN::writeImageForFullRefresh(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
58{
59 _writeImage(0x26, bitmap, x, y, w, h, invert, mirror_y, pgm); // set previous
60 _writeImage(0x24, bitmap, x, y, w, h, invert, mirror_y, pgm);
61}
62
63
64void GxEPD2_213_BN::writeImageAgain(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
65{
66 _writeImage(0x26, bitmap, x, y, w, h, invert, mirror_y, pgm); // set previous
67 _writeImage(0x24, bitmap, x, y, w, h, invert, mirror_y, pgm); // set current
68}
69
70void GxEPD2_213_BN::_writeImage(uint8_t command, const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
71{
72 if (_initial_write) writeScreenBuffer(); // initial full screen buffer clean
73 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
74 int16_t wb = (w + 7) / 8; // width bytes, bitmaps are padded
75 x -= x % 8; // byte boundary
76 w = wb * 8; // byte boundary
77 int16_t x1 = x < 0 ? 0 : x; // limit
78 int16_t y1 = y < 0 ? 0 : y; // limit
79 int16_t w1 = x + w < int16_t(WIDTH) ? w : int16_t(WIDTH) - x; // limit
80 int16_t h1 = y + h < int16_t(HEIGHT) ? h : int16_t(HEIGHT) - y; // limit
81 int16_t dx = x1 - x;
82 int16_t dy = y1 - y;
83 w1 -= dx;
84 h1 -= dy;
85 if ((w1 <= 0) || (h1 <= 0)) return;
86 if (!_using_partial_mode) _Init_Part();
87 _setPartialRamArea(x1, y1, w1, h1);
88 _writeCommand(command);
89 for (int16_t i = 0; i < h1; i++)
90 {
91 for (int16_t j = 0; j < w1 / 8; j++)
92 {
93 uint8_t data;
94 // use wb, h of bitmap for index!
95 int16_t idx = mirror_y ? j + dx / 8 + ((h - 1 - (i + dy))) * wb : j + dx / 8 + (i + dy) * wb;
96 if (pgm)
97 {
98#if defined(__AVR) || defined(ESP8266) || defined(ESP32)
99 data = pgm_read_byte(&bitmap[idx]);
100#else
101 data = bitmap[idx];
102#endif
103 }
104 else
105 {
106 data = bitmap[idx];
107 }
108 if (invert) data = ~data;
109 _writeData(data);
110 }
111 }
112 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
113}
114
115void GxEPD2_213_BN::writeImagePart(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
116 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
117{
118 _writeImagePart(0x24, bitmap, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
119}
120
121void GxEPD2_213_BN::writeImagePartAgain(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
122 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
123{
124 _writeImagePart(0x26, bitmap, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm); // set previous
125 _writeImagePart(0x24, bitmap, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm); // set current
126}
127
128void GxEPD2_213_BN::_writeImagePart(uint8_t command, const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
129 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
130{
131 if (_initial_write) writeScreenBuffer(); // initial full screen buffer clean
132 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
133 if ((w_bitmap < 0) || (h_bitmap < 0) || (w < 0) || (h < 0)) return;
134 if ((x_part < 0) || (x_part >= w_bitmap)) return;
135 if ((y_part < 0) || (y_part >= h_bitmap)) return;
136 int16_t wb_bitmap = (w_bitmap + 7) / 8; // width bytes, bitmaps are padded
137 x_part -= x_part % 8; // byte boundary
138 w = w_bitmap - x_part < w ? w_bitmap - x_part : w; // limit
139 h = h_bitmap - y_part < h ? h_bitmap - y_part : h; // limit
140 x -= x % 8; // byte boundary
141 w = 8 * ((w + 7) / 8); // byte boundary, bitmaps are padded
142 int16_t x1 = x < 0 ? 0 : x; // limit
143 int16_t y1 = y < 0 ? 0 : y; // limit
144 int16_t w1 = x + w < int16_t(WIDTH) ? w : int16_t(WIDTH) - x; // limit
145 int16_t h1 = y + h < int16_t(HEIGHT) ? h : int16_t(HEIGHT) - y; // limit
146 int16_t dx = x1 - x;
147 int16_t dy = y1 - y;
148 w1 -= dx;
149 h1 -= dy;
150 if ((w1 <= 0) || (h1 <= 0)) return;
151 if (!_using_partial_mode) _Init_Part();
152 _setPartialRamArea(x1, y1, w1, h1);
153 _writeCommand(command);
154 for (int16_t i = 0; i < h1; i++)
155 {
156 for (int16_t j = 0; j < w1 / 8; j++)
157 {
158 uint8_t data;
159 // use wb_bitmap, h_bitmap of bitmap for index!
160 int16_t idx = mirror_y ? x_part / 8 + j + dx / 8 + ((h_bitmap - 1 - (y_part + i + dy))) * wb_bitmap : x_part / 8 + j + dx / 8 + (y_part + i + dy) * wb_bitmap;
161 if (pgm)
162 {
163#if defined(__AVR) || defined(ESP8266) || defined(ESP32)
164 data = pgm_read_byte(&bitmap[idx]);
165#else
166 data = bitmap[idx];
167#endif
168 }
169 else
170 {
171 data = bitmap[idx];
172 }
173 if (invert) data = ~data;
174 _writeData(data);
175 }
176 }
177 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
178}
179
180void GxEPD2_213_BN::writeImage(const uint8_t* black, const uint8_t* color, int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
181{
182 if (black)
183 {
184 writeImage(black, x, y, w, h, invert, mirror_y, pgm);
185 }
186}
187
188void GxEPD2_213_BN::writeImagePart(const uint8_t* black, const uint8_t* color, int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
189 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
190{
191 if (black)
192 {
193 writeImagePart(black, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
194 }
195}
196
197void GxEPD2_213_BN::writeNative(const uint8_t* data1, const uint8_t* data2, int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
198{
199 if (data1)
200 {
201 writeImage(data1, x, y, w, h, invert, mirror_y, pgm);
202 }
203}
204
205void GxEPD2_213_BN::drawImage(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
206{
207 writeImage(bitmap, x, y, w, h, invert, mirror_y, pgm);
208 refresh(x, y, w, h);
209 writeImageAgain(bitmap, x, y, w, h, invert, mirror_y, pgm);
210}
211
212void GxEPD2_213_BN::drawImagePart(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
213 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
214{
215 writeImagePart(bitmap, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
216 refresh(x, y, w, h);
217 writeImagePartAgain(bitmap, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
218}
219
220void GxEPD2_213_BN::drawImage(const uint8_t* black, const uint8_t* color, int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
221{
222 if (black)
223 {
224 drawImage(black, x, y, w, h, invert, mirror_y, pgm);
225 }
226}
227
228void GxEPD2_213_BN::drawImagePart(const uint8_t* black, const uint8_t* color, int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
229 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
230{
231 if (black)
232 {
233 drawImagePart(black, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
234 }
235}
236
237void GxEPD2_213_BN::drawNative(const uint8_t* data1, const uint8_t* data2, int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
238{
239 if (data1)
240 {
241 drawImage(data1, x, y, w, h, invert, mirror_y, pgm);
242 }
243}
244
245void GxEPD2_213_BN::refresh(bool partial_update_mode)
246{
247 if (partial_update_mode) refresh(0, 0, WIDTH, HEIGHT);
248 else
249 {
250 if (_using_partial_mode) _Init_Full();
251 _Update_Full();
252 _initial_refresh = false; // initial full update done
253 }
254}
255
256void GxEPD2_213_BN::refresh(int16_t x, int16_t y, int16_t w, int16_t h)
257{
258 if (_initial_refresh) return refresh(false); // initial update needs be full update
259 // intersection with screen
260 int16_t w1 = x < 0 ? w + x : w; // reduce
261 int16_t h1 = y < 0 ? h + y : h; // reduce
262 int16_t x1 = x < 0 ? 0 : x; // limit
263 int16_t y1 = y < 0 ? 0 : y; // limit
264 w1 = x1 + w1 < int16_t(WIDTH) ? w1 : int16_t(WIDTH) - x1; // limit
265 h1 = y1 + h1 < int16_t(HEIGHT) ? h1 : int16_t(HEIGHT) - y1; // limit
266 if ((w1 <= 0) || (h1 <= 0)) return;
267 // make x1, w1 multiple of 8
268 w1 += x1 % 8;
269 if (w1 % 8 > 0) w1 += 8 - w1 % 8;
270 x1 -= x1 % 8;
271 if (!_using_partial_mode) _Init_Part();
272 _setPartialRamArea(x1, y1, w1, h1);
273 _Update_Part();
274}
275
277{
278 _PowerOff();
279}
280
282{
283 _PowerOff();
284 if (_rst >= 0)
285 {
286 _writeCommand(0x10); // deep sleep mode
287 _writeData(0x1); // enter deep sleep
288 _hibernating = true;
289 }
290}
291
292void GxEPD2_213_BN::_setPartialRamArea(uint16_t x, uint16_t y, uint16_t w, uint16_t h)
293{
294 _writeCommand(0x11); // set ram entry mode
295 _writeData(0x03); // x increase, y increase : normal mode
296 _writeCommand(0x44);
297 _writeData(x / 8);
298 _writeData((x + w - 1) / 8);
299 _writeCommand(0x45);
300 _writeData(y % 256);
301 _writeData(y / 256);
302 _writeData((y + h - 1) % 256);
303 _writeData((y + h - 1) / 256);
304 _writeCommand(0x4e);
305 _writeData(x / 8);
306 _writeCommand(0x4f);
307 _writeData(y % 256);
308 _writeData(y / 256);
309}
310
311void GxEPD2_213_BN::_PowerOn()
312{
313 if (!_power_is_on)
314 {
315 _writeCommand(0x22);
316 _writeData(0xf8);
317 _writeCommand(0x20);
318 _waitWhileBusy("_PowerOn", power_on_time);
319 }
320 _power_is_on = true;
321}
322
323void GxEPD2_213_BN::_PowerOff()
324{
325 if (_power_is_on)
326 {
327 _writeCommand(0x22);
328 _writeData(0x83);
329 _writeCommand(0x20);
330 _waitWhileBusy("_PowerOff", power_off_time);
331 }
332 _power_is_on = false;
333 _using_partial_mode = false;
334}
335
336void GxEPD2_213_BN::_InitDisplay()
337{
338 if (_hibernating) _reset();
339 delay(10); // 10ms according to specs
340 _writeCommand(0x12); //SWRESET
341 delay(10); // 10ms according to specs
342 _writeCommand(0x01); //Driver output control
343 _writeData(0x27);
344 _writeData(0x01);
345 _writeData(0x00);
346 _writeCommand(0x11); //data entry mode
347 _writeData(0x03);
348 _writeCommand(0x3C); //BorderWavefrom
349 _writeData(0x05);
350 _writeCommand(0x21); // Display update control
351 _writeData(0x00);
352 _writeData(0x80);
353 _writeCommand(0x18); //Read built-in temperature sensor
354 _writeData(0x80);
355 _setPartialRamArea(0, 0, WIDTH, HEIGHT);
356}
357
358const unsigned char GxEPD2_213_BN::lut_partial[] PROGMEM =
359{
360 0x0, 0x40, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
361 0x80, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
362 0x40, 0x40, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
363 0x0, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
364 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
365 0x0A, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2,
366 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
367 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
368 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
369 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
370 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
371 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
372 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
373 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
374 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
375 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
376 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
377 0x22, 0x22, 0x22, 0x22, 0x22, 0x22, 0x0, 0x0, 0x0,
378};
379
380void GxEPD2_213_BN::_Init_Full()
381{
382 _InitDisplay();
383 _PowerOn();
384 _using_partial_mode = false;
385}
386
387void GxEPD2_213_BN::_Init_Part()
388{
389 _InitDisplay();
390 _writeCommand(0x32);
391 _writeDataPGM(lut_partial, sizeof(lut_partial));
392 _PowerOn();
393 _using_partial_mode = true;
394}
395
396void GxEPD2_213_BN::_Update_Full()
397{
398 _writeCommand(0x22);
399 _writeData(0xf4);
400 _writeCommand(0x20);
401 _waitWhileBusy("_Update_Full", full_refresh_time);
402}
403
404void GxEPD2_213_BN::_Update_Part()
405{
406 _writeCommand(0x22);
407 _writeData(0xcc);
408 _writeCommand(0x20);
409 _waitWhileBusy("_Update_Part", partial_refresh_time);
410}
const unsigned char GxEPD2_213_BN::lut_partial[] PROGMEM
void drawImagePart(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap, int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
void writeImageAgain(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
void writeScreenBuffer(uint8_t value=0xFF)
void writeImage(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
void refresh(bool partial_update_mode=false)
static const uint16_t partial_refresh_time
void writeImagePart(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap, int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
static const uint16_t HEIGHT
void drawNative(const uint8_t *data1, const uint8_t *data2, int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
static const uint16_t power_off_time
void clearScreen(uint8_t value=0xFF)
static const uint16_t power_on_time
GxEPD2_213_BN(int16_t cs, int16_t dc, int16_t rst, int16_t busy)
static const uint16_t full_refresh_time
void writeScreenBufferAgain(uint8_t value=0xFF)
static const uint16_t WIDTH
void writeImageForFullRefresh(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
void writeImagePartAgain(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap, int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
void drawImage(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
void writeNative(const uint8_t *data1, const uint8_t *data2, int16_t x, int16_t y, int16_t w, int16_t h, bool invert=false, bool mirror_y=false, bool pgm=false)
void _writeCommand(uint8_t c)
void _writeData(uint8_t d)
bool _using_partial_mode
Definition GxEPD2_EPD.h:118
void _waitWhileBusy(const char *comment=0, uint16_t busy_time=5000)
bool _power_is_on
Definition GxEPD2_EPD.h:118
bool _initial_refresh
Definition GxEPD2_EPD.h:117
void _reset()
bool _initial_write
Definition GxEPD2_EPD.h:117
int16_t _rst
Definition GxEPD2_EPD.h:112
bool _hibernating
Definition GxEPD2_EPD.h:118
void _writeDataPGM(const uint8_t *data, uint16_t n, int16_t fill_with_zeroes=0)