-
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathchatGPT_DALLE.ino
1088 lines (978 loc) · 31.4 KB
/
chatGPT_DALLE.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// This project works on the ESP32-S3-N16R2 (8MB of PSRAM)
// PSRAM is needed for this project.
// The round display (https://amzn.to/3L4pud6) use the GC9A01 driver from the TFT_eSPI library
// The Micro SD Card Module (optional) is this one : https://amzn.to/46tSgvn
// The Rotary Encoder (optional) is this one: https://amzn.to/3Cj7t61
//
// See the full tutorial here : https://youtu.be/zBzIBRsckTw
#include <PNGdec.h> // Install this library with the Arduino IDE Library Manager
#include <TFT_eSPI.h> // Install this library with the Arduino IDE Library Manager
// Don't forget to configure the driver for your display!
#include <AnimatedGIF.h> // Install this library with the Arduino IDE Library Manager
#include <SimpleRotary.h> // Install this library with the Arduino IDE Library Manager
#include <SD.h> // Install this library with the Arduino IDE Library Manager
#include <WiFiClientSecure.h> // Install this library with the Arduino IDE Library Manager
#include "secrets.h"
#include "arduino_base64.hpp"
#include "display.h"
#include "switch.h"
#include "images\ai.h" // AI Animated GIF
#include "images\readyPng.h" // Ready PNG
#include "images\readyAnimation.h" // Ready Animated GIF
#define SIMULATE_CALL_DALLE // Test images will be used, uncomment this line to make the real call to the DALLE API
#define DEBUG_ON // Comment this line if you don't want detailed messages on the serial monitor, all errors will be printed
#ifndef SIMULATE_CALL_DALLE
// The images folder and index file are created automatically,
// just make your sure SD card is 32GB Max and formatted as FAT32. See this video that shows how to format it: https://youtu.be/Np0cbretbns?t=110
#define USE_SD_CARD // Comment this line if you don't have an SD Card module
#endif
#ifdef USE_SD_CARD
// The rotary encoder can be used only with the SD Card Module
#define USE_ROTARY_ENCODER // Comment this line if you don't want to use the rotary encoder, it is used to navigate files on the SD card module
#endif
#ifdef DEBUG_ON
#define DEBUG_PRINT(x) Serial.print(x)
#define DEBUG_PRINTLN(x) Serial.println(x)
#define DEBUG_PRINTF(...) Serial.printf(__VA_ARGS__)
#else
#define DEBUG_PRINT(x)
#define DEBUG_PRINTLN(x)
#define DEBUG_PRINTF(...)
#endif
#ifdef SIMULATE_CALL_DALLE
#include "base64_test_images\mandalaBase64Png.h"
#include "base64_test_images\rainbowBase64Png.h"
#include "base64_test_images\resistance.h"
#include "base64_test_images\spaceship.h"
#include "base64_test_images\hal.h"
// Test PNG Images
const char *testPngImages[] = {mandalaBase64Png, resistance64Png, spaceship64Png, hal64Png};
const int testImagesCount = 4;
#endif
#ifdef USE_ROTARY_ENCODER
#define ROTARY_PIN_A 41
#define ROTARY_PIN_B 42
#define ROTARY_PUSH_BUTTON 42 // Not used
SimpleRotary rotary(ROTARY_PIN_A, ROTARY_PIN_B, ROTARY_PUSH_BUTTON);
#endif
#ifdef USE_SD_CARD
#define SD_CARD_CS_PIN 5 // Chip Select Pin for the SD Card Module
const char *ID_FILENAME = "id.txt";
const char *IMAGES_FOLDER_NAME = "/images";
int idForNewFile = 1;
unsigned int nbfOfimagesOnSdCard = -1;
#endif
#define ANIMATED_AI_IMAGE ai
AnimatedGIF gif;
TaskHandle_t taskHandlePlayGif = NULL;
void GIFDraw(GIFDRAW *pDraw);
// Switch
SwitchReader generationSwitch(1);
bool runImageGeneration = false; // Flag to indicate if generation of images is running or stopped
#define MAX_IMAGE_WIDTH 1024
#define PSRAM_BUFFER_DECODED_LENGTH 4000000L // Length of buffer for base64 data decoding in PSRAM
#define PSRAM_BUFFER_READ_ENCODED_LENGTH 2000000L // Length of buffer for reading the base64 encoded data in PSRAM
#define BUFFER_RESPONSE_LENGTH 1024 // Length of buffer for reading api response in chunks
// Delimiters to extract base64 data in Json
const char *startToken = "\"b64_json\": \"";
const char *endToken = "\"";
// Prompts
const int promptsCount = 11;
char *prompts[promptsCount] = {"An Alien Planet with Ships Orbiting", "A Futuristic City", "A Star Wars Spaceship", "A Futurisctic HUD Screen",
"A Star Wars Empire Spaceship Attacking", "The Interior of a Spaceship", "Control Panels of an Alien Spaceship", "An Alien World",
"A Star Trek LCARS Interface", "Battle in Space between Spaceships", "A Star Wars Vessel Cockpit View in Space"};
enum imageGenerationMode
{
MODE_SEQUENTIAL, // Images are generated in sequential order of the prompts
MODE_RANDOM // Images are generated by using a random prompt
};
imageGenerationMode generationMode = MODE_SEQUENTIAL; // Choose your mode here
int currentPrompt = 0; // Used when MODE_SEQUENTIAL is used
int16_t xpos = 0;
int16_t ypos = 0;
PNG png; // PNG decoder instance
// Display - You must disable chip select pin definitions in the user_setup.h and the driver setup (ex.: Setup46_GC9A01_ESP32.h)
TFT_eSPI tft = TFT_eSPI(); // Make sure SPI_FREQUENCY is 20000000 in your TFT_eSPI driver for your display if on a breadboard
const int NUM_DISPLAYS = 4; // Adjust this value based on the number of displays
Display display[NUM_DISPLAYS] = {
Display(15), // Assign a chip select pin for each display
Display(7),
Display(6),
Display(16)};
String base64Data; // String buffer for base64 encoded Data
uint8_t *decodedBase64Data; // Buffer to decode base64 data
void setup()
{
Serial.begin(115200);
connectToWifiNetwork();
if (!initDisplayPinsAndStorage())
{
Serial.println("!!! Cannot allocate enough PSRAM to store images");
Serial.println("!!! Code Execution stopped!");
while (true)
{
// Infinite loop, code execution useless without PSRAM
}
}
gif.begin(BIG_ENDIAN_PIXELS);
if (!allocatePsramMemory())
{
Serial.println("!!! Code Execution stopped!");
while (true)
{
// Infinite loop, code execution useless without PSRAM
}
}
#ifdef USE_SD_CARD
if (!initSDCard())
{
Serial.println("!!! Code Execution stopped!");
while (true)
{
// Infinite loop, code execution useless without SD Card Module
}
}
idForNewFile = readNextId(SD) + 1;
DEBUG_PRINTF("ID for the next file is %d\n", idForNewFile);
createDir(SD, IMAGES_FOLDER_NAME);
#endif
createTaskCore();
// delay(5000); // You can safely remove this delay
playReadyOnScreens();
}
void loop()
{
if (runImageGeneration)
{
generateAIImages(generationMode);
}
#ifdef USE_ROTARY_ENCODER
else
{
if (nbfOfimagesOnSdCard > NUM_DISPLAYS - 1)
{
readRotaryEncoder();
}
else
{
Serial.printf("Not enough images on SD Card = %u\n", nbfOfimagesOnSdCard);
}
}
#endif
}
#ifdef USE_ROTARY_ENCODER
void readRotaryEncoder(void)
{
byte i;
int fileIndex;
i = rotary.rotate();
if (i == 1)
{
DEBUG_PRINTLN("Clockwise");
fileIndex = findPreviousFileIndexOnSDCard(display[NUM_DISPLAYS - 1].fileIndex);
shifImagesOnDisplayRight();
displayPngFileFromSDCard(fileIndex, NUM_DISPLAYS - 1);
}
if (i == 2)
{
DEBUG_PRINTLN("Counter Clockwise");
fileIndex = findNextFileIndexOnSDCard(display[0].fileIndex);
shifImagesOnDisplayLeft();
displayPngFileFromSDCard(fileIndex, 0);
}
}
int findPreviousFileIndexOnSDCard(int fileIndex)
{
fileIndex -= 1;
while (fileIndex > 0)
{
String filename = String(IMAGES_FOLDER_NAME) + "/" + String(fileIndex) + ".png";
if (!SD.exists(filename))
{
fileIndex -= 1;
}
else
{
return fileIndex;
}
}
return findPreviousFileIndexOnSDCard(idForNewFile); // Cycle back
}
int findNextFileIndexOnSDCard(int fileIndex)
{
fileIndex += 1;
while (fileIndex < idForNewFile)
{
String filename = String(IMAGES_FOLDER_NAME) + "/" + String(fileIndex) + ".png";
if (!SD.exists(filename))
{
fileIndex += 1;
}
else
{
return fileIndex;
}
}
return findNextFileIndexOnSDCard(0); // Cycle back
}
void displayPngFileFromSDCard(int fileIndex, int screenIndex)
{
size_t imageSize;
const uint8_t *image = NULL;
if (verifyScreenIndex(screenIndex))
{
String filename = String(IMAGES_FOLDER_NAME) + "/" + String(fileIndex) + ".png";
DEBUG_PRINTF("Reading %s from SD Card\n", filename.c_str());
image = readPNGImageFromSDCard(filename.c_str(), &imageSize);
if (image != NULL)
{
displayPngFromRam(image, imageSize, screenIndex);
display[screenIndex].storeImage((uint8_t *)image, imageSize);
display[screenIndex].fileIndex = fileIndex;
delete image; // delete buffer for image
}
}
}
#endif
// Play an animated GIF asynchronously using a CPU Task
// This function is called by the task playAIGifTask
void playAIGif(void *parameter)
{
for (;;)
{
gif.playFrame(true, NULL);
vTaskDelay(5 / portTICK_PERIOD_MS);
}
}
// Start the task to play an animated GIF asynchronously
void startPlayAIGifAsync(void)
{
gif.open((uint8_t *)ANIMATED_AI_IMAGE, sizeof(ANIMATED_AI_IMAGE), GIFDraw);
display[0].activate();
tft.startWrite();
if (taskHandlePlayGif == NULL)
{
taskHandlePlayGif = playAIGifTask();
}
}
// Stop the task to play an animated GIF asynchronously
void stopPlayAIGifAsync(void)
{
if (taskHandlePlayGif != NULL)
{
vTaskDelete(taskHandlePlayGif);
taskHandlePlayGif = NULL; // Set to NULL to indicate that the task is no longer running
}
gif.close();
tft.endWrite();
display[0].deActivate();
}
// Create the task to play an animated GIF asynchronously
TaskHandle_t playAIGifTask()
{
TaskHandle_t taskHandle = NULL;
xTaskCreate(
playAIGif, // Task function
"playAIGif", // Name of task
2048, // Stack size of task (adjust as needed)
NULL, // Parameter of the task
1, // Priority of the task
&taskHandle // Task handle
);
return taskHandle;
}
// Play an animated GIF synchronously
void playAnimatedGIFSync(uint8_t *image, size_t imageSize, int screenIndex)
{
if (verifyScreenIndex(screenIndex))
{
gif.open(image, imageSize, GIFDraw);
display[screenIndex].activate();
tft.startWrite();
while (gif.playFrame(true, NULL))
{
yield();
}
gif.close();
tft.endWrite();
display[screenIndex].deActivate();
}
}
#ifdef USE_SD_CARD
// Initialize Micro SD card Module
bool initSDCard(void)
{
pinMode(SD_CARD_CS_PIN, OUTPUT);
digitalWrite(SD_CARD_CS_PIN, HIGH);
if (!SD.begin(SD_CARD_CS_PIN))
{
// You can get this error if no Micro SD card is inserted into the module
Serial.println("!!! SD Card initialization failed!");
return false;
}
else
{
Serial.println("SD Card initialization success");
}
uint8_t cardType = SD.cardType();
if (cardType == CARD_NONE)
{
Serial.println("!!! No SD card attached");
return false;
}
DEBUG_PRINTLN("SD Card Type: ");
if (cardType == CARD_MMC)
{
DEBUG_PRINTLN("MMC");
}
else if (cardType == CARD_SD)
{
DEBUG_PRINTLN("SDSC");
}
else if (cardType == CARD_SDHC)
{
DEBUG_PRINTLN("SDHC");
}
else
{
DEBUG_PRINTLN("UNKNOWN");
}
uint64_t cardSize = SD.cardSize() / (1024 * 1024);
DEBUG_PRINTF("SD Card Size: %lluMB\n", cardSize);
nbfOfimagesOnSdCard = countFilesInDirectory(IMAGES_FOLDER_NAME);
return true;
}
// Read a PNG image stored on the SD card
uint8_t *readPNGImageFromSDCard(const char *filename, size_t *imageSize)
{
DEBUG_PRINTLN("Reading PNG Image");
File file = SD.open(filename, FILE_READ);
if (!file)
{
Serial.println("!!! Failed to open PNG file for reading");
return NULL;
}
// Get the size of the file
size_t fileSize = file.size();
if (fileSize == 0)
{
Serial.println("!!! PNG file is empty");
file.close();
return NULL;
}
DEBUG_PRINTF("File Size on SD Card is %u\n", fileSize);
// Allocate memory for the file content
uint8_t *buffer = new uint8_t[fileSize];
if (!buffer)
{
Serial.println("!!! Failed to allocate memory for PNG image");
file.close();
return NULL;
}
// Read the file into the buffer
size_t bytesRead = file.read(buffer, fileSize);
file.close();
if (bytesRead != fileSize)
{
Serial.println("!!! Failed to read the entire PNG file");
delete[] buffer;
return NULL;
}
// Set the output parameters
*imageSize = fileSize;
return buffer;
}
// Read the file ID to use on the SD card, for naming the next image to be stored on the SD Card
int readNextId(fs::FS &fs)
{
DEBUG_PRINTLN("Reading next ID");
String idFilename = ID_FILENAME;
File file = fs.open("/" + idFilename);
if (!file)
{
Serial.println("!!! Failed to open ID file for reading");
return 0;
}
String fileContent = "";
while (file.available())
{
fileContent += (char)file.read();
}
file.close();
return fileContent.toInt(); // Convert string to integer and return
}
// Write the ID on the SD card to be used for naming the next image to be stored on the SD card
void writeNextId(fs::FS &fs, int id)
{
DEBUG_PRINTLN("Writing next ID");
String idFilename = ID_FILENAME;
File file = fs.open("/" + idFilename, FILE_WRITE);
if (!file)
{
Serial.println("!!! Failed to open file for writing");
return;
}
// Convert the integer to a string and write it to the file
file.print(id);
file.close();
DEBUG_PRINTLN("ID written successfully");
}
// Store a PNG file on the SD card
void writeImage(fs::FS &fs, const char *path, uint8_t *image, size_t length)
{
DEBUG_PRINTF("Writing file: %s\n", path);
File file = fs.open(path, FILE_WRITE);
if (!file)
{
Serial.println("!!! Failed to open file for writing");
return;
}
if (file.write(image, length))
{
DEBUG_PRINTLN("File written");
}
else
{
Serial.println("!!! Write failed");
}
file.close();
}
// Create a folder on the SD card,
void createDir(fs::FS &fs, const char *path)
{
DEBUG_PRINTF("Creating Dir: %s\n", path);
if (fs.mkdir(path))
{
DEBUG_PRINTLN("Dir created"); // if the folder already exist it reports that it was created without touching it
}
else
{
Serial.println("!!! mkdir failed");
}
}
// Count the number of files in a folder
unsigned int countFilesInDirectory(const char *dirPath)
{
unsigned int fileCount = 0;
File dir = SD.open(dirPath);
if (!dir)
{
Serial.println("Failed to open directory");
return 0;
}
if (!dir.isDirectory())
{
Serial.println("Not a directory");
return 0;
}
File file = dir.openNextFile();
while (file)
{
if (!file.isDirectory())
{
fileCount++; // Count only files, not sub-directories
}
file.close();
file = dir.openNextFile();
}
dir.close();
DEBUG_PRINTF("Number of images on SD card=%u\n", fileCount);
return fileCount;
}
#endif
// CPU Task to read the momentary switch asynchronously
void generationSwitchTask(void *parameter)
{
for (;;)
{ // Task loop
if (generationSwitch.read())
{
runImageGeneration = !runImageGeneration;
if (runImageGeneration)
{
Serial.println("Image generation starting...");
}
else
{
Serial.println("Image generation will be stopped");
}
}
vTaskDelay(50 / portTICK_PERIOD_MS); // Delay to prevent task from running too frequently
}
}
// Generate an image on a screen
void generateAIImages(imageGenerationMode mode)
{
#ifdef SIMULATE_CALL_DALLE
// Test images are used instead of calling the DALLE API
display[0].activate();
tft.fillScreen(TFT_BLACK);
tft.setTextColor(TFT_WHITE);
printTextWithWordWrap("Generating test image", 70, 70, 100);
display[0].deActivate();
startPlayAIGifAsync();
unsigned long t = millis();
while (millis() - t < 5000)
{
}
stopPlayAIGifAsync();
const char *image = testPngImages[myRandom(testImagesCount)];
size_t length = displayPngImage(image, 0);
display[0].storeImage(decodedBase64Data, length);
#else
// Calling the DALLE API to generate images using prompts
size_t length;
if (mode == MODE_SEQUENTIAL)
{
// Sequential mode
length = genereteDalleImage(prompts[currentPrompt]);
currentPrompt++;
if (currentPrompt >= promptsCount)
{
currentPrompt = 0;
}
}
else
{
// Random mode
length = generateDalleImageRandomPrompt();
}
display[0].storeImage(decodedBase64Data, length);
#ifdef USE_SD_CARD // IF a SD card module is used, the image is stored on the SD card
String filename = String(IMAGES_FOLDER_NAME) + "/" + String(idForNewFile) + ".png";
idForNewFile += 1;
writeImage(SD, filename.c_str(), decodedBase64Data, length);
writeNextId(SD, idForNewFile);
#endif
#endif
delay(5000);
if (runImageGeneration)
{
shifImagesOnDisplayLeft();
}
else
{
Serial.println("Image generation stopped");
}
}
// Shift all images to the left on the screens to make place for the next image
void shifImagesOnDisplayLeft(void)
{
for (int i = NUM_DISPLAYS - 1; i >= 0; i--)
{
int displaySource = i - 1;
if (i == 0)
{
return;
}
switchImageOnDisplay(displaySource, i);
}
}
// Shift all images to the right on the screens to make place for the next image
void shifImagesOnDisplayRight(void)
{
for (int i = 0; i < NUM_DISPLAYS; i++)
{
int displaySource = i - 1;
if (i == NUM_DISPLAYS)
{
return;
}
switchImageOnDisplay(i, displaySource);
}
}
// Switch images between screens, the image shown on sourceDisplay is copied to destinationDisplay
void switchImageOnDisplay(int sourceDisplay, int destinationDisplay)
{
if (sourceDisplay == destinationDisplay)
{
Serial.printf("!!! Source and destination display are the same (%d)\n", sourceDisplay);
return;
}
if (verifyScreenIndex(sourceDisplay) && verifyScreenIndex(destinationDisplay))
{
if (display[sourceDisplay].imageSize() > 0)
{
display[destinationDisplay].fileIndex = display[sourceDisplay].fileIndex;
display[destinationDisplay].storeImage(display[sourceDisplay].image(), display[sourceDisplay].imageSize());
displayPngFromRam(display[destinationDisplay].image(), display[destinationDisplay].imageSize(), destinationDisplay);
display[sourceDisplay].activate();
tft.fillScreen(TFT_BLACK);
display[sourceDisplay].deActivate();
}
else
{
display[destinationDisplay].activate();
tft.fillScreen(TFT_BLACK);
display[destinationDisplay].deActivate();
}
return;
}
}
// Verify screen index, should be 0 to NUM_DISPLAYS-1
bool verifyScreenIndex(int screenIndex)
{
if (screenIndex < 0 || screenIndex >= NUM_DISPLAYS)
{
Serial.printf("!!! Wrong screen index=%d, must from 0 to %d\n", screenIndex, NUM_DISPLAYS - 1);
return false;
}
return true;
}
// Initialization of all the displays including buffer space to store the image currently shown
bool initDisplayPinsAndStorage(void)
{
tft.init();
tft.setFreeFont(&FreeSans9pt7b);
for (int i = 0; i < NUM_DISPLAYS; i++)
{
pinMode(display[i].chipSelectPin(), OUTPUT);
display[i].activate();
tft.setRotation(2); // Adjust orientation as needed (0-3)
tft.fillScreen(TFT_BLACK);
display[i].deActivate();
display[i].fileIndex = i;
}
for (int i = 0; i < NUM_DISPLAYS; i++)
{
if (!display[i].reserveMemoryForStorage())
{
return false;
}
}
return true;
}
// Play the "Ready" animated GIF on each screen in sequence
void playReadyOnScreens(void)
{
for (int i = 0; i < NUM_DISPLAYS; i++)
{
playAnimatedGIFSync((uint8_t *)readyAnimation, sizeof(readyAnimation), i);
size_t length = base64::decodeLength(ready64Png);
base64::decode(ready64Png, decodedBase64Data);
display[i].storeImage(decodedBase64Data, length);
}
}
// Return a random PNG image generated by DALLE encoded in base64
size_t generateDalleImageRandomPrompt(void)
{
char *randomPrompt = prompts[myRandom(promptsCount)];
return genereteDalleImage(randomPrompt);
}
// Call DALLE to generate an image using a prompt
size_t genereteDalleImage(char *prompt)
{
display[0].activate();
tft.fillScreen(TFT_BLACK);
tft.setTextColor(TFT_WHITE);
printTextWithWordWrap(prompt, 70, 70, 100);
display[0].deActivate();
callOpenAIAPIDalle(&base64Data, prompt);
size_t length = displayPngImage(base64Data.c_str(), 0);
return length;
}
// Call the DALLE API to generate an image using a prompt
// readBuffer will contain the generated image encoded in base64
void callOpenAIAPIDalle(String *readBuffer, const char *prompt)
{
startPlayAIGifAsync();
*readBuffer = ""; // Clear the buffer
WiFiClientSecure client;
client.setInsecure(); // Only for demonstration, use a proper certificate validation in production
const char *host = "api.openai.com";
const int httpsPort = 443;
if (!client.connect(host, httpsPort))
{
// playAIGif();
Serial.println("!!! Connection failed");
stopPlayAIGifAsync();
return;
}
String jsonPayload = "{\"model\": \"dall-e-2\", \"prompt\": \"";
jsonPayload += prompt;
jsonPayload += "\", \"n\": 1, \"size\": \"256x256\", \"response_format\": \"b64_json\"}";
String request = "POST /v1/images/generations HTTP/1.1\r\n";
request += "Host: " + String(host) + "\r\n";
request += "Content-Type: application/json\r\n";
request += "Authorization: Bearer " + String(chatGPT_APIKey) + "\r\n";
request += "Content-Length: " + String(jsonPayload.length()) + "\r\n";
request += "Connection: close\r\n\r\n";
request += jsonPayload;
client.print(request);
DEBUG_PRINTF("Request sent with prompt : %s\n", prompt);
while (client.connected())
{
String line = client.readStringUntil('\n');
if (line == "\r")
{
DEBUG_PRINTLN("headers received");
break;
}
}
// Read the response in chunks
int bufferLength = BUFFER_RESPONSE_LENGTH;
char buffer[bufferLength];
bool base64StartFound = false, base64EndFound = false;
// Read and process the response in chunks
while (client.available() && !base64EndFound)
{
// playAIGif();
bufferLength = client.readBytes(buffer, sizeof(buffer) - 1);
buffer[bufferLength] = '\0';
if (!base64StartFound)
{
char *base64Start = strstr(buffer, startToken);
if (base64Start)
{
base64StartFound = true;
char *startOfData = base64Start + strlen(startToken); // Skip "base64,"
// Check if the end of base64 data is in the same buffer
char *endOfData = strstr(startOfData, endToken);
if (endOfData)
{
*endOfData = '\0'; // Replace the quote with a null terminator
*readBuffer += startOfData;
base64EndFound = true;
}
else
{
*readBuffer += startOfData;
}
}
}
else
{
char *endOfData = strstr(buffer, endToken);
if (endOfData)
{
*endOfData = '\0'; // Replace the quote with a null terminator
*readBuffer += buffer;
base64EndFound = true;
}
else
{
*readBuffer += buffer;
}
}
}
client.stop();
if (!base64StartFound)
{
Serial.println("!!! No Json Base64 data in Response:");
Serial.println(buffer);
}
DEBUG_PRINTLN("Request call completed");
stopPlayAIGifAsync();
}
// Display a PNG image (decoded) on a specific screen (screenIndex)
void displayPngFromRam(const unsigned char *pngImageinC, size_t length, int screenIndex)
{
if (verifyScreenIndex(screenIndex))
{
int res = png.openRAM((uint8_t *)pngImageinC, length, pngDraw);
if (res == PNG_SUCCESS)
{
DEBUG_PRINTLN("Successfully opened png file");
DEBUG_PRINTF("image specs: (%d x %d), %d bpp, pixel type: %d\n", png.getWidth(), png.getHeight(), png.getBpp(), png.getPixelType());
DEBUG_PRINTF("Image size: %d\n", length);
DEBUG_PRINTF("Buffer size: %d\n", png.getBufferSize());
display[screenIndex].activate();
tft.startWrite();
uint32_t dt = millis();
res = png.decode(NULL, 0);
if (res != PNG_SUCCESS)
{
printPngError(png.getLastError());
}
DEBUG_PRINT(millis() - dt);
DEBUG_PRINTLN("ms");
tft.endWrite();
display[screenIndex].deActivate();
}
else
{
printPngError(res);
}
}
}
// Allocate memory in PSRAM
bool allocatePsramMemory(void)
{
if (psramFound())
{
DEBUG_PRINTF("PSRAM Size=%ld\n", ESP.getPsramSize());
decodedBase64Data = (uint8_t *)ps_malloc(PSRAM_BUFFER_DECODED_LENGTH);
if (decodedBase64Data == NULL)
{
Serial.println("!!! Failed to allocate memory in PSRAM for image Buffer");
return false;
}
if (!base64Data.reserve(PSRAM_BUFFER_READ_ENCODED_LENGTH))
{
Serial.println("!!! Failed to allocate memory in PSRAM for deconding base64 Data");
return false;
}
}
else
{
Serial.println("!!! No PSRAM detected, needed to perform reading and decoding large base64 encoded images");
return false;
}
return true;
}
// Create the CPU task to read the momentary switch asynchronously
void createTaskCore(void)
{
/* Core where the task should run */
xTaskCreatePinnedToCore(
generationSwitchTask, /* Function to implement the task */
"generationSwitchTask", /* Name of the task */
10000, /* Stack size in words */
NULL, /* Task input parameter */
1, /* Priority of the task */
NULL, /* Task handle. */
1); /* Core where the task should run */
}
// Function to print text on the screen with word wrapping
void printTextWithWordWrap(const String &text, int16_t x, int16_t y, uint16_t maxWidth)
{
int16_t cursorX = x;
int16_t cursorY = y;
String currentLine = "";
String word = "";
char currentChar;
tft.setTextWrap(false); // Disable automatic text wrapping
for (int i = 0; i < text.length(); i++)
{
currentChar = text.charAt(i);
// Check for space or newline
if (currentChar == ' ' || currentChar == '\n')
{
if (tft.textWidth(currentLine + word) <= maxWidth)
{
currentLine += word + (currentChar == ' ' ? " " : "");
word = "";
}
else
{
tft.drawString(currentLine, cursorX, cursorY);
cursorY += tft.fontHeight();
currentLine = word + " ";
word = "";
}
if (currentChar == '\n')
{
tft.drawString(currentLine, cursorX, cursorY);
cursorY += tft.fontHeight();
currentLine = "";
}
}
else
{
word += currentChar;
}
}
if (tft.textWidth(currentLine + word) <= maxWidth)
{
tft.drawString(currentLine + word, cursorX, cursorY);
}
else
{
tft.drawString(currentLine, cursorX, cursorY);
cursorY += tft.fontHeight();
tft.drawString(word, cursorX, cursorY);
}
}
//=========================================v==========================================
// pngDraw
//====================================================================================
// This next function will be called during decoding of the png file to
// render each image line to the TFT. If you use a different TFT library
// you will need to adapt this function to suit.
// Callback function to draw pixels to the display
void pngDraw(PNGDRAW *pDraw)
{
uint16_t lineBuffer[MAX_IMAGE_WIDTH];
png.getLineAsRGB565(pDraw, lineBuffer, PNG_RGB565_BIG_ENDIAN, 0xffffffff);
// png.getLineAsRGB565(pDraw, lineBuffer, PNG_RGB565_BIG_ENDIAN, -1);
tft.pushImage(xpos, ypos + pDraw->y, pDraw->iWidth, 1, lineBuffer);
}