Multi Custom character
I have created the Ferris image using 6 adjacent grids with the generator from the previous page. Here’s the Rust code to utilize those byte arrays.

Generated Byte array for the characters
#![allow(unused)] fn main() { const SYMBOL1: [u8; 8] = [ 0b00110, 0b01000, 0b01110, 0b01000, 0b00100, 0b00011, 0b00100, 0b01000, ]; const SYMBOL2: [u8; 8] = [ 0b00000, 0b00000, 0b00000, 0b10001, 0b10001, 0b11111, 0b00000, 0b00000, ]; const SYMBOL3: [u8; 8] = [ 0b01100, 0b00010, 0b01110, 0b00010, 0b00100, 0b11000, 0b00100, 0b00010, ]; const SYMBOL4: [u8; 8] = [ 0b01000, 0b01000, 0b00100, 0b00011, 0b00001, 0b00010, 0b00101, 0b01000, ]; const SYMBOL5: [u8; 8] = [ 0b00000, 0b00000, 0b00000, 0b11111, 0b01010, 0b10001, 0b00000, 0b00000, ]; const SYMBOL6: [u8; 8] = [ 0b00010, 0b00010, 0b00100, 0b11000, 0b10000, 0b01000, 0b10100, 0b00010, ]; }
Declare them as character
#![allow(unused)] fn main() { lcd.custom_char(&mut Delay, &SYMBOL1, 0); lcd.custom_char(&mut Delay, &SYMBOL2, 1); lcd.custom_char(&mut Delay, &SYMBOL3, 2); lcd.custom_char(&mut Delay, &SYMBOL4, 3); lcd.custom_char(&mut Delay, &SYMBOL5, 4); lcd.custom_char(&mut Delay, &SYMBOL6, 5); }
Display
Let’s write the first 3 grids into the first row, then the second half into the second row of the LCD display.
#![allow(unused)] fn main() { lcd.set_cursor(&mut Delay, 0, 4) .write(&mut Delay, CustomChar(0)) .write(&mut Delay, CustomChar(1)) .write(&mut Delay, CustomChar(2)); lcd.set_cursor(&mut Delay, 1, 4) .write(&mut Delay, CustomChar(3)) .write(&mut Delay, CustomChar(4)) .write(&mut Delay, CustomChar(5)); }
Clone the existing project
You can clone (or refer) project I created and navigate to the lcd-custom-multi
folder.
git clone https://github.com/ImplFerris/esp32-projects
cd esp32-projects/lcd-custom-multi/
The Full code
#![no_std] #![no_main] use defmt::info; use embassy_executor::Spawner; use embassy_time::Delay; use embassy_time::{Duration, Timer}; use esp_hal::clock::CpuClock; use esp_hal::time::Rate; use esp_hal::timer::timg::TimerGroup; use esp_println as _; use liquid_crystal::prelude::*; use liquid_crystal::LiquidCrystal; use liquid_crystal::I2C; #[panic_handler] fn panic(_: &core::panic::PanicInfo) -> ! { loop {} } const SYMBOL1: [u8; 8] = [ 0b00110, 0b01000, 0b01110, 0b01000, 0b00100, 0b00011, 0b00100, 0b01000, ]; const SYMBOL2: [u8; 8] = [ 0b00000, 0b00000, 0b00000, 0b10001, 0b10001, 0b11111, 0b00000, 0b00000, ]; const SYMBOL3: [u8; 8] = [ 0b01100, 0b00010, 0b01110, 0b00010, 0b00100, 0b11000, 0b00100, 0b00010, ]; const SYMBOL4: [u8; 8] = [ 0b01000, 0b01000, 0b00100, 0b00011, 0b00001, 0b00010, 0b00101, 0b01000, ]; const SYMBOL5: [u8; 8] = [ 0b00000, 0b00000, 0b00000, 0b11111, 0b01010, 0b10001, 0b00000, 0b00000, ]; const SYMBOL6: [u8; 8] = [ 0b00010, 0b00010, 0b00100, 0b11000, 0b10000, 0b01000, 0b10100, 0b00010, ]; #[esp_hal_embassy::main] async fn main(_spawner: Spawner) { // generator version: 0.3.1 let config = esp_hal::Config::default().with_cpu_clock(CpuClock::max()); let peripherals = esp_hal::init(config); let timer0 = TimerGroup::new(peripherals.TIMG1); esp_hal_embassy::init(timer0.timer0); info!("Embassy initialized!"); let i2c_bus = esp_hal::i2c::master::I2c::new( peripherals.I2C0, esp_hal::i2c::master::Config::default().with_frequency(Rate::from_khz(400)), ) .unwrap() .with_scl(peripherals.GPIO18) .with_sda(peripherals.GPIO23) .into_async(); let mut i2c_interface = I2C::new(i2c_bus, 0x27); let mut lcd = LiquidCrystal::new(&mut i2c_interface, Bus4Bits, LCD16X2); lcd.begin(&mut Delay); // Define the characters lcd.custom_char(&mut Delay, &SYMBOL1, 0); lcd.custom_char(&mut Delay, &SYMBOL2, 1); lcd.custom_char(&mut Delay, &SYMBOL3, 2); lcd.custom_char(&mut Delay, &SYMBOL4, 3); lcd.custom_char(&mut Delay, &SYMBOL5, 4); lcd.custom_char(&mut Delay, &SYMBOL6, 5); lcd.set_cursor(&mut Delay, 0, 4) .write(&mut Delay, CustomChar(0)) .write(&mut Delay, CustomChar(1)) .write(&mut Delay, CustomChar(2)); lcd.set_cursor(&mut Delay, 1, 4) .write(&mut Delay, CustomChar(3)) .write(&mut Delay, CustomChar(4)) .write(&mut Delay, CustomChar(5)); loop { Timer::after(Duration::from_secs(1)).await; } }