This example shows you how to use an ILI9341 display to output a series of images loaded from SD card (there's an SD card connector on the back of the display).
LCD | Pico |
---|---|
SD_SCK | B3 |
SD_MISO | B4 |
SD_MOSI | B5 |
SD_CS | B6 |
Before we can get started, we need to cheat a bit. Espruino can't decode JPEG files on-chip - the display is 320x240 pixels at 16bpp - that's 153600 bytes - which is more RAM than Espruino has in total!
Instead, we're going to store the pictures in a format that Espruino can load directly onto the LCD.
To do that, we need to resize the pictures by 320x240, rotate them (the display is actually 240x320), and convert them to a raw 16bpp RGB565 file formatr.
This is something that you can do with the ImageMagick command-line tools. You just need the command:
convert -size 320x240 -rotate 90 -depth 8 inputfile.jpg \
-channel R +level 0,31 -evaluate multiply 2048 +channel \
-channel G +level 0,63 -evaluate multiply 32 +channel \
-channel B +level 0,31 -evaluate multiply 1 +channel \
-channel RGBA -separate +channel \
-channel R -evaluate-sequence add +channel \
-depth 16 R:outputfile.raw
There's one final step - the display expects 16 bits per pixel, but those bits are sent as two 8 bit bytes. Unfortunately when ImageMagick writes the file, it writes those two bytes in the wrong order so you need to swap them.
On Linux, this is easy:
dd if=outputfile.raw of=fixedoutputfile.raw conv=swab
But on Windows/Mac you may have to find another method.
Once you've got the files sorted, copy them into the root of an SD empty card and put in in the LCD.
Just put the code below into the right-hand side of the Web IDE and click upload. It won't work immediately - you'll have to type save()
on the left-hand side to load it and start it working.
Comments are inline
/* global variables - as we're going to be directly accessing the display,
we need to remember what the CE and DC pins are */
var gspi = new SPI();
var gce = B14;
var gdc = B10;
// graphics object for display - we don't use this directly, but we could do
// if we wanted to draw text over the image
var g;
// List of our files - loaded at power on
var files = [];
function onInit() {
// Set up SD card - load a list of files
var SPI1 = new SPI();
SPI1.setup({mosi:B5,miso:B4,sck:B3,baud:10000000});
E.connectSDCard(SPI1, B6);
files = require("fs").readdirSync();
// Turn LCD backlight on
A6.set();
// Set up LCD (we're using software SPI to avoid wiring the LCD up properly)
gspi.setup({sck:A7, /*miso:A5, */mosi:B1});
g = require("ILI9341").connect(gspi, gdc, gce, B13, function() {
// When initialised, load an image
nextSlide();
});
}
// Load an image onto the screen
function updateScreen(filename) {
// screen width and height
var x2 = 239;
var y2 = 319;
// this code is copied from the ILI9341 module - we need
// to send pixels directly....
// enable SPI bus for LCD
gce.reset();
// Write the 'window' - this will be the whole screen
gspi.write(0x2A,gdc);
gspi.write(0,0,x2>>8,x2);
gspi.write(0x2B,gdc);
gspi.write(0,0,y2>>8,y2);
gspi.write(0x2C,gdc);
// Now open a file, and copy the contents in
// big chunks onto the LCD
var fd = E.openFile(filename,'r');
var d = fd.read(8192);
while (d !== undefined) {
gspi.write(d);
d = fd.read(8192);
}
fd.close();
// disable SPI for the LCD
gce.set();
}
// Choose a new image from our list of filenames
function nextSlide() {
var r = (E.hwRand()&0xFFFFFF)%files.length;
// load
updateScreen(files[r]);
// reschedule again in 5 seconds
setTimeout(nextSlide, 5000);
}
This page is auto-generated from GitHub. If you see any mistakes or have suggestions, please let us know.