[This same post was also made in the Arduino section because it is relevant to both: Arduino and AnimatedGIF on Adafruit Qualia ESP32-S3 for TTL RGB-666 Display ]
Hello. I have the following items:
- Adafruit Qualia ESP32-S3 for TTL RGB-666 Displays
- Round RGB 666 TTL TFT Display - 2.8" 480x480 - No Touch - TL028WVC01-B1621A
Using Arduino IDE, esp32 20.0.17 (by Espressif Systems) for the board manager, AnimatedGIF 2.2.0 (by Larry Bank), and GFX Library for Arduino 1.4.0 (by Moon on our nation), I uploaded the following code:
#include <Arduino_GFX_Library.h>
#include <AnimatedGIF.h>
#include "portal_image.h"
#define DISPLAY_WIDTH 480
#define DISPLAY_HEIGHT 480
Arduino_XCA9554SWSPI *expander = new Arduino_XCA9554SWSPI(
PCA_TFT_RESET, PCA_TFT_CS, PCA_TFT_SCK, PCA_TFT_MOSI,
&Wire, 0x3F); //I2C expander used for BACKLIGHT LED CONTROL
Arduino_ESP32RGBPanel *rgbpanel = new Arduino_ESP32RGBPanel(
TFT_DE, TFT_VSYNC, TFT_HSYNC, TFT_PCLK,
TFT_R1, TFT_R2, TFT_R3, TFT_R4, TFT_R5,
TFT_G0, TFT_G1, TFT_G2, TFT_G3, TFT_G4, TFT_G5,
TFT_B1, TFT_B2, TFT_B3, TFT_B4, TFT_B5,
1 /* hync_polarity */, 46 /* hsync_front_porch */, 2 /* hsync_pulse_width */, 44 /* hsync_back_porch */,
1 /* vsync_polarity */, 50 /* vsync_front_porch */, 16 /* vsync_pulse_width */, 16 /* vsync_back_porch */);
Arduino_RGB_Display *gfx = new Arduino_RGB_Display(
// 2.8" round
DISPLAY_WIDTH /* width */, DISPLAY_HEIGHT /* height */, rgbpanel, 0 /* rotation */, true /* auto_flush */,
expander, GFX_NOT_DEFINED /* RST */, TL028WVC01_init_operations, sizeof(TL028WVC01_init_operations));
AnimatedGIF gif;
int x_offset = 0;
int y_offset = 0;
// Draw a line of image directly on the TFT
void GIFDraw(GIFDRAW *pDraw){
uint8_t *s;
uint16_t *d, *usPalette, usTemp[320];
int x, y, iWidth;
iWidth = pDraw->iWidth;
if (iWidth + pDraw->iX > DISPLAY_WIDTH)
iWidth = DISPLAY_WIDTH - pDraw->iX;
usPalette = pDraw->pPalette;
y = pDraw->iY + pDraw->y; // current line
if (y >= DISPLAY_HEIGHT || pDraw->iX >= DISPLAY_WIDTH || iWidth < 1)
return;
s = pDraw->pPixels;
if (pDraw->ucDisposalMethod == 2) // restore to background color
{
for (x=0; x<iWidth; x++)
{
if (s[x] == pDraw->ucTransparent)
s[x] = pDraw->ucBackground;
}
pDraw->ucHasTransparency = 0;
}
// Apply the new pixels to the main image
if (pDraw->ucHasTransparency) // if transparency used
{
uint8_t *pEnd, c, ucTransparent = pDraw->ucTransparent;
int x, iCount;
pEnd = s + iWidth;
x = 0;
iCount = 0; // count non-transparent pixels
while(x < iWidth)
{
c = ucTransparent-1;
d = usTemp;
while (c != ucTransparent && s < pEnd)
{
c = *s++;
if (c == ucTransparent) // done, stop
{
s--; // back up to treat it like transparent
}
else // opaque
{
*d++ = usPalette[c];
iCount++;
}
} // while looking for opaque pixels
if (iCount) // any opaque pixels?
{
gfx->draw16bitRGBBitmap(pDraw->iX+x+x_offset, y+y_offset, usTemp, iCount, 1);
x += iCount;
iCount = 0;
}
// no, look for a run of transparent pixels
c = ucTransparent;
while (c == ucTransparent && s < pEnd)
{
c = *s++;
if (c == ucTransparent)
iCount++;
else
s--;
}
if (iCount)
{
x += iCount; // skip these
iCount = 0;
}
}
}
else
{
s = pDraw->pPixels;
// Translate the 8-bit pixels through the RGB565 palette (already byte reversed)
for (x=0; x<iWidth; x++)
usTemp[x] = usPalette[*s++];
gfx->draw16bitRGBBitmap(pDraw->iX+x_offset, y+y_offset, usTemp, iWidth, 1);
}
} /* GIFDraw() */
void setup(void)
{
Serial.begin(115200);
#ifdef GFX_EXTRA_PRE_INIT
GFX_EXTRA_PRE_INIT();
#endif
Serial.println("Starting RGB screen");
// Init Display
Wire.setClock(400000); // speed up I2C for expander
if (!gfx->begin()) {
Serial.println("gfx->begin() failed!!!");
while (1) yield();
}
gfx->fillScreen(BLACK);
expander->pinMode(PCA_TFT_BACKLIGHT, OUTPUT);
expander->digitalWrite(PCA_TFT_BACKLIGHT, HIGH); //turn on backlight led
//TEST CODE TO MAKE SURE DISPLAY IS OK BEGINS
Serial.println("Hello!");
gfx->fillScreen(BLACK);
gfx->setCursor(100, gfx->height() / 2 - 75);
gfx->setTextSize(5);
gfx->setTextColor(WHITE);
gfx->println("Hello World!");
Serial.println("Hello in RED!");
gfx->setCursor(100, gfx->height() / 2 - 25);
gfx->setTextColor(RED);
gfx->println("RED");
Serial.println("Hello in GREEEN!");
gfx->setCursor(100, gfx->height() / 2 + 25);
gfx->setTextColor(GREEN);
gfx->println("GREEN");
Serial.println("Hello in BL;UE!");
gfx->setCursor(100, gfx->height() / 2 + 75);
gfx->setTextColor(BLUE);
gfx->println("BLUE");
delay(1000);
gfx->fillScreen(BLACK);
//TEST CODE TO MAKE SURE DISPLAY IS OK ENDS
}
void loop(){
// put your main code here, to run repeatedly:
if (gif.open((uint8_t *)gif_data, sizeof(gif_data), GIFDraw)){
Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
if( gif.getCanvasWidth() < DISPLAY_WIDTH || gif.getCanvasHeight() < DISPLAY_HEIGHT){
x_offset = (DISPLAY_WIDTH - gif.getCanvasWidth()) / 2;
y_offset = (DISPLAY_HEIGHT - gif.getCanvasHeight())/ 2;
}
/*
while (gif.playFrame(true, NULL))
{
}
gif.close();*/
Serial.println("Starting GIF playback...");
while (gif.playFrame(true, NULL)) {
Serial.println("Playing frame...");
delay(10); // Optional delay to control frame rate
}
Serial.println("Finished GIF playback.");
gif.close();
} else {
Serial.println("Failed to open GIF.");
}
}
The āportal_image.hā is an animated GIF 480x480 pixel that has already been converted to binary.
The current code prints the following information to the Serial Monitor:
Hello!
Hello in RED!
Hello in GREEEN!
Hello in BL;UE!
Successfully opened GIF; Canvas size = 480 x 480
Starting GIF playback...
And finally for what it actually displays on the screen, it successfully prints out Hello World and a number of other coloured words. Then shortly after, the device crashes and reboots when attempting to load the GIF.
Please note: I have already attempted to ask this question on the official Adafruit and Arduino forums months ago and they were thoroughly unhelpful. The code youāre seeing above is actually provided to me by a programmar on Fiverr who is currently trying to solve the issue. Every online tutorial Iāve found online hasnāt been very helpful. The animated GIF in question, as far as Iām aware from the uploading output, successfully transferred to the device.
Here is the output:
Sketch uses 959245 bytes (45%) of program storage space. Maximum is 2097152 bytes.
Global variables use 54756 bytes (16%) of dynamic memory, leaving 272924 bytes for local variables. Maximum is 327680 bytes.
esptool.py v4.5.1
Serial port COM7
Connecting...
Chip is ESP32-S3 (revision v0.2)
Features: WiFi, BLE
Crystal is 40MHz
MAC: 74:4d:bd:9d:41:84
Uploading stub...
Running stub...
Stub running...
Changing baud rate to 921600
Changed.
Configuring flash size...
Erasing flash (this may take a while)...
Chip erase completed successfully in 3.4s
Compressed 21696 bytes to 13543...
Writing at 0x00000000... (100 %)
Wrote 21696 bytes (13543 compressed) at 0x00000000 in 0.3 seconds (effective 510.5 kbit/s)...
Hash of data verified.
Compressed 3072 bytes to 136...
Writing at 0x00008000... (100 %)
Wrote 3072 bytes (136 compressed) at 0x00008000 in 0.1 seconds (effective 224.2 kbit/s)...
Hash of data verified.
Compressed 8192 bytes to 47...
Writing at 0x0000e000... (100 %)
Wrote 8192 bytes (47 compressed) at 0x0000e000 in 0.1 seconds (effective 600.2 kbit/s)...
Hash of data verified.
Compressed 959616 bytes to 811640...
Writing at 0x00010000... (2 %)
Writing at 0x0001409f... (4 %)
Writing at 0x00018099... (6 %)
Writing at 0x0001c094... (8 %)
Writing at 0x0002008f... (10 %)
Writing at 0x0002408a... (12 %)
Writing at 0x00028085... (14 %)
Writing at 0x0002c080... (16 %)
Writing at 0x0003007b... (18 %)
Writing at 0x00034076... (20 %)
Writing at 0x00038071... (22 %)
Writing at 0x0003c06c... (24 %)
Writing at 0x00040067... (26 %)
Writing at 0x00044062... (28 %)
Writing at 0x0004805d... (30 %)
Writing at 0x0004c058... (32 %)
Writing at 0x00050053... (34 %)
Writing at 0x0005404e... (36 %)
Writing at 0x00058049... (38 %)
Writing at 0x0005c044... (40 %)
Writing at 0x0006003f... (42 %)
Writing at 0x0006403a... (44 %)
Writing at 0x00068035... (46 %)
Writing at 0x0006c030... (48 %)
Writing at 0x0007002b... (50 %)
Writing at 0x00074026... (52 %)
Writing at 0x00078021... (54 %)
Writing at 0x0007c01c... (56 %)
Writing at 0x00080017... (58 %)
Writing at 0x00084012... (60 %)
Writing at 0x0008800d... (62 %)
Writing at 0x0008c008... (64 %)
Writing at 0x00090003... (66 %)
Writing at 0x00093ffe... (68 %)
Writing at 0x00097ff9... (70 %)
Writing at 0x0009bff4... (72 %)
Writing at 0x0009ffef... (74 %)
Writing at 0x000a3fea... (76 %)
Writing at 0x000a8b3b... (78 %)
Writing at 0x000b795e... (80 %)
Writing at 0x000c0da4... (82 %)
Writing at 0x000c6080... (84 %)
Writing at 0x000cb943... (86 %)
Writing at 0x000d0ff8... (88 %)
Writing at 0x000d6437... (90 %)
Writing at 0x000db6d0... (92 %)
Writing at 0x000e1a4a... (94 %)
Writing at 0x000ebf43... (96 %)
Writing at 0x000f180a... (98 %)
Writing at 0x000f734e... (100 %)
Wrote 959616 bytes (811640 compressed) at 0x00010000 in 9.6 seconds (effective 802.2 kbit/s)...
Hash of data verified.
Compressed 157728 bytes to 101221...
Writing at 0x00410000... (14 %)
Writing at 0x00417b41... (28 %)
Writing at 0x0041d0ff... (42 %)
Writing at 0x0042404f... (57 %)
Writing at 0x0042ae12... (71 %)
Writing at 0x00430367... (85 %)
Writing at 0x004354e5... (100 %)
Wrote 157728 bytes (101221 compressed) at 0x00410000 in 1.3 seconds (effective 943.8 kbit/s)...
Hash of data verified.
Leaving...
Hard resetting via RTS pin...
What Iām after is:
-Advice on why it is crashing when attempting to load the GIF.
-What should I do to fix this?