GY-63_MS5611/libraries/I2C_LCD/README.md

300 lines
9.9 KiB
Markdown
Raw Normal View History

2023-12-17 07:01:57 -05:00
[![Arduino CI](https://github.com/RobTillaart/I2C_LCD/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
[![Arduino-lint](https://github.com/RobTillaart/I2C_LCD/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/I2C_LCD/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/I2C_LCD/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/I2C_LCD/actions/workflows/jsoncheck.yml)
[![GitHub issues](https://img.shields.io/github/issues/RobTillaart/I2C_LCD.svg)](https://github.com/RobTillaart/I2C_LCD/issues)
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/I2C_LCD/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/I2C_LCD.svg?maxAge=3600)](https://github.com/RobTillaart/I2C_LCD/releases)
[![PlatformIO Registry](https://badges.registry.platformio.org/packages/robtillaart/library/I2C_LCD.svg)](https://registry.platformio.org/libraries/robtillaart/I2C_LCD)
# I2C_LCD
2023-12-24 02:18:11 -05:00
Arduino library for I2C_LCD LCD displays.
2023-12-20 15:07:32 -05:00
2023-12-17 07:01:57 -05:00
## Description
2023-12-24 02:18:11 -05:00
**Experimental** Arduino library for the I2C LCD display, typical **20x4** characters.
2023-12-17 07:01:57 -05:00
2023-12-20 15:07:32 -05:00
The library is inspired by the excellent LiquidCrystel_I2C library of F. Malpartida.
Therefore the interface is kept quite identical, extended with some additional functions.
2023-12-17 07:01:57 -05:00
The goal of the library is to minimize footprint and improve performance.
This is partly done by dedicate the library to I2C only.
2023-12-20 15:07:32 -05:00
The library implements the print interface, so all data types are printable
assuming they fit on the display.
The reference:
2023-12-17 07:01:57 -05:00
2023-12-20 15:07:32 -05:00
- https://github.com/fmalpartida/New-LiquidCrystal
2023-12-17 07:01:57 -05:00
2023-12-21 13:55:56 -05:00
Additional functions not in the reference
| name | description |
|:----------------------|:-------------|
| isConnected() | test if display address is seen on I2C bus.
| clearEOL() | clears line from current position
| \t | printing of a tab char moves to next multiple of 4.
| special(index) | prints the customized char at index (0..7)
| center(row, str) | centers a char array on chosen row.
| right(col, row, str) | right align a char array from chosen position.
2023-12-17 07:01:57 -05:00
#### Compatibility
2023-12-20 15:07:32 -05:00
Reasonable compatible with F. Malpartida's I2C library, relative minor differences
mostly in the constructors.
Furthermore the current version does not support all functionality, and
more important it is not tested as much as the reference.
Only the 5x8 font is supported.
2023-12-21 13:55:56 -05:00
So if you need full functionality, robustness or reliability, you should use
2023-12-20 15:07:32 -05:00
the reference **New-liquidCrystal** library.
2023-12-17 07:01:57 -05:00
#### Tests
2023-12-24 02:18:11 -05:00
Tested on an UNO and a **20x4** character LCD, and limited on a **16x2** LCD.
The versions 0.1.0 and 0.1.1 were not stable and gave a garbled display.
It looked like the data came too fast for the display to handle.
The problems seems to be solved since version 0.1.2 as the problem did not
occur any more.
2023-12-17 07:01:57 -05:00
## Performance
2023-12-20 15:07:32 -05:00
The most important optimization is to send every byte in a single I2C transaction.
2023-12-17 07:01:57 -05:00
This takes 5 bytes to transport, which is 3 less than the reference.
2023-12-20 15:07:32 -05:00
Furthermore there is an optimization if the pins are in ascending order, as then
it is far easier to get the nibble (half bytes) to send.
2023-12-17 07:01:57 -05:00
First performance tests are good. See example **I2C_LCD_performance.ino**.
2023-12-21 13:55:56 -05:00
The performance measurement is done on an UNO, data pins are in ascending order.
2023-12-20 15:07:32 -05:00
| I2C clock | 0.1.0 | 0.1.1 | notes |
|:-----------:|:-----------:|:-----------:|:-------:|
| 100000 | 4316 | 4640 |
| 200000 | 2440 | 2760 |
| 300000 | 1780 | 2108 |
| 400000 | 1496 | 1820 | (1)
| 500000 | 1308 | 1632 |
| 600000 | 1176 | 1500 |
| 700000 | 1076 | 1400 |
| 800000 | 1024 | 1348 |
Note 1: 0.1.0 problems with spectrum examples - too much data too fast killed my display.
2023-12-17 07:01:57 -05:00
2023-12-21 13:55:56 -05:00
Timing in the 0.1.1 version is roughly 400 us slower than 0.1.0 for 8 characters.
However the 0.1.1 is more robust as far as tested.
2023-12-24 02:18:11 -05:00
In the file **I2C_LCD.cpp** there is this line you can tune the hard delay
in microseconds after every character.
```cpp
// 40 us is a save value at any speed.
// 20 us is a save value for I2C at 400K.
const uint8_t I2C_LCD_CHAR_DELAY = 20;
```
The datasheet states one need the 37 us so 40 us is a very safe value.
However the I2C at 400K takes at least 40 us to send an address and the first 4 bits.
So 20 us is a safe value, and 10 us or even 0 us should work well.
The math above does not include other overhead like preparing the bits etc.
At 100K the I2C for 2 bytes takes 160 us, so it can safely set to 0.
Setting the delay to zero, gives roughly the 0.1.0 timing again.
2023-12-21 13:55:56 -05:00
Note: Performance is also a matter of developing an optimal algorithm.
This is often a trade between code size, memory used and speed.
See **I2C_LCD_demo_spectrum_row.ino** for an example.
2023-12-17 07:01:57 -05:00
#### Related
2023-12-21 13:55:56 -05:00
- https://github.com/fmalpartida/New-LiquidCrystal The reference.
- https://github.com/RobTillaart/ANSI for VT100 alike terminals.
2023-12-17 07:01:57 -05:00
## Interface
```cpp
#include "I2C_LCD.h"
```
#### Constructor
- **I2C_LCD(uint8_t address, TwoWire \* wire = &Wire)** Constructor,
mandatory address and optional alternative I2C bus.
2023-12-20 15:07:32 -05:00
- **void config(uint8_t address, uint8_t enable, uint8_t readWrite, uint8_t registerSelect,
2023-12-17 07:01:57 -05:00
uint8_t data4, uint8_t data5, uint8_t data6, uint8_t data7,
2023-12-21 13:55:56 -05:00
uint8_t backlight, uint8_t polarity)** pin configuration.
2023-12-17 07:01:57 -05:00
Will change in the future.
- **void begin(uint8_t cols = 20, uint8_t rows = 4)** initializes library size.
2023-12-20 15:07:32 -05:00
User must call the appropriate **Wire.begin()** before calling **lcd.begin(()**
- **bool isConnected()** returns true if address can be seen on the I2C bus chosen.
#### Backlight
2023-12-17 07:01:57 -05:00
2023-12-21 13:55:56 -05:00
- **void setBacklightPin(uint8_t pin, uint8_t polarity)** idem.
2023-12-20 15:07:32 -05:00
- **void setBacklight(bool on)** idem.
- **void backlight()** wrapper for setBacklight()
- **void noBacklight()** wrapper for setBacklight()
2023-12-17 07:01:57 -05:00
2023-12-20 15:07:32 -05:00
#### Display
2023-12-17 07:01:57 -05:00
- **void display()** set display on
- **void noDisplay()** set display off
2023-12-20 15:07:32 -05:00
- **void on()** set display on
- **void off()** set display off
#### Positioning and cursor
- **void clear()** clear whole screen and set cursor to 0, 0.
- **void clearEOL()** clears line from current pos. **NEW**
2023-12-17 07:01:57 -05:00
- **void home()** set cursor to 0, 0.
2023-12-20 15:07:32 -05:00
- **bool setCursor(uint8_t col, uint8_t row)** set cursor to given position.
There is a check if this is out of range, if so it return false.
- **void noBlink()** idem.
- **void blink()** idem.
- **void noCursor()** idem.
- **void cursor()** idem.
#### Miscellaneous
Minimal tested.
- **void scrollDisplayLeft(void)**
- **void scrollDisplayRight(void)**
2023-12-24 02:18:11 -05:00
- **void moveCursorRight(uint8_t n = 1)** moves cursor n places to right or
until end of line reached.
- **void moveCursorLeft(uint8_t n = 1)** moves cursor n places to left or
until start of line reached.
2023-12-20 15:07:32 -05:00
The next 4 have only limited support
(either autoscroll or leftright can be set, not both)
- **void autoscroll(void)**
- **void noAutoscroll(void)**
- **void leftToRight(void)**
- **void rightToLeft(void)**
#### Create your own characters
A charmap consists of an array of 8 bytes with values 0..31 (5 bits).
2023-12-21 13:55:56 -05:00
There are 8 slots to place a special character, index 0..7.
The custom characters can be printed with **special(index)** which is
a wrapper around **write((uint8_t)index)**
2023-12-20 15:07:32 -05:00
- **void createChar(uint8_t index, uint8_t \* charmap)**
- **size_t special(uint8_t index)** to print the special char.
2023-12-21 13:55:56 -05:00
See spectrum examples for how to use custom characters.
2023-12-17 07:01:57 -05:00
#### Print interface
- **size_t write(uint8_t c)**
2023-12-21 13:55:56 -05:00
Two helper functions, please note these work only with a char array.
- **size_t center(uint8_t row, char \* message)** centers a string on the defined row.
- **size_t right(uint8_t col, uint8_t row, char \* message)** right align a string.
col is the align position.
2023-12-20 15:07:32 -05:00
## Experimental
2023-12-21 13:55:56 -05:00
#### Tab printing
When '\t' (character 9) is printed to the LCD, it moves to the first position
that is a multiple of 4.
This allows a simple way to get some sort of tabular data representation.
See the example.
Idea for the future might be to set the tab-stops instead of current hardcoded ones.
## Debug
2023-12-20 15:07:32 -05:00
#### Position tracking
The library tries to keep track of the current position on the line it is.
When going beyond the number of columns, it will not print any more.
This protects against printing beyond screen position (and writing
one some other line).
Additional it saves some time.
This position tracking only works for "standard" use, no display
scrolling, right to left mode etc.
So it can cause unwanted side effects, which can be fixed by removing
the "pos < cols" condition from **write()**.
- **uint8_t getColumn()**
2023-12-17 07:01:57 -05:00
2023-12-20 15:07:32 -05:00
The library does not track the row (yet)
2023-12-17 07:01:57 -05:00
2023-12-20 15:07:32 -05:00
2023-12-21 13:55:56 -05:00
#### getWriteCount()
2023-12-20 15:07:32 -05:00
As I encountered problems during tests (display garbled) I added a counter
of the number of writes (each char => 5 bytes I2C).
For now a development only, so expect it to be removed in future.
- **uint32_t getWriteCount()** idem.
2023-12-21 13:55:56 -05:00
Not reset-able.
2023-12-17 07:01:57 -05:00
## Future
#### Must
- update documentation
#### Should
2023-12-24 02:18:11 -05:00
- test, test, test.
- test other platforms.
- test other display sizes.
2023-12-17 07:01:57 -05:00
#### Could
2023-12-20 15:07:32 -05:00
- function to define the tab-stops, instead of hard coded ones.
- make a separate include file for charmaps by name.
2023-12-21 13:55:56 -05:00
- investigate reading busy flag over I2C.
2023-12-24 02:18:11 -05:00
- convenience **repeat(char c, uint8_t times)**
2023-12-17 07:01:57 -05:00
2023-12-20 15:07:32 -05:00
#### Wont for now.
2023-12-17 07:01:57 -05:00
2023-12-24 02:18:11 -05:00
- **size_t write(array, length)** is not implemented as there was no gain.
2023-12-20 15:07:32 -05:00
- implement unit tests (possible?)
- add timestamp last print
2023-12-21 13:55:56 -05:00
- investigate other special characters to support, like
2023-12-24 02:18:11 -05:00
- \r => go to begin of current line.
- \n => go to begin of next line.
2023-12-21 13:55:56 -05:00
- FF => form feed is clear screen.
- BELL => blink of the display (oeps 7 is already a special char )
2023-12-17 07:01:57 -05:00
## Support
If you appreciate my libraries, you can support the development and maintenance.
Improve the quality of the libraries by providing issues and Pull Requests, or
donate through PayPal or GitHub sponsors.
Thank you,