-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathSH1106Lib.h
744 lines (634 loc) · 19.8 KB
/
SH1106Lib.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
/***************************************************************************
This is a library for the SH1106 I2C OLED display
These displays use I2C to communicate, 2 pins are required to interface
The implementation needed to go here instead of a .cpp file, because that
way the configuration #defines would not work!
See: https://github.com/arduino/Arduino/pull/1808#issuecomment-370210551
Written by https://github.com/notisrac
MIT license, all text above must be included in any redistribution
***************************************************************************/
#pragma once
#ifndef _SH1106LIB_H
#define _SH1106LIB_H
#include <Arduino.h>
#include <TinyPrint.h>
#include <SoftI2CMaster.h>
// region display_constants
#define SH1106_I2C_ADDRESS 0x3C // 011110+SA0+RW - 0x3C or 0x3D
#define SH1106_LCDWIDTH 128
#define SH1106_LCDHEIGHT 64
#define SH1106_SETCONTRAST 0x81
#define SH1106_DISPLAYALLON_RESUME 0xA4
#define SH1106_DISPLAYALLON 0xA5
#define SH1106_NORMALDISPLAY 0xA6
#define SH1106_INVERTDISPLAY 0xA7
#define SH1106_DISPLAYOFF 0xAE
#define SH1106_DISPLAYON 0xAF
#define SH1106_SETDISPLAYOFFSET 0xD3
#define SH1106_SETCOMPINS 0xDA
#define SH1106_SETVCOMDETECT 0xDB
#define SH1106_SETDISPLAYCLOCKDIV 0xD5
#define SH1106_SETPRECHARGE 0xD9
#define SH1106_SETMULTIPLEX 0xA8
#define SH1106_SETCOLUMNADDRLOW 0x00
#define SH1106_SETCOLUMNADDRHIGH 0x10
#define SH1106_SETSTARTLINE 0x40
#define SH1106_MEMORYMODE 0x20
//#define SH1106_COLUMNADDR 0x21
#define SH1106_PAGEADDR 0xB0
#define SH1106_COMSCANINC 0xC0
#define SH1106_COMSCANDEC 0xC8
#define SH1106_SEGREMAP 0xA0
#define SH1106_CHARGEPUMP 0x8D
#define SH1106_EXTERNALVCC 0x1
#define SH1106_SWITCHCAPVCC 0x2
#define SH1106_MESSAGETYPE_COMMAND 0x80
#define SH1106_MESSAGETYPE_DATA 0x40
#define SH1106_READMODIFYWRITE_START 0xE0
#define SH1106_READMODIFYWRITE_END 0xEE
#define SH1106_MAXSEGMENTSPERWRITE 16
#define SH1106_ROWHEIGHT 8
#define SH1106_NUMBEROF_SEGMENTS 128
#define SH1106_NUMBEROF_PAGES 8
// endregion display_constants
#define BLACK 0
#define WHITE 1
#define TRANSPARENT 255
#define SOLID 127
#define I2CREAD 1
#define I2CWRITE 0
// font flags
#define FONT_NUMBERS 1
#define FONT_UPPERCASECHARS 2
#define FONT_LOWERCASECHARS 4
#define FONT_HASSPACE 8
#define FONT_FULL 255
class SH1106Lib : public TinyPrint
{
public:
SH1106Lib();
~SH1106Lib();
void initialize();
void sendCommand(byte c);
void sendData(byte data);
void clearDisplay(void);
void fillRect(uint8_t left, uint8_t top, uint8_t width, uint8_t height, uint8_t color);
void drawBitmap(uint8_t x, uint8_t y, const byte *bitmap, uint8_t w, uint8_t h, uint8_t color, uint8_t backgroundType);
void drawPixel(uint8_t x, uint8_t y, uint8_t color);
void setCursor(uint8_t x, uint8_t y);
void setFont(const unsigned char *font, uint8_t width, uint8_t height, int8_t offset = 0, uint8_t flags = FONT_FULL);
void setTextWrap(bool enableWrap);
void setTextColor(uint8_t color, uint8_t backgroundType);
// Inherited via TinyPrint
virtual byte write(uint8_t) override;
void drawChar(uint8_t x, uint8_t y, uint8_t character, uint8_t color, uint8_t backgroundType);
private:
void _setDisplayWritePosition(uint8_t x, uint8_t y);
void _beginTransmission(byte operation = I2CWRITE, bool startNewTransmission = false);
void _endTransmission();
/**
Moves the display write position, and starts the ReadModifyWrite mode
@param x The x coordinate 0..127 (this will select column)
@param y The y coordinate 0..63 (this will select the page - will be converted to page number)
@returns nothing
*/
void _startRMWMode(uint8_t x, uint8_t y);
/**
Sopts the ReadModifyWrite mode
@returns nothing
*/
void _stopRMWMode();
/**
Draws a single column (8bits vertically) on the current page.
Note: the ReadModifyWrite must be enabled for this to work
@param data The byte that will be output to the column
@param color The color that the 1 pixels is displayed should be black or whit - ANDs or ORs the data to the pixels on the screen
@param backgroundType The type of the background. Valid values: SOLID, TRANSPARENT
@param backgroundMask A byte mask to specify which bits are actually relevant in the data
@returns nothing
*/
void _drawColumn(uint8_t data, uint8_t color, uint8_t backgroundType = TRANSPARENT, byte backgroundMask = B00000000);
/**
Draws a number of columns (same 8bits vertically) on the current page.
Note: the ReadModifyWrite must be enabled for this to work
@param x The x coordinate 0..127 (this will select column)
@param y The y coordinate 0..63 (this will select the page - will be converted to page number)
@param data The byte that will be output to the column
@param count How many columns should be filled with the data
@param color The color that the 1 pixels is displayed should be black or whit - ANDs or ORs the data to the pixels on the screen
@returns nothing
*/
void _drawColumns(uint8_t x, uint8_t y, uint8_t data, uint8_t count, uint8_t color);
// text related variables
uint8_t _cursorX = 0;
uint8_t _cursorY = 0;
uint8_t _textColor = WHITE;
uint8_t _backgroundType = TRANSPARENT;
uint8_t _fontWidth = 5;
uint8_t _fontHeight = 8;
int8_t _fontOffset = 0;
uint8_t _fontFlags = FONT_FULL;
bool _wrap = true;
const unsigned char* _font;
uint8_t _pixelPosX = 0;
uint8_t _pixelPosY = 0;
// i2c vars
bool _i2cTransmissionInProgress = false;
};
#define intCeil(x,y) (((x) + (y) - 1) / (y))
SH1106Lib::SH1106Lib()
{
}
SH1106Lib::~SH1106Lib()
{
}
/*
Initializes the display
*/
void SH1106Lib::initialize()
{
if (!i2c_init()) { // Initialize everything and check for bus lockup
//Serial.println("I2C init failed");
}
//TWBR = 12; // upgrade i2c speed to 400KHz!
_beginTransmission(I2CWRITE, true);
sendCommand(SH1106_DISPLAYOFF); // 0xAE
sendCommand(SH1106_SETDISPLAYCLOCKDIV); // 0xD5
sendCommand(0x80); // the suggested ratio 0x80
sendCommand(SH1106_SETMULTIPLEX); // 0xA8
sendCommand(0x3F);
sendCommand(SH1106_SETDISPLAYOFFSET); // 0xD3
sendCommand(0x00); // no offset
sendCommand(SH1106_SETSTARTLINE | 0x0); // line #0 0x40
sendCommand(SH1106_CHARGEPUMP); // 0x8D
// SH1106_EXTERNALVCC
sendCommand(0x10);
sendCommand(SH1106_MEMORYMODE); // 0x20
sendCommand(0x00); // 0x0 act like ks0108
sendCommand(SH1106_SEGREMAP | 0x1);
sendCommand(SH1106_COMSCANDEC);
sendCommand(SH1106_SETCOMPINS); // 0xDA
sendCommand(0x12);
sendCommand(SH1106_SETCONTRAST); // 0x81
// SH1106_EXTERNALVCC
sendCommand(0x9F);
sendCommand(SH1106_SETPRECHARGE); // 0xd9
// SH1106_EXTERNALVCC
sendCommand(0x22);
sendCommand(SH1106_SETVCOMDETECT); // 0xDB
sendCommand(0x40);
sendCommand(SH1106_DISPLAYALLON_RESUME); // 0xA4
sendCommand(SH1106_NORMALDISPLAY); // 0xA6
sendCommand(SH1106_DISPLAYON); //--turn on oled panel
// ?????
sendCommand(SH1106_SETCOLUMNADDRLOW | 0x0); // low col = 0
sendCommand(SH1106_SETCOLUMNADDRHIGH | 0x0); // hi col = 0
sendCommand(SH1106_SETSTARTLINE | 0x0); // line #0
_endTransmission();
}
/*
Sends a single command to the display
command: the command (byte) that needs to be sent to the device
*/
void SH1106Lib::sendCommand(byte command)
{
// I2C
_beginTransmission(I2CWRITE, false);
i2c_write(SH1106_MESSAGETYPE_COMMAND); // This is a Command
i2c_write(command);
//if (useOwnTransmission)
//{
// _endTransmission();
//}
}
/*
Sends a single byte of data to the display
data: the data (byte) that needs to be sent to the device
*/
void SH1106Lib::sendData(byte data)
{
// I2C
_beginTransmission(I2CWRITE, false);
i2c_write(SH1106_MESSAGETYPE_DATA); // This byte is DATA
i2c_write(data);
//if (useOwnTransmission)
//{
// _endTransmission();
//}
}
/*
Clears the display
*/
void SH1106Lib::clearDisplay(void)
{
uint8_t page, segments, parts;
//byte buffer[16] = { 0x00 };
_beginTransmission(I2CWRITE, true);
// clear the buffer so we can fill the screen with zeroes
for (page = 0; page < SH1106_NUMBEROF_PAGES; page++)
{
// move to the beginning of the next page
_setDisplayWritePosition(0, page * SH1106_ROWHEIGHT);
for (segments = 0; segments < (SH1106_NUMBEROF_SEGMENTS / SH1106_MAXSEGMENTSPERWRITE); segments++)
{
// no need to set the draw position, as every memory write advances the write pos to the next one
for (parts = 0; parts < SH1106_MAXSEGMENTSPERWRITE; parts++)
{
sendData(0x00);
}
//sendData(buffer, SH1106_MAXSEGMENTSPERWRITE, false);
}
}
_endTransmission();
}
//unsigned char reverse(unsigned char b) {
// b = (b & 0xF0) >> 4 | (b & 0x0F) << 4;
// b = (b & 0xCC) >> 2 | (b & 0x33) << 2;
// b = (b & 0xAA) >> 1 | (b & 0x55) << 1;
// return b;
//}
//
//char* printBits(byte myByte) {
// static char str[9];
// byte mask = 0x80;
//
// for (uint8_t i = 0; i < 8; i++)
// {
// if (mask & myByte)
// {
// str[i] = '1';
// }
// else
// {
// str[i] = '0';
// }
// mask >>= 1;
// }
// str[8] = '\0';
//
// return str;
//}
/*
Displays a filled rectangle
left: x coordinate of the top left corner of the rectangle
top: y coordinate of the top left corner of the rectangle
width: width of the rectangle
height: height of the rectangle
color: color of the rectangle
*/
void SH1106Lib::fillRect(uint8_t left, uint8_t top, uint8_t width, uint8_t height, uint8_t color)
{
// sanity check
if (left > SH1106_LCDWIDTH || top > SH1106_LCDHEIGHT)
{
return;
}
if (left + width > SH1106_LCDWIDTH)
{
width = SH1106_LCDWIDTH - left;
}
if (top + height > SH1106_LCDHEIGHT)
{
height = SH1106_LCDHEIGHT - top - 1;
}
uint8_t i = 0;
byte maskStart, maskEnd = B00000000;
// calculate the start and the end masks
maskStart = B11111111 << ((top % 8));
maskEnd = ~(B11111111 << (((top + height) % 8) + 1));
if (height <= 8 && (floor(top / 8) == floor((top + height) / 8)))
{ // smaller than a page, so combine the start and the end mask
maskStart = maskStart & maskEnd;
}
else
{
// write the end mask to the last page it touches
_drawColumns(left, top + height, maskEnd, width, color);
}
// write the start mask
_drawColumns(left, top, maskStart, width, color);
if (height > 8)
{ // larger than 8
// loop through the inner parts
for (i = top + (8 - (top % 8)); i < top + height - ((top + height) % 8); i += 8)
{
_drawColumns(left, i, B11111111, width, color);
}
}
}
/*
Draws a bitmap from the program memory to the display
x: the x coordinate to put the bitmap
y: the y coordiante to put the bitmap
bitmap: the byte array representing the image
w: the width of the image
h: the height of the image
color: the color which the image should be displayed (BLACK/WHITE)
backgroundType: SOLID or TRANSPARENT background
*/
void SH1106Lib::drawBitmap(uint8_t x, uint8_t y, const byte * bitmap, uint8_t w, uint8_t h, uint8_t color, uint8_t backgroundType = SOLID)
{
if (w > SH1106_LCDWIDTH || h > SH1106_LCDHEIGHT)
{ // sanity check
return;
}
uint8_t i, j, n, diff, yActual;
uint8_t actualByte = 0x00;
//uint8_t startSize = 8 - (y % 8);
//uint8_t endSize = (y + h) % 8;
// height / 8, because we will plot in columns with the height of 8
for (j = 0; j < intCeil(h, 8) /*ceil(h / 8.0)*/; j++) {
// calculate the start pos
diff = (y + j * 8) % 8;
for (n = 0; n < ((0 == diff) ? 1 : 2); n++) // if it starts on the page border, then we can do it in one run
{
yActual = y + ((j + n) * 8);
if (yActual > SH1106_LCDHEIGHT)
{ // don't try to write outside the display area
continue;
}
_startRMWMode(x, y + ((j + n) * 8));
// loop through the width of the image, and plot the columns
for (i = 0; i < w; i++) {
if ((x + i) > SH1106_LCDWIDTH)
{ // don't try to write outside the display area
continue;
}
actualByte = pgm_read_byte(bitmap + (i + (j * w)));
if (0 == n)
{ // this is the below the page barrier
actualByte = actualByte << diff;
}
else
{ // this is the leftover - this only comes in play when the current part of the image crosses the page boundary
actualByte = actualByte >> (8 - diff);
}
// display the column of pixels
_drawColumn(actualByte, color, backgroundType, B11111111);
}
_stopRMWMode();
}
}
}
/*
Draws a single pixel on th display
x: the x coordinate to put the pixel
y: the y coordiante to put the pixel
color: the color of the pixel
*/
void SH1106Lib::drawPixel(uint8_t x, uint8_t y, uint8_t color)
{
if (x < 0 || x > SH1106_LCDWIDTH
|| y < 0 || y > SH1106_LCDHEIGHT)
{
return;
}
_drawColumns(x, y, 1 << (y & 7), 1, color);
}
/*
Sets the text cursor to this position
x: the x coordinate
y: the y coordiante
*/
void SH1106Lib::setCursor(uint8_t x, uint8_t y)
{
_cursorX = x;
_cursorY = y;
}
/*
Sets the font to write with
font: pinter to the array containing the font
width: the width of the font in pixels
height: the height of the font in pixels
offset: signed value to offset the position the character is found in the font
flags: set of flags describing the properties of the font
*/
void SH1106Lib::setFont(const unsigned char *font, uint8_t width, uint8_t height, int8_t offset/* = 0*/, uint8_t flags/* = FONT_FULL*/)
{
_font = font;
_fontWidth = width;
_fontHeight = height;
_fontOffset = offset;
_fontFlags = flags;
}
/*
Sets whether the text should continue on the next row, if it has reached the end of the current one
enableWrap: true to enable, false to disable
*/
void SH1106Lib::setTextWrap(bool enableWrap)
{
_wrap = enableWrap;
}
/*
Sets whether the text should continue on the next row, if it has reached the end of the current one
color: the color of the text
backColor: the color of the background of the text. Set it to TRANSPARENT to have a transparend background
*/
void SH1106Lib::setTextColor(uint8_t color, uint8_t backgroundType)
{
_textColor = color;
_backgroundType = backgroundType;
}
/*
Displays a single character
c: the character to display
returns: 1
*/
byte SH1106Lib::write(uint8_t c)
{
if (c == '\n') { // on a linebreak move the cursor down one line, and back to the start
setCursor(0, _cursorY + _fontHeight);
}
else if (c == '\r') {
// skip carrage return, as we already handle newline
}
else {
// draw the pixels for the actual character
drawChar(_cursorX, _cursorY, c, _textColor, _backgroundType);
// advance the cursor
setCursor(_cursorX + _fontWidth + 1, _cursorY);
// if the next letter would not fit in the current row, jump to the beginnign of the next one
if (_wrap && (_cursorX > (SH1106_LCDWIDTH - _fontWidth + 1))) {
//Serial.print();
setCursor(0, _cursorY + _fontHeight + 1);
}
}
return 1;
}
/*
Draws a character on the screen from the font
x: x coordinate where the character should be displayed
y: y coordinate where the character should be displayed
character: The character to display
color: the color of the character
backgroundType: SOLID or TRANSPARENT
*/
void SH1106Lib::drawChar(uint8_t x, uint8_t y, uint8_t character, uint8_t color, uint8_t backgroundType)
{
if ((x + _fontWidth >= SH1106_LCDWIDTH) || // Clip right
(y + _fontHeight >= SH1106_LCDHEIGHT) || // Clip bottom
((x + _fontWidth + 1) < 0) || // Clip left
((y + _fontHeight) < 0)) // Clip top
return;
uint8_t i, j, n;
uint8_t diff, yActual, actualByte;
uint8_t byteHeight = intCeil(_fontHeight, 8) /*ceil(_fontHeight / 8.0)*/;
// try to mod the character, if the font does not have the required case
//if (isLowerCase(character))
//{
// if ((_fontFlags & FONT_LOWERCASECHARS) != FONT_LOWERCASECHARS)
// {
// character = toUpperCase(character);
// }
//}
//else
//{
// if ((_fontFlags & FONT_UPPERCASECHARS) != FONT_UPPERCASECHARS)
// {
// character = toLowerCase(character);
// }
//}
// height / 8, because we will plot in columns with the height of 8
for (j = 0; j < byteHeight; j++) {
// calculate the start pos
diff = (y + j * 8) % 8;
for (n = 0; n < ((0 == diff) ? 1 : 2); n++) // if it starts on the page border, then we can do it in one run
{
yActual = y + ((j + n) * 8);
if (yActual > SH1106_LCDHEIGHT)
{ // don't try to write outside the display area
continue;
}
_startRMWMode(x, y + ((j + n) * 8));
// loop through the width of the image, and plot the columns
for (i = 0; i < _fontWidth; i++) {
if ((x + i) > SH1106_LCDWIDTH)
{ // don't try to write outside the display area
continue;
}
if (i == _fontWidth || (character == ' ' && ((_fontFlags & FONT_HASSPACE) != FONT_HASSPACE))) {
actualByte = 0x00;
}
else {
actualByte = pgm_read_byte(_font + ((character - _fontOffset) * _fontWidth + i) * byteHeight + j);
}
if (0 == n)
{ // this is the below the page barrier
actualByte = actualByte << diff;
}
else
{ // this is the leftover - this only comes in play when the current part of the image crosses the page boundary
actualByte = actualByte >> (8 - diff);
}
// display the column of pixels
_drawColumn(actualByte, color, backgroundType, B11111111);
}
_stopRMWMode();
}
}
}
/*
Draws a character on the screen from the font
x: the x coordinate
y: the y coordinate
*/
void SH1106Lib::_setDisplayWritePosition(uint8_t x, uint8_t y)
{
if (x == _pixelPosX && ((y >> 3) == _pixelPosY))
{ // should not try to set on the same position again
return;
}
_pixelPosX = x;
_pixelPosY = y >> 3;
// the SH1106 display starts at x = 2! (there are two columns of off screen pixels)
x += 2;
//set page address - there are 8 pages (B0H-B7H), each loop sets the next page - THIS IS THE Y COORD IN 8 INCREMENTS, 0->7
sendCommand(SH1106_PAGEADDR + (y >> 3));
// set lower column address (00H - 0FH) => need the upper half only - THIS IS THE X, 0->127
sendCommand((x & 0x0F));
// set higher column address (10H - 1FH) => 0x10 | (2 >> 4) = 10
sendCommand(SH1106_SETCOLUMNADDRHIGH + (x >> 4));
}
void SH1106Lib::_beginTransmission(byte operation/* = I2CWRITE*/, bool startNewTransmission/* = false*/)
{
if (_i2cTransmissionInProgress && !startNewTransmission)
{ // send a repeated start if there is no need to start a new one
if (!i2c_rep_start((SH1106_I2C_ADDRESS << 1) | operation))
{
//Serial.println("I2C device busy");
return;
}
}
else
{
if (_i2cTransmissionInProgress)
{ // stop the current transmission if there is one
_endTransmission();
}
if (!i2c_start((SH1106_I2C_ADDRESS << 1) | operation)) { // start transfer
//Serial.println("I2C device busy");
return;
}
}
_i2cTransmissionInProgress = true;
}
void SH1106Lib::_endTransmission()
{
i2c_stop();
_i2cTransmissionInProgress = false;
}
void SH1106Lib::_startRMWMode(uint8_t x, uint8_t y)
{
_beginTransmission(I2CWRITE, true);
// set the position and enable the readmodifywrite mode
_setDisplayWritePosition(x, y);
sendCommand(SH1106_READMODIFYWRITE_START);
i2c_write(0xC0); // ????
}
void SH1106Lib::_stopRMWMode()
{
// end the readmodifywrite mode
sendCommand(SH1106_READMODIFYWRITE_END);
_endTransmission();
}
void SH1106Lib::_drawColumn(uint8_t data, uint8_t color, uint8_t backgroundType/* = TRANSPARENT*/, byte backgroundMask/* = B00000000*/)
{
byte b = data;
// read the pixel data from the display
_beginTransmission(I2CREAD, false); // restart in read mode
i2c_read(false); // dummy read
b = i2c_read(true);
// handle the solid/transparent background
if (SOLID == backgroundType)
{
if (BLACK == color)
{ // solid white background
b |= backgroundMask;
}
else
{ // solid black background
b &= ~backgroundMask;
}
}
// modify the read byte with the data, based on the color
if (WHITE == color)
{
b |= data;
}
else
{
b &= ~data;
}
// write the modified data back
sendData(b);
}
void SH1106Lib::_drawColumns(uint8_t x, uint8_t y, uint8_t data, uint8_t count, uint8_t color)
{
_startRMWMode(x, y);
for (uint8_t i = 0; i < count; i++)
{
_drawColumn(data, color);
}
_stopRMWMode();
}
#endif // _SH1106LIB_H