Paperd.Ink Library 0.0.5
Library for interacting with Paperd.Ink devices.
Loading...
Searching...
No Matches
GxEPD2_583c.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: these e-papers require 3.3V supply AND data lines!
3//
4// based on Demo Example from Good Display: http://www.e-paper-display.com/download_list/downloadcategoryid=34&isMode=false.html
5// Controller: IL0371 : http://www.e-paper-display.com/download_detail/downloadsId=536.html
6//
7// Author: Jean-Marc Zingg
8//
9// Version: see library.properties
10//
11// Library: https://github.com/ZinggJM/GxEPD2
12
13#include "GxEPD2_583c.h"
14
15GxEPD2_583c::GxEPD2_583c(int16_t cs, int16_t dc, int16_t rst, int16_t busy) :
16 GxEPD2_EPD(cs, dc, rst, busy, LOW, 40000000, WIDTH, HEIGHT, panel, hasColor, hasPartialUpdate, hasFastPartialUpdate)
17{
18}
19
20void GxEPD2_583c::clearScreen(uint8_t value)
21{
22 _initial_write = false; // initial full screen buffer clean done
23 if (value == 0xFF) value = 0x33; // white value for this controller
24 _Init_Part();
25 _writeCommand(0x91); // partial in
26 _setPartialRamArea(0, 0, WIDTH, HEIGHT);
27 _writeCommand(0x10);
29 for (uint32_t i = 0; i < uint32_t(WIDTH) * uint32_t(HEIGHT) / 2; i++)
30 {
31 _transfer(value);
32 }
34 _Update_Part();
35 _writeCommand(0x92); // partial out
36}
37
38void GxEPD2_583c::clearScreen(uint8_t black_value, uint8_t color_value)
39{
40 _initial_write = false; // initial full screen buffer clean done
41 _Init_Part();
42 _writeCommand(0x91); // partial in
43 _setPartialRamArea(0, 0, WIDTH, HEIGHT);
44 _writeCommand(0x10);
46 for (uint32_t i = 0; i < uint32_t(WIDTH) * uint32_t(HEIGHT) / 8; i++)
47 {
48 _send8pixel(~black_value, ~color_value);
49 }
51 _Update_Part();
52 _writeCommand(0x92); // partial out
53}
54
56{
57 _initial_write = false; // initial full screen buffer clean done
58 if (value == 0xFF) value = 0x33; // white value for this controller
59 _Init_Part();
60 _writeCommand(0x91); // partial in
61 _setPartialRamArea(0, 0, WIDTH, HEIGHT);
62 _writeCommand(0x10);
64 for (uint32_t i = 0; i < uint32_t(WIDTH) * uint32_t(HEIGHT) / 2; i++)
65 {
66 _transfer(value);
67 }
69 _writeCommand(0x92); // partial out
70}
71
72void GxEPD2_583c::writeScreenBuffer(uint8_t black_value, uint8_t color_value)
73{
74 _initial_write = false; // initial full screen buffer clean done
75 _Init_Part();
76 _writeCommand(0x91); // partial in
77 _setPartialRamArea(0, 0, WIDTH, HEIGHT);
78 _writeCommand(0x10);
80 for (uint32_t i = 0; uint32_t(WIDTH) * uint32_t(HEIGHT) / 8; i++)
81 {
82 _send8pixel(~black_value, ~color_value);
83 }
85 _writeCommand(0x92); // partial out
86}
87
88void GxEPD2_583c::writeImage(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
89{
90 writeImage(bitmap, NULL, x, y, w, h, invert, mirror_y, pgm);
91}
92
93void GxEPD2_583c::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)
94{
95 if (_initial_write) writeScreenBuffer(); // initial full screen buffer clean
96 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
97 uint16_t wb = (w + 7) / 8; // width bytes, bitmaps are padded
98 x -= x % 8; // byte boundary
99 w = wb * 8; // byte boundary
100 int16_t x1 = x < 0 ? 0 : x; // limit
101 int16_t y1 = y < 0 ? 0 : y; // limit
102 int16_t w1 = x + w < int16_t(WIDTH) ? w : int16_t(WIDTH) - x; // limit
103 int16_t h1 = y + h < int16_t(HEIGHT) ? h : int16_t(HEIGHT) - y; // limit
104 int16_t dx = x1 - x;
105 int16_t dy = y1 - y;
106 w1 -= dx;
107 h1 -= dy;
108 if ((w1 <= 0) || (h1 <= 0)) return;
109 _Init_Part();
110 _writeCommand(0x91); // partial in
111 _setPartialRamArea(x1, y1, w1, h1);
112 _writeCommand(0x10);
114 for (int16_t i = 0; i < h1; i++)
115 {
116 for (int16_t j = 0; j < w1 / 8; j++)
117 {
118 uint8_t black_data = 0xFF;
119 uint8_t color_data = 0xFF;
120 if (black)
121 {
122 // use wb, h of bitmap for index!
123 uint16_t idx = mirror_y ? j + dx / 8 + uint16_t((h - 1 - (i + dy))) * wb : j + dx / 8 + uint16_t(i + dy) * wb;
124 if (pgm)
125 {
126#if defined(__AVR) || defined(ESP8266) || defined(ESP32)
127 black_data = pgm_read_byte(&black[idx]);
128#else
129 black_data = black[idx];
130#endif
131 }
132 else
133 {
134 black_data = black[idx];
135 }
136 if (invert) black_data = ~black_data;
137 }
138 if (color)
139 {
140 // use wb, h of bitmap for index!
141 uint16_t idx = mirror_y ? j + dx / 8 + uint16_t((h - 1 - (i + dy))) * wb : j + dx / 8 + uint16_t(i + dy) * wb;
142 if (pgm)
143 {
144#if defined(__AVR) || defined(ESP8266) || defined(ESP32)
145 color_data = pgm_read_byte(&color[idx]);
146#else
147 color_data = color[idx];
148#endif
149 }
150 else
151 {
152 color_data = color[idx];
153 }
154 if (invert) color_data = ~color_data;
155 }
156 _send8pixel(~black_data, ~color_data);
157 }
158#if defined(ESP8266)
159 yield();
160#endif
161 }
162 _endTransfer();
163 _writeCommand(0x92); // partial out
164 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
165}
166
167void GxEPD2_583c::writeImagePart(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
168 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
169{
170 writeImagePart(bitmap, NULL, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
171}
172
173void GxEPD2_583c::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,
174 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
175{
176 if (_initial_write) writeScreenBuffer(); // initial full screen buffer clean
177 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
178 if ((w_bitmap < 0) || (h_bitmap < 0) || (w < 0) || (h < 0)) return;
179 if ((x_part < 0) || (x_part >= w_bitmap)) return;
180 if ((y_part < 0) || (y_part >= h_bitmap)) return;
181 uint16_t wb_bitmap = (w_bitmap + 7) / 8; // width bytes, bitmaps are padded
182 x_part -= x_part % 8; // byte boundary
183 w = w_bitmap - x_part < w ? w_bitmap - x_part : w; // limit
184 h = h_bitmap - y_part < h ? h_bitmap - y_part : h; // limit
185 x -= x % 8; // byte boundary
186 w = 8 * ((w + 7) / 8); // byte boundary, bitmaps are padded
187 int16_t x1 = x < 0 ? 0 : x; // limit
188 int16_t y1 = y < 0 ? 0 : y; // limit
189 int16_t w1 = x + w < int16_t(WIDTH) ? w : int16_t(WIDTH) - x; // limit
190 int16_t h1 = y + h < int16_t(HEIGHT) ? h : int16_t(HEIGHT) - y; // limit
191 int16_t dx = x1 - x;
192 int16_t dy = y1 - y;
193 w1 -= dx;
194 h1 -= dy;
195 if ((w1 <= 0) || (h1 <= 0)) return;
196 if (!_using_partial_mode) _Init_Part();
197 _writeCommand(0x91); // partial in
198 _setPartialRamArea(x1, y1, w1, h1);
199 _writeCommand(0x10);
201 for (int16_t i = 0; i < h1; i++)
202 {
203 for (int16_t j = 0; j < w1 / 8; j++)
204 {
205 uint8_t black_data = 0xFF;
206 uint8_t color_data = 0xFF;
207 // use wb_bitmap, h_bitmap of bitmap for index!
208 uint16_t idx = mirror_y ? x_part / 8 + j + dx / 8 + uint16_t((h_bitmap - 1 - (y_part + i + dy))) * wb_bitmap : x_part / 8 + j + dx / 8 + uint16_t(y_part + i + dy) * wb_bitmap;
209 if (black)
210 {
211 if (pgm)
212 {
213#if defined(__AVR) || defined(ESP8266) || defined(ESP32)
214 black_data = pgm_read_byte(&black[idx]);
215#else
216 black_data = black[idx];
217#endif
218 }
219 else
220 {
221 black_data = black[idx];
222 }
223 if (invert) black_data = ~black_data;
224 }
225 if (color)
226 {
227 if (pgm)
228 {
229#if defined(__AVR) || defined(ESP8266) || defined(ESP32)
230 color_data = pgm_read_byte(&color[idx]);
231#else
232 color_data = color[idx];
233#endif
234 }
235 else
236 {
237 color_data = color[idx];
238 }
239 if (invert) color_data = ~color_data;
240 }
241 _send8pixel(~black_data, ~color_data);
242 }
243#if defined(ESP8266)
244 yield();
245#endif
246 }
247 _endTransfer();
248 _writeCommand(0x92); // partial out
249 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
250}
251
252void GxEPD2_583c::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)
253{
254 if (data1)
255 {
256 if (_initial_write) writeScreenBuffer(); // initial full screen buffer clean
257 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
258 uint16_t wb = (w + 1) / 2; // width bytes, bitmaps are padded
259 x -= x % 2; // byte boundary
260 w = wb * 2; // byte boundary
261 int16_t x1 = x < 0 ? 0 : x; // limit
262 int16_t y1 = y < 0 ? 0 : y; // limit
263 int16_t w1 = x + w < int16_t(WIDTH) ? w : int16_t(WIDTH) - x; // limit
264 int16_t h1 = y + h < int16_t(HEIGHT) ? h : int16_t(HEIGHT) - y; // limit
265 int16_t dx = x1 - x;
266 int16_t dy = y1 - y;
267 w1 -= dx;
268 h1 -= dy;
269 if ((w1 <= 0) || (h1 <= 0)) return;
270 _Init_Part();
271 _writeCommand(0x91); // partial in
272 _setPartialRamArea(x1, y1, w1, h1);
273 _writeCommand(0x10);
275 for (int16_t i = 0; i < h1; i++)
276 {
277 for (int16_t j = 0; j < w1 / 2; j++)
278 {
279 uint8_t data;
280 // use wb, h of bitmap for index!
281 uint16_t idx = mirror_y ? j + dx / 2 + uint16_t((h - 1 - (i + dy))) * wb : j + dx / 2 + uint16_t(i + dy) * wb;
282 if (pgm)
283 {
284#if defined(__AVR) || defined(ESP8266) || defined(ESP32)
285 data = pgm_read_byte(&data1[idx]);
286#else
287 data = data1[idx];
288#endif
289 }
290 else
291 {
292 data = data1[idx];
293 }
294 if (invert) data = ~data;
295 _transfer(data);
296 }
297 }
298 _endTransfer();
299 _writeCommand(0x92); // partial out
300 delay(1); // yield() to avoid WDT on ESP8266 and ESP32
301 }
302}
303
304void GxEPD2_583c::drawImage(const uint8_t bitmap[], int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
305{
306 writeImage(bitmap, x, y, w, h, invert, mirror_y, pgm);
307 refresh(x, y, w, h);
308}
309
310void GxEPD2_583c::drawImagePart(const uint8_t bitmap[], int16_t x_part, int16_t y_part, int16_t w_bitmap, int16_t h_bitmap,
311 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
312{
313 writeImagePart(bitmap, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
314 refresh(x, y, w, h);
315}
316
317void GxEPD2_583c::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)
318{
319 writeImage(black, color, x, y, w, h, invert, mirror_y, pgm);
320 refresh(x, y, w, h);
321}
322
323void GxEPD2_583c::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,
324 int16_t x, int16_t y, int16_t w, int16_t h, bool invert, bool mirror_y, bool pgm)
325{
326 writeImagePart(black, color, x_part, y_part, w_bitmap, h_bitmap, x, y, w, h, invert, mirror_y, pgm);
327 refresh(x, y, w, h);
328}
329
330void GxEPD2_583c::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)
331{
332 writeNative(data1, data2, x, y, w, h, invert, mirror_y, pgm);
333 refresh(x, y, w, h);
334}
335
336void GxEPD2_583c::refresh(bool partial_update_mode)
337{
338 if (partial_update_mode) refresh(0, 0, WIDTH, HEIGHT);
339 else _Update_Full();
340}
341
342void GxEPD2_583c::refresh(int16_t x, int16_t y, int16_t w, int16_t h)
343{
344 // intersection with screen
345 int16_t w1 = x < 0 ? w + x : w; // reduce
346 int16_t h1 = y < 0 ? h + y : h; // reduce
347 int16_t x1 = x < 0 ? 0 : x; // limit
348 int16_t y1 = y < 0 ? 0 : y; // limit
349 w1 = x1 + w1 < int16_t(WIDTH) ? w1 : int16_t(WIDTH) - x1; // limit
350 h1 = y1 + h1 < int16_t(HEIGHT) ? h1 : int16_t(HEIGHT) - y1; // limit
351 if ((w1 <= 0) || (h1 <= 0)) return;
352 // make x1, w1 multiple of 8
353 w1 += x1 % 8;
354 if (w1 % 8 > 0) w1 += 8 - w1 % 8;
355 x1 -= x1 % 8;
356 _Init_Part();
357 _setPartialRamArea(x1, y1, w1, h1);
358 _Update_Part();
359}
360
362{
363 _PowerOff();
364}
365
367{
368 _PowerOff();
369 if (_rst >= 0)
370 {
371 // check if it supports this command!
372 _writeCommand(0x07); // deep sleep
373 _writeData(0xA5); // check code
374 _hibernating = true;
375 }
376}
377
378void GxEPD2_583c::_send8pixel(uint8_t black_data, uint8_t color_data)
379{
380 for (uint8_t j = 0; j < 8; j++)
381 {
382 uint8_t t = 0x00; // black
383 if (black_data & 0x80); // keep black
384 else if (color_data & 0x80) t = 0x04; //color
385 else t = 0x03; // white
386 t <<= 4;
387 black_data <<= 1;
388 color_data <<= 1;
389 j++;
390 if (black_data & 0x80); // keep black
391 else if (color_data & 0x80) t |= 0x04; //color
392 else t |= 0x03; // white
393 black_data <<= 1;
394 color_data <<= 1;
395 _transfer(t);
396 }
397}
398
399void GxEPD2_583c::_setPartialRamArea(uint16_t x, uint16_t y, uint16_t w, uint16_t h)
400{
401 uint16_t xe = (x + w - 1) | 0x0007; // byte boundary inclusive (last byte)
402 uint16_t ye = y + h - 1;
403 x &= 0xFFF8; // byte boundary
404 xe |= 0x0007; // byte boundary
405 _writeCommand(0x90); // partial window
406 _writeData(x / 256);
407 _writeData(x % 256);
408 _writeData(xe / 256);
409 _writeData(xe % 256);
410 _writeData(y / 256);
411 _writeData(y % 256);
412 _writeData(ye / 256);
413 _writeData(ye % 256);
414 //_writeData(0x01); // distortion on full right half
415 _writeData(0x00); // distortion on right half
416}
417
418void GxEPD2_583c::_PowerOn()
419{
420 if (!_power_is_on)
421 {
422 _writeCommand(0x04);
423 _waitWhileBusy("_PowerOn", power_on_time);
424 }
425 _power_is_on = true;
426}
427
428void GxEPD2_583c::_PowerOff()
429{
430 _writeCommand(0x02); // power off
431 _waitWhileBusy("_PowerOff", power_off_time);
432 _power_is_on = false;
433}
434
435void GxEPD2_583c::_InitDisplay()
436{
437 if (_hibernating) _reset();
438 _writeCommand(0x01);
439 _writeData (0x37); //POWER SETTING
440 _writeData (0x00);
441 _writeCommand(0X00); //PANNEL SETTING
442 _writeData(0xCF);
443 _writeData(0x08);
444 _writeCommand(0x06); //boost
445 _writeData (0xc7);
446 _writeData (0xcc);
447 _writeData (0x28);
448 _writeCommand(0x30); //PLL setting
449 //_writeData (0x3a); //PLL: 0-15��:0x3C, 15+:0x3A
450 _writeData (0x3c); //PLL: 0-15��:0x3C, 15+:0x3A
451 _writeCommand(0X41); //TEMPERATURE SETTING
452 _writeData(0x00);
453 _writeCommand(0X50); //VCOM AND DATA INTERVAL SETTING
454 _writeData(0x77);
455 _writeCommand(0X60); //TCON SETTING
456 _writeData(0x22);
457 _writeCommand(0x61); //600*448
458 _writeData (0x02); //source 600
459 _writeData (0x58);
460 _writeData (0x01); //gate 448
461 _writeData (0xc0);
462 _writeCommand(0X82); //VCOM VOLTAGE SETTING
463 _writeData(0x28); //all temperature range
464 _writeCommand(0xe5); //FLASH MODE
465 _writeData(0x03);
466}
467
468void GxEPD2_583c::_Init_Full()
469{
470 _InitDisplay();
471 _PowerOn();
472}
473
474void GxEPD2_583c::_Init_Part()
475{
476 _InitDisplay();
477 _PowerOn();
478}
479
480void GxEPD2_583c::_Update_Full()
481{
482 _writeCommand(0x12); //display refresh
483 _waitWhileBusy("_Update_Full", full_refresh_time);
484}
485
486void GxEPD2_583c::_Update_Part()
487{
488 _writeCommand(0x12); //display refresh
489 _waitWhileBusy("_Update_Part", partial_refresh_time);
490}
static const uint16_t HEIGHT
Definition GxEPD2_583c.h:24
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)
static const uint16_t WIDTH
Definition GxEPD2_583c.h:22
static const uint16_t partial_refresh_time
Definition GxEPD2_583c.h:32
static const uint16_t full_refresh_time
Definition GxEPD2_583c.h:31
void clearScreen(uint8_t value=0x33)
GxEPD2_583c(int16_t cs, int16_t dc, int16_t rst, int16_t busy)
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 refresh(bool partial_update_mode=false)
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)
void writeScreenBuffer(uint8_t value=0x33)
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 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)
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)
static const uint16_t power_on_time
Definition GxEPD2_583c.h:29
static const uint16_t power_off_time
Definition GxEPD2_583c.h:30
void _endTransfer()
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
void _startTransfer()
void _transfer(uint8_t value)
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