Embedded System Project 5: ESP32 External Display and PWM

Michelle Lim
11 min readMar 15, 2023

--

Hi, I’m Michelle and today I’ll be continuing my blog series of the embedded systems project! ✨🤩 (You can read the previous blog here 🤗)

The fifth project I’ll be doing is about external display on ESP32. There are various kinds of external displays, such as LED (Light Emitting Diode), 7 Segment LED, LCD (Liquid-Crystal Display), OLED (Organic LED), TFT (Thin-Film Transistor). But on this blog, I’ll be experimenting with the LCD or Liquid-Crystal Display. 😄

LCD (Liquid-Crystal Display) is a type of flat panel display which uses liquid crystals in its primary form of operation. LCDs have a large and varying set of use cases for consumers and businesses, as they can be commonly found in smartphones, televisions, computer monitors, and instrument panels. LCD is used in embedded system as a simple display that is connected directly to the board to print sensor data, simple count down, displaying scrolling text and more.

The LCD I’ll be using on this project is a 16x2 I2C LCD, which I got from the marketplace. The advantage of using an I2C LCD is that the wiring is really simple as we only need to wire the SDA and SCL pins. Additionally, it comes with a built-in potentiometer for us to adjust the contrast between the background and characters on the LCD.

I2C LCD Display

In this project, we will be needing these components listed below

  1. ESP32 Development Board
  2. Micro-USB Cable
  3. Breadboard
  4. 16x2 I2C Liquid Crystal Display (LCD)
  5. Male-to-Female jumper wires
  6. Laptop/PC with Arduino IDE installed and set

Pre-Project: Installing the LiquidCrystal_I2C Library

To use the I2C LCD, we will need to install the LiquidCrystal_I2C Library first. Follow these steps below to install it:

  1. Click here to download the LiquidCrystal_I2C Library and you should have a .zip folder in your Downloads
  2. Unzip the folder and rename the folder to LiquidCrystal_I2C
  3. Move the renamed folder to your Arduino IDE installation libraries folder
  4. Finally, re-open your Arduino IDE

And you should now have the library installed in your Arduino IDE ready to be used 😉

Project: Wiring the LCD to the ESP32

As explained earlier, we’ll be wiring the default I2C pins on the LCD to connect to the ESP32, as shown in the schematics diagram below.

schematic diagram of ESP32 with I2C LCD from randomnerdtutorials.com

The pinout is shown on the diagram below.

Pinout of I2C LCD to ESP32
My schematics for reference

The black wire connects the GND pin in I2C LCD to GND pin in ESP32; The red wire connects the VCC pin in I2C LCD to VIN pin in ESP32; The green wire connects the SDA pin in I2C LCD to GPIO21 in ESP32; The blue wire connects the SCL pin in I2C LCD to GPIO22 in ESP32.

Project: Getting the LCD Address

1. Copy the code block below, compile, and upload it on your Arduino IDE to get the I2C address.

/*********
Rui Santos
Complete project details at https://randomnerdtutorials.com
*********/

#include <Wire.h>

void setup() {
Wire.begin();
Serial.begin(115200);
Serial.println("\nI2C Scanner");
}

void loop() {
byte error, address;
int nDevices;
Serial.println("Scanning...");
nDevices = 0;
for(address = 1; address < 127; address++ ) {
Wire.beginTransmission(address);
error = Wire.endTransmission();
if (error == 0) {
Serial.print("I2C device found at address 0x");
if (address<16) {
Serial.print("0");
}
Serial.println(address,HEX);
nDevices++;
}
else if (error==4) {
Serial.print("Unknow error at address 0x");
if (address<16) {
Serial.print("0");
}
Serial.println(address,HEX);
}
}
if (nDevices == 0) {
Serial.println("No I2C devices found\n");
}
else {
Serial.println("done\n");
}
delay(5000);
}

2. After uploading the code, open the serial monitor at baud rate of 115200. Press the ESP32 EN button and the I2C address should be displayed in the serial monitor.

I2C address found printed in serial monitor

In my case, the address is 0x27. This address will probably be the same for you if you’re using a similar 16x2.

Project: Display Static Text on the LCD

You just need to simply copy the code below, that will display a static text on the first and second row on the LCD, compile and upload it.

/*********
Rui Santos
Complete project details at https://randomnerdtutorials.com
*********/

#include <LiquidCrystal_I2C.h>

// set the LCD number of columns and rows
int lcdColumns = 16;
int lcdRows = 2;

// set LCD address, number of columns and rows
// if you don't know your display address, run an I2C scanner sketch
LiquidCrystal_I2C lcd(0x27, lcdColumns, lcdRows);

void setup(){
// initialize LCD
lcd.init();
// turn on LCD backlight
lcd.backlight();
}

void loop(){
// set cursor to first column, first row
lcd.setCursor(0, 0);
// print message
lcd.print("Hello, World!");
delay(1000);
// clears the display to print new message
lcd.clear();
// set cursor to first column, second row
lcd.setCursor(0,1);
lcd.print("Hello, World!");
delay(1000);
lcd.clear();
}

The code explanation is given on the //comment on the code block above.

After uploading, you will get the value displayed on the LCD like shown in the gif attached below.

Static Text Display on LCD

As commented on the code block, I tried to change the printed value into the LCD by changing the lcd.print() code, and here’s the result.

Static Text Display on LCD; change printed value

Project: Display Scrolling Text on the LCD

You just need to simply copy the code below, that will display a scrolling text containing more than 16 characters on the second row and a static text on the first row of the LCD, compile and upload it.

#include <LiquidCrystal_I2C.h>

// set the LCD number of columns and rows
int lcdColumns = 16;
int lcdRows = 2;

// set LCD address, number of columns and rows
// if you don't know your display address, run an I2C scanner sketch
LiquidCrystal_I2C lcd(0x27, lcdColumns, lcdRows);

String messageStatic = "Michelle Lim";
String messageToScroll = "Embedded System Project 5: External Display with I2C LCD";

// Function to scroll text
// The function acepts the following arguments:
// row: row number where the text will be displayed
// message: message to scroll
// delayTime: delay between each character shifting
// lcdColumns: number of columns of your LCD
void scrollText(int row, String message, int delayTime, int lcdColumns) {
for (int i=0; i < lcdColumns; i++) {
message = " " + message;
}
message = message + " ";
for (int pos = 0; pos < message.length(); pos++) {
lcd.setCursor(0, row);
lcd.print(message.substring(pos, pos + lcdColumns));
delay(delayTime);
}
}

void setup(){
// initialize LCD
lcd.init();
// turn on LCD backlight
lcd.backlight();
}

void loop(){
// set cursor to first column, first row
lcd.setCursor(0, 0);
// print static message
lcd.print(messageStatic);
// print scrolling message
scrollText(1, messageToScroll, 250, lcdColumns);
}

Code explanation: the scrollText(1, messageToScroll , 250, lcdColumns) function is newly created on this code block. The function allows the messageToScroll variable to be displayed in the second row (1 corresponds to the second row), with a delay time of 250 ms. Aside from the scrollText function, the code is explained on the //comment on the code block above.

After uploading, you will get the value displayed on the LCD like shown in the gif attached below.

Scrolling Text Display on LCD; video speed: 2x

Project: Display Custom Characters on the LCD

You just need to simply copy the code below, that will display custom character of heart as many as the number of columns on the first row of the LCD display which is 16 and print an “I LOVE U” on the second row, compile and upload it.

#include <LiquidCrystal_I2C.h>

// set the LCD number of columns and rows
int lcdColumns = 16;
int lcdRows = 2;

// set LCD address, number of columns and rows
// if you don't know your display address, run an I2C scanner sketch
LiquidCrystal_I2C lcd(0x27, lcdColumns, lcdRows);

//generate byte variable for heart
byte heart[8] = {
0b00000,
0b01010,
0b11111,
0b11111,
0b11111,
0b01110,
0b00100,
0b00000
};

void setup(){
// initialize LCD
lcd.init();
// turn on LCD backlight
lcd.backlight();
// create custom character heart
lcd.createChar(0, heart);
}

void loop(){
// set cursor to fourth column, second row; print
lcd.setCursor(3,1);
lcd.print("I LOVE YOU");

// set cursor to first column, first row
for (int i = 0; i < lcdColumns; i++){
lcd.setCursor(i, 0);
//display the character
lcd.write(0);
}
}

The code is explained in the //comment in the code block above. As for the custom character, you can actually generate other characters aside from heart through this link. 🧡

After uploading, you will get the custom character displayed on the LCD like shown in the gif attached below.

Custom Character Display on I2C LCD

Note: A block in a 16x2 LCD with a total of 32 blocks is only able to display a character with 5x8 pixels at most. Hence, you won’t be able to create a customized character/picture with too many details on it as it may require more pixels than the maximum pixel a character can display. If you were to create a detailed picture/character, it is advised to use an OLED display.

Bonus Project: Experiment with ESP32 Pulse Width Modulator

Pulse Width Modulator (PWM) is a technique to obtain an analog signal from a certain digital system. This technique is usable to control the ‘brightness’ of an LED, speed of a DC motor, and servo motor.

The GPIOs in ESP32 outputs a 3.3V (HIGH) or 0V (LOW) to create a square signal wave, with a constant amplitude and frequency. To dim an LED, the digital pin is unable to release voltage between 0 and 3.3V. However, the ‘brightness’ of an LED is controlled by the time when the ON and OFF signal is generated.

There are several terminologies on PWM:

  • T_ON (ON Time), is the time when the signal is HIGH.
  • T_OFF (OFF Time), is the time when the signal is LOW.
  • Period, is the sum of ON time and OFF time.
  • Duty Cycle, is the percentage of time when the signal was high during the time of period.

At 50% duty cycle and 1Hz frequency, the LED will be HIGH for half a second and LOW for the other half second. If the frequency is increased to 50Hz (50 times ON and OFF per second), the LED will be seen glowing at half brightness by the human eye. 🔅👁️

There are various I/O interfaces of the ESP32 board, one of them are the 16 independent channels PWM controller output. the ESP32 PWM has 3 properties that needs to be configured on the code, which are the Channel (total of 16 channels, from 0 to 15), Frequency (maximum of 78125Hz), and Duty Cycle Resolution (maximum of 13 bits). The PWM signals can be set in any GPIO pin of the ESP32 that can act as an output.

FYI: 8-bit resolution means that the duty cycle has a resolution value of 0–255. Resolution value of 0 means that the duty cycle is 0%, while resolution value of 255 means that the duty cycle is 100%. To obtain 50% duty cycle, the resolution is 50% of 255 which is 127.

Bonus Project: A Step by Step Guide to Experiment with ESP32 PWM

For this bonus project, I will be building a simple circuit that dims the LEDs using LED PWM controller of the ESP32.

Here’s the steps you’ll have to follow to dim an LED with PWM using the Arduino IDE:

  1. First, you need to choose a PWM channel. There are 16 channels from 0 to 15.
  2. Second, set the PWM signal frequency. For an LED, a frequency of 5000 Hz is fine to use.
  3. Set the signal duty’s resolution (1 to 16 bits). We’ll use 8-bit resolution to control the LED brightness using a value from 0 to 255.
  4. Specify which GPIO(s) the signal will appear on. We will be using the ledcAttachPin(GPIO, channel) function that accepts two arguments; the GPIO that will output the signal, and channel that will generate the signal.
  5. Use the ledcWrite(channel, dutycycle) function to control the LED brigtness using PWM. The function accepts two arguments; the channel that generates the PWM signal, and the duty cycle.

I will be building a circuit where 3 LEDs will receive the same signal despite being connected to different GPIOs. The components needed are three LEDs, three 330 Ohm resistors, an ESP32 development board, a Micro-USB cable, a breadboard, and jumper wires.

Bonus Project: Schematics

The schematics is shown in the picture below.

Schematics for ESP32 PWM
my schematics on breadboard

Bonus Project: Code

Copy the code attached below, compile and run it on the Arduino IDE.

// the number of the LED pin
const int ledPin = 22; // 22 corresponds to GPIO22
const int ledPin2 = 21; // 21 corresponds to GPIO21
const int ledPin3 = 5; // 5 corresponds to GPIO5

// setting PWM properties
const int freq = 5000;
const int ledChannel = 0;
const int resolution = 8;

void setup(){
// configure LED PWM functionalitites
ledcSetup(ledChannel, freq, resolution);

// attach the channel to the GPIO to be controlled
ledcAttachPin(ledPin, ledChannel);
ledcAttachPin(ledPin2, ledChannel);
ledcAttachPin(ledPin3, ledChannel);
}

void loop(){
// increase the LED brightness
for(int dutyCycle = 0; dutyCycle <= 255; dutyCycle++){
// changing the LED brightness with PWM
ledcWrite(ledChannel, dutyCycle);
delay(15);
}

// decrease the LED brightness
for(int dutyCycle = 255; dutyCycle >= 0; dutyCycle--){
// changing the LED brightness with PWM
ledcWrite(ledChannel, dutyCycle);
delay(15);
}
}

The three ledPins are all getting the same signal, that is being generated on channel 0. This allows us to see three LEDs increasing and decreasing the brightness simultaneously, resulting in a synchronized effect. The code is explained on the //comment on the code block above.

Bonus Project: Result

After uploading, you should see the LEDs dimming and glowing simultaneously as shown on the gif below.

LEDs increasing and decreasing the brightness simultaneously

So yea, that’s the end of our fifth Embedded System Project: ESP32 External Display and PWM. (yeah!! 🥳) Stay tune for the next projects and stay safe and healthy 🥰

--

--