DIY Thermal Camera: Build your Own Infrared Imaging Device with ESP32 and MLX90640 Sensor

Published  October 13, 2023   5
DIY Thermal Camera using ESP32 and MLX90640 Sensor

 

Ditch X-ray vision, thermal cameras are the real industrial superheroes! They don't just see light, they see heat, making them masters of uncovering hidden energy drains, pinpointing overheating gadgets, and even aiding soldiers in the dark. They're invaluable in industries like tech repair, where they sniff out shorted circuits like electronic bloodhounds. The only catch? Their price tag can be a bit frosty. But hey, technology is always evolving, and soon, thermal vision might be just as common as your phone camera! So remember, that seemingly ordinary camera might just be packing some extraordinary heat-sensing powers.

So, in this project, we are going to find some solution for that by making a DIY thermal imager using much cheaper available components. The cheaper thermal imaging sensors we have considered are the AMG8833 from Panasonic, MLX90640 and MLX90641 from Melexis. Even though the AMG8833 is the cheapest of them all it only has a resolution of 8x8, while MLX90640 offers a resolution of 32x24 and MLX90641 offers a resolution of 16x12. Since the MLX90640 offers the best resolution in the price range we choose it for our DIY thermal camera.

Before this we have worked on Thermal Camera with Raspberry Pi and Thermal Camera with Arduino, Be sure to check them out as well.

Features of DIY Thermal camera

  • Image sensor resolution: 32x24.
  • Sensor field of view (FoV): 55°x35°
  • Temperature measurement range: -40 to 300°C
  • -40 to 85°C operational temperature range.
  • Adjustable refresh rate – 4Hz – 32Hz.
  • 10 different colour pallets.
  • 5 different interpolation modes.
  • Easy to use GUI.
  • 2.4” TFT display with 320x240 resolution.
  • Save thermal image to SD card.
  • Built-in battery and charging circuit.
  • Open source

Components Required to Build the DIY Thermal Camera

The components required to build the DIY Thermal Camera are listed below. The exact value of each component can be found in the schematics or the BOM.

  • ESP32 Wrover Module with 8Mb flash and PSRAM – x1
  • MLX90640 Far infrared thermal sensor array Sensor – x1
  • 2.4” TFT display 320x240 resolution ILI9341 driver – x1
  • CH340K USB - UART controller – x1
  • TP4056 Li-ion charger IC – x1
  • MIC5219-3.3YM5 3.3v LDO – x1
  • AO3401 P - MOSFET – x1
  • 2n7002DW dual N - MOSFET – x1
  • S8050 Transistor - x1
  • SS34 Diode – x1
  • SD Card Reader – x1
  • Type C USB Connector 16Pin – x1
  • SMD resistors and capacitors
  • SMD LEDs
  • SMD Tactile switches
  • SDM Slide Switch
  • Connectors
  • Custom PCB
  • 3D printed Enclosure and mounting screws
  • Other tools and consumables.

Circuit Diagram - DIY Thermal Camera

The complete circuit diagram for the DIY Thermal Camera is shown below. It can also be downloaded in PDF format. We have used KiCad to design the schematics and PCB for this project, you can find the complete files on out GitHub page given at the bottom of this article. 

DIY Thermal Camera using ESP32 and MLX90640 Sensor Circuit Diagram

Let’s discuss the Schematics section by section for better understanding. A type C USB port is used for both charging as well as programming purposes. The power from the USB port is connected to a power path controller circuit built around a P-Channel MOSFET U2 and a diode D1. When the USB power is available the device will operate from the USB power and also will charge the internal battery, when the USB power is cut, the device will automatically change to battery power. For voltage regulation, we have used a MIC5219 3.3V LDO from Microchip, which is capable of providing up to 500mA of current with a very low dropout voltage of 500mV at full load. A slide switch with a pullup resistor is connected to the enable pin of MIC5219. This switch is used for turning on and off the thermal camera. When this pin is pulled to the ground the LDO will be shut down and hence the other parts of the device too, except the battery charging section. For charging the internal battery we are using a TP4056 charge controller which is capable of a maximum charge current of 1A. For battery voltage sensing we have used a classic voltage divider, which will reduce the battery voltage to a safe level for measurement.

MIC5219 and TP4056 Circuit Diagram

In the next section, we have the ESP32 SoC itself along with the programming circuit and the MLX90640 Far Infrared thermal sensor array. The programming circuit consists of a CH340K USB to UART controller along with a 2N7002DW dual N – N-Channel MOSFET from ON Semi. We chose the CH340K because of its small footprint as well as its low cost. The tiny dual MOSFET will act as an auto-reset circuit for the ESP32 thus removing any need for manual rest or boot selection during programming. Even though the auto-reset works flawlessly, we have also added boot and reset buttons, just in case. The circuit around the ESP32 is standard, just bypass capacitors and pullup resistors. For the image sensor, we have decided to solder it directly to the PCB to make the whole device compact. The sensor is interfaced to the SoC using I2C and only has 4 pins, including power and ground. If we want to use an image sensor module instead of soldering the sensor to the PCB, or if we want to add some other I2C device, we can use the additional I2C connector added to the PCB.

ESP32 SoC  and MLX90640 Circuit Diagram

In the last section, we have the TFT display along with the navigation buttons and Micro SD slot. For display, we have used a 2.4” TFT display with a resolution of 320x240. The display uses an ILI9341 driver chip. It is interfaced to the ESP32 using SPI and supports SPI speed up to 65MHz. The display is directly soldered to the PCB. For backlight control, we have used a S8050 transistor. We will be able to control the backlight brightness using PWM signals. The display is connected to the VSPI interface of ESP32 while the Micro SD slot is connected to the HSPI interface. This will ensure that the ESP32 can access or control both the display as well as the SD card at the same time in case needed.

TFT Display and Navigation Switches Circuit Diagram

PCB for DIY Thermal Camera

For this project, we have decided to make a custom PCB. This will ensure that the final product is as compact as possible as well as easy to assemble and use. The PCB has a dimension of 80mm x 50mm. Here are the top and bottom layers of the PCB.

Thermal Camera PCB Design

And here is the 3D view of the PCB.

Thermal Camera 3D PCB Design

Here is the fully assembled PCB.

PCB for Thermal Camera using ESP32 and mlx90640

3D Printed Parts

We have designed a cool-looking 3D-printed enclosure for the DIY thermal camera. The files for all the 3D printed parts can be downloaded from the GitHub link provided at the end of the article along with the Arduino sketch and bitmap file. Learn more about 3D printing and how to get started with it by following the link. 

You can get all the STL files from here

DIY Thermal Camera

Thermal Camera 3D Printed Parts

And here is the fully assembled DIY Thermal Camera.

Thermal Camera using ESP32 and MLX90640 in 3D printed casing

DIY Thermal Camera Overview

Here is the overall overview of a fully assembled DIY Thermal Camera.

DIY Thermal Camera Overview

The button usage is the following.

  • Up/+ Button:
    • Default: Change/cycle through interpolation mode.
    • In Settings: Change the selected value.
  • Middle/Ok Button:
    • Default:
      • Short press: Save images to SD card.
      • Long Press: Enter the settings menu.
    • In Settings:
      • Short press: Change the selection in the menu.
      • Long Press: Exit the settings menu.
  • Down/- Button:
    • Default: Change colour pallet.
    • In Settings: Change the selected value.

As mentioned above you can cycle through different interpolation modes and colour pallets using the Up or Down button respectively while on the main screen. You can save the current view as a BMP file to the SD card with a short press on the middle button. If the image is saved successfully a save success animation will be played. If there is some error, like the SD card is not inserted or not supported it will show an animated SD card error message. Make sure to format the SD card to FAT32 format and turn off and then turn off the camera if you have inserted the SD card while the camera is in on state. If you don’t reboot after inserting the SD card the device may fail to detect the SD card. So it is important to either insert the SD card while the camera is in the off state or just reboot the device.

Here is the main screen interface. On the main screen, you can see the thermal image itself along with the minimum, maximum and middle point temperatures and battery icon.

DIY Thermal Camera Temperature Reading

The image below shows the Settings screen of the DIY thermal camera. The settings have 7 options. The selected option will show in green text while others will be shown in white text. You can change the selection by short pressing the middle button. The value of the selected option can be adjusted using the Up/+ or Down/- button.

Here are the settings options and the corresponding values available.

  • Auto Scale
    • On: Auto scale on. Colour will display according to the maximum and minimum temperature readings from the sensor.
    • Off: Auto scale on. Colour will display according to the maximum and minimum temperature set manually.
  • Min Temperature: Minimum temperature for colour palette calculation when the auto scale is off.
  • Max Temperature: Maximum temperature for colour palette calculation when the auto scale is off.
  • Interpolation: Interpolation method. It refers to the interpolation algorithm used to upscale the low-resolution image from the sensor(32x24) to the high-resolution image(320x240) displayed on the screen. Depending on the interpolation method the quality of the image as well as the image processing time will change. The available methods are the following.
    • Nearest Neighbor
    • Average
    • Bilinear
    • Bilinear Fast
    • Triangle
  • Palette: The colour palette refers to the different display modes. There are 10 different palettes available to select.
  • Refresh Rate: Refresh rate of the image sensor. The MLX90640 supports refresh rates from 0.5Hz to 64Hz. But while testing the most usable refresh rates were 4Hz, 8Hz, 16Hz and 32 Hz. So, we have only included these refresh rates.
  • Backlight: Backlight brightness. You can set it from 10% to 100%.

Arduino Code for DIY Thermal Camera

Now let’s look at the code. As usual, we have included all the necessary libraries to the code using the include function, which included TFT_eSPI, Adafruit_MLX90640, Preferences, AnimatedGIF and other standard libraries. We have also included the animated image data along with the font files. You can download all the necessary files from the GitHub repo linked at the bottom of this article. After that, we have defined all the necessary global variables. Later we created instances for each individual. We will use these instances to access the corresponding function.

#include <Preferences.h>
#include <Wire.h>
#include <SPI.h>
#include <SD.h>
#include "FS.h"
#include <Adafruit_MLX90640.h>
#include <TFT_eSPI.h>
#include <AnimatedGIF.h>
#include <Fonts/GFXFF/gfxfont.h>  //Include a library of Fonts
#include "Open_Sans_ExtraBold_10.h"
#include "BootAnimation.h"
#include "success.h"
#include "Error.h"
SPIClass spiSD(HSPI);
#define SD_CS 15
//#define USE_DMA
#define NORMAL_SPEED
#define TFT_WIDTH 320
#define TFT_HEIGHT 240
#define BUFFER_SIZE 320
uint16_t usTemp[BUFFER_SIZE];
#define VBAT_PIN 33
#define BATTV_MAX 4.2  // maximum voltage of battery
#define BATTV_MIN 3.2  // what we regard as an empty battery
#define GIF_IMAGE BootAnimationIMG
#define SGIF_IMAGE success_GIF
#define EGIF_IMAGE Error_GIF
bool dmaBuf = 0;
Adafruit_MLX90640 mlx;
Preferences preferences;
AnimatedGIF gif;
TFT_eSPI tft = TFT_eSPI();
File bmpFile;
float frame[32 * 24];
float batv;
const int upButton = 35;
const int middleButton = 36;
const int downButton = 39;
const int backlightPin = 4;
int interpolationMode = 0;
int AutoScale = 0;
volatile bool MenuChange = false;
volatile bool _upShort = false;
volatile bool _downShort = false;
volatile bool middleShort = false;
volatile bool middlePressed = false;
unsigned long middlePressStartTime = 0;
int Menu = 0, Menuitem = 0;
int BLPWM = 0;
int RefreshRate = 0;
float MinT, MaxT;
String menuItems[7] = { "Auto Scale    : ", "Min Temp      : ", "Max Temp      : ", "Interpolation : ", "Palette       : ", "Refresh Rate  : ", "BackLight     : " };
String ASindex[2] = { "Off", "On" };
String IPindex[6] = { "Nearest Neighbor", "Average", "Bilinear", "Bilinear Fast", "Triangle" };
String RRindex[4] = { "4 Hz", "8 Hz", "16 Hz", "32 Hz" };
float xRatios[320];
float yRatios[240];
float xOppositeRatios[320];
float yOppositeRatios[240];
#define PALETTE_COUNT 10
uint16_t colorPalettes[PALETTE_COUNT][6] = {
  { TFT_BLUE, TFT_CYAN, TFT_GREEN, TFT_YELLOW, TFT_RED, TFT_MAGENTA },
  { TFT_BLACK, TFT_DARKGREY, TFT_LIGHTGREY, TFT_WHITE, TFT_ORANGE, TFT_PINK },
  { TFT_NAVY, TFT_OLIVE, TFT_DARKGREEN, TFT_DARKCYAN, TFT_MAROON, TFT_PURPLE },
  { TFT_BLUE, TFT_GREEN, TFT_DARKGREEN, TFT_ORANGE, TFT_MAROON, TFT_RED },
  { TFT_NAVY, TFT_DARKGREEN, TFT_GREEN, TFT_YELLOW, TFT_ORANGE, TFT_RED },
  { TFT_CYAN, TFT_BLUE, TFT_MAGENTA, TFT_YELLOW, TFT_GREEN, TFT_RED },
  { TFT_WHITE, TFT_ORANGE, TFT_RED, TFT_BLUE, TFT_GREEN, TFT_BLACK },
  { TFT_PURPLE, TFT_MAGENTA, TFT_RED, TFT_ORANGE, TFT_YELLOW, TFT_GREEN },
  { TFT_YELLOW, TFT_PINK, TFT_WHITE, TFT_BLUE, TFT_DARKCYAN, TFT_DARKGREEN },
  { TFT_RED, TFT_YELLOW, TFT_GREEN, TFT_CYAN, TFT_BLUE, TFT_MAGENTA }
};
int paletteIndex = 0;
float tempMin = 20.0;  // Minimum temperature
float tempMax = 32.0;  // Maximum temperature
TFT_eSprite sprite = TFT_eSprite(&tft);

The upButton_ISR, downButton_ISR and middleButton_ISR functions are used to detect and process switch presses. It uses hardware interrupts for detecting the changes of the pins that are connected to the switches. We have used software debounce for proper detections of switch presses and avoid any noise. Depending on the pressed switch and pressed time the function will set a corresponding variable to true which will be processed with in the main program.

void IRAM_ATTR upButton_ISR() {
  static unsigned long last_interrupt_time = 0;
  unsigned long interrupt_time = millis();
  if (interrupt_time - last_interrupt_time > 200) {  // simple debounce
    _upShort = true;
  }
  last_interrupt_time = interrupt_time;
}
void IRAM_ATTR downButton_ISR() {
  static unsigned long last_interrupt_time = 0;
  unsigned long interrupt_time = millis();
  if (interrupt_time - last_interrupt_time > 200) {  // simple debounce
    _downShort = true;
  }
  last_interrupt_time = interrupt_time;
}
void IRAM_ATTR middleButton_ISR() {
  if (digitalRead(middleButton) == LOW) {
    // Button press event
    if (!middlePressed) {  // If button was not already being pressed
      middlePressed = true;
      middlePressStartTime = millis();  // Save the start time of button press
    }
  } else {
    // Button release event
    if (middlePressed) {                             // If the button was being pressed
      if (millis() - middlePressStartTime < 1000) {  // If the button was pressed for less than 1 second
        middleShort = true;
      }
      middlePressed = false;
      middlePressStartTime = 0;  // Reset the start time of button press
    }
  }
}

The functions MLXInit, ConfigRefreshrate, ReadConfig and WriteConfig functions are used to communicate with and to configure the MLX90640 sensor. The MLXInit function is used to initialise the sensor at the startup. ConfigRefresh rate function us used to set the image sensor refresh rate as the name suggests. The other to functions are used to read and write configurations to the corresponding registers in MLX90640.

void MLXInit() {
  Wire.begin();
  Wire.setClock(1000000);
  if (!mlx.begin()) {
    Serial.println("Failed to initialize MLX90640!");
  }
  mlx.setResolution(MLX90640_ADC_18BIT);
  ConfigRefreshrate();
}
void ConfigRefreshrate() {
  switch (RefreshRate) {
    case 1:
      mlx.setRefreshRate(MLX90640_4_HZ);
      break;
    case 2:
      mlx.setRefreshRate(MLX90640_8_HZ);
      break;
    case 3:
      mlx.setRefreshRate(MLX90640_16_HZ);
      break;
    case 4:
      mlx.setRefreshRate(MLX90640_32_HZ);
      break;
    default:
      break;
  }
}
void ReadConfig() {
  preferences.begin("Config", false);
  String temp;
  temp = preferences.getString("AutoScale", "");
  AutoScale = temp.toInt();
  temp = preferences.getString("MinTe", "");
  MinT = temp.toFloat();
  temp = preferences.getString("MaxTe", "");
  MaxT = temp.toFloat();
  temp = preferences.getString("Imode", "");
  interpolationMode = temp.toInt();
  temp = preferences.getString("PIndex", "");
  paletteIndex = temp.toInt();
  temp = preferences.getString("RefreshRate", "");
  RefreshRate = temp.toInt();
  temp = preferences.getString("BLPWM", "");
  BLPWM = temp.toInt();
  preferences.end();
}
void WriteConfig() {
  preferences.begin("Config", false);
  preferences.putString("AutoScale", String(AutoScale));
  preferences.putString("MinTe", String(MinT));
  preferences.putString("MaxTe", String(MaxT));
  preferences.putString("Imode", String(interpolationMode));
  preferences.putString("PIndex", String(paletteIndex));
  preferences.putString("RefreshRate", String(RefreshRate));
  preferences.putString("BLPWM", String(BLPWM));
  preferences.end();
}

initSDcard function is used to initialize the SD card as well as to detect the presence of the SD card prior to saving the image. If the SD card is not detected or it is not supported the function will return corresponding error code.

void initSDcard() {
  //spiSD.begin(14, 12, 13, SD_CS);  //CLK,MISO,MOIS,SS
  if (!SD.begin(SD_CS, spiSD)) {
    Serial.println("Card Mount Failed");
    return;
  } else {
    Serial.println("Card Mount Successful");
  }
  uint8_t cardType = SD.cardType();
  if (cardType == CARD_NONE) {
    Serial.println("No SD card attached");
    return;
  }
}

The generateFilename function is used to generate image file names. It will generate file name in increment. If the file already exists in the SD card the function will check for it and generate a new name with an incremented number. The writeBMP function is used to write the image file to the SD card when the image save button is pressed. This function will generate a BMP image file with needed header data and will save the screen buffer to it. Essentially saving the image on the screen to the SD card.

String generateFilename(fs::FS &fs) {
  for (int i = 0; i <= 9999; i++) {
    char filename[23];                               // Allocate the char array
    sprintf(filename, "/ThermalCamera%04d.bmp", i);  // Print formatted string into char array
    if (!fs.exists(filename)) {
      return String(filename);  // Return a String object
    }
  }
  return "";  // return empty string if all filenames are taken
}
int writeBMP(fs::FS &fs, const char *path, TFT_eSprite *sprite) {
  const int width = sprite->width();
  const int height = sprite->height();
  // BMP file header (14 bytes)
  uint8_t bmpFileHeader[14] = { 'B', 'M', 0, 0, 0, 0, 0, 0, 0, 0, 54, 0, 0, 0 };
  // The size of the BMP file in bytes
  uint32_t fileSize = 54 + width * height * 2;
  bmpFileHeader[2] = (uint8_t)(fileSize);
  bmpFileHeader[3] = (uint8_t)(fileSize >> 8);
  bmpFileHeader[4] = (uint8_t)(fileSize >> 16);
  bmpFileHeader[5] = (uint8_t)(fileSize >> 24);
  // BMP info header (40 bytes)
  uint8_t bmpInfoHeader[40] = { 40, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 16, 0 };
  bmpInfoHeader[4] = (uint8_t)(width);
  bmpInfoHeader[5] = (uint8_t)(width >> 8);
  bmpInfoHeader[6] = (uint8_t)(width >> 16);
  bmpInfoHeader[7] = (uint8_t)(width >> 24);
  bmpInfoHeader[8] = (uint8_t)(height);
  bmpInfoHeader[9] = (uint8_t)(height >> 8);
  bmpInfoHeader[10] = (uint8_t)(height >> 16);
  bmpInfoHeader[11] = (uint8_t)(height >> 24);
  File file = fs.open(path, FILE_WRITE);
  if (!file) {
    Serial.println("Failed to open file for writing");
    return 0;
  } else {
    //showSavingImageMessage();
    // Write headers
    file.write(bmpFileHeader, 14);
    file.write(bmpInfoHeader, 40);
    // Write pixel data
    for (int y = height - 1; y >= 0; y--) {  // BMP is stored bottom-top
      for (int x = 0; x < width; x++) {
        uint16_t pixel = sprite->readPixel(x, y);
        // Swap red and green channels
        uint16_t r = (pixel >> 11) & 0x1F;
        uint16_t g = (pixel >> 5) & 0x3F;
        uint16_t b = pixel & 0x1F;
        pixel = (b << 11) | (r << 5) | g;
        file.write(pixel >> 8);  // high byte
        file.write(pixel);       // low byte
      }
    }
    file.close();
    return 1;
  }
}

The navigationUpdate function is responsible for all the routines related to the tactile switches and menu navigation. Depending on which screen we are currently in this function will process the key presses accordingly.

void navigationUpdate() {
  if (_upShort == true) {
    _upShort = false;
    if (Menu == 0) {
      interpolationMode++;
      if (interpolationMode > 4) interpolationMode = 0;
    } else {
      if (Menuitem == 0) {
        AutoScale++;
        if (AutoScale > 1) AutoScale = 1;
      } else if (Menuitem == 1) {
        MinT++;
        if (MinT > 300) MinT = 300;
      } else if (Menuitem == 2) {
        MaxT++;
        if (MaxT > 300) MaxT = 300;
      } else if (Menuitem == 3) {
        interpolationMode++;
        if (interpolationMode > 4) interpolationMode = 4;
      } else if (Menuitem == 4) {
        paletteIndex++;
        if (paletteIndex > 9) paletteIndex = 9;
      } else if (Menuitem == 5) {
        RefreshRate++;
        if (RefreshRate > 3) RefreshRate = 3;
        ConfigRefreshrate();
      } else if (Menuitem == 6) {
        BLPWM = BLPWM + 10;
        if (BLPWM > 100) BLPWM = 100;
        analogWrite(backlightPin, map(BLPWM, 0, 100, 0, 255));  // Turn on backlight
      }
      MenuChange = true;
    }
    WriteConfig();
    Serial.println("Up Short Press");
  }
  if (_downShort == true) {
    _downShort = false;
    if (Menu == 0) {
      paletteIndex = (paletteIndex + 1) % PALETTE_COUNT;
    } else {
      if (Menuitem == 0) {
        AutoScale--;
        if (AutoScale < 0) AutoScale = 0;
      } else if (Menuitem == 1) {
        MinT--;
        if (MinT < 0) MinT = 0;
      } else if (Menuitem == 2) {
        MaxT--;
        if (MaxT < 5) MaxT = 5;
      } else if (Menuitem == 3) {
        interpolationMode--;
        if (interpolationMode < 0) interpolationMode = 0;
      } else if (Menuitem == 4) {
        paletteIndex--;
        if (paletteIndex < 0) paletteIndex = 0;
      } else if (Menuitem == 5) {
        RefreshRate--;
        if (RefreshRate < 0) RefreshRate = 0;
        ConfigRefreshrate();
      } else if (Menuitem == 6) {
        BLPWM = BLPWM - 10;
        if (BLPWM < 10) BLPWM = 10;
        analogWrite(backlightPin, map(BLPWM, 0, 100, 0, 255));  // Turn on backlight
      }
      MenuChange = true;
    }
    WriteConfig();
    Serial.println("down Short Press");
  }
  if (middleShort == true) {
    middleShort = false;
    if (Menu == 0) {
      if (!SD.begin(SD_CS, spiSD)) {
        Serial.println("Card Mount Failed");
        if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
          Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
          while (gif.playFrame(true, NULL)) {
            sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
            yield();
          }
          gif.close();
        }
        return;
      } else {
        Serial.println("Card Mount Successful");
      }
      uint8_t cardType = SD.cardType();
      if (cardType == CARD_NONE) {
        Serial.println("No SD card attached");
        if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
          Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
          while (gif.playFrame(true, NULL)) {
            sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
            yield();
          }
          gif.close();
        }
        return;
      } else {
        String filename = generateFilename(SD);
        if (filename != "") {
          if (writeBMP(SD, filename.c_str(), &sprite)) {
            if (gif.open((uint8_t *)SGIF_IMAGE, sizeof(SGIF_IMAGE), GIFDraw1)) {
              Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
              while (gif.playFrame(true, NULL)) {
                sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
                yield();
              }
              gif.close();
            }
          } else {
            if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
              Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
              while (gif.playFrame(true, NULL)) {
                sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
                yield();
              }
              gif.close();
            }
          }
        } else {
          Serial.println("Failed to create filename.");
          if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
            Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
            while (gif.playFrame(true, NULL)) {
              sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
              yield();
            }
            gif.close();
          }
        }
      }
    } else {
      Menuitem++;
      if (Menuitem > 6) {
        Menuitem = 0;
      }
      MenuChange = true;
    }
    Serial.println("Middle Short Press");
    Serial.print(Menu);
    Serial.print("");
    Serial.println(Menuitem);
  }
  if (middlePressed && !middleShort && millis() - middlePressStartTime > 1000 && middlePressed && !middleShort && millis() - middlePressStartTime < 1500) {
    // If the button is being pressed, no short press has been registered, and it has been over 1 second
    Serial.println("Middle Long Press");
    middlePressed = false;     // Reset the pressed flag
    middlePressStartTime = 0;  // Reset the press start time
    if (Menu == 0) {
      Menu = 1;
      MenuChange = true;
      Menuitem = 0;
    } else {
      Menu = 0;
    }
  }
}

Now the displayUpdate function is responsible for all the graphics related routines as well as responsible for acquiring image data from MLX90640 and processing it. This function will read data from the image sensor when there is new data is available. Then it will upscale it is using the selected interpolation method. Once its upscaled it will call drawPixel function to draw the image pixel by pixel. The drawPixel function will use the selected colour palette to determine the appropriate colour for each pixel. The displayUpdate function is also responsible for printing minimum, maximum and middle point temperature along with the battery icon (using drawBattery function) to the display while on main screen. The same function is also used to draw and process the settings menu too.

void displayUpdate() {
  if (!mlx.getFrame(frame)) {
    // Failed to get frame, so reinitialize
    MLXInit();
  }
  float tempMinRead = 1000;   // some high value
  float tempMaxRead = -1000;  // some low value
  float tempCenter = 0.0;     // Temperature at center
  // Update the minimum and maximum temperatures read from the sensor
  for (int i = 0; i < 32 * 24; i++) {
    if (frame[i] < tempMinRead) {
      tempMinRead = frame[i];
    }
    if (frame[i] > tempMaxRead) {
      tempMaxRead = frame[i];
    }
    // Get the temperature at center
    if (i == 32 * 12 + 16) {
      tempCenter = frame[i];
    }
  }
  sprite.fillSprite(TFT_BLACK);
  if (interpolationMode == 0) {
    // Nearest neighbor interpolation
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      for (int x = 0; x < 320; x++) {
        float val = frame[yIndex + (x / 10)];
        drawPixel(319 - x, y, val);
      }
    }
  } else if (interpolationMode == 1) {
    // Average Interpolation
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      int yNextIndex = ((y / 10) + 1) * 32;  // Next row in original data
      for (int x = 0; x < 320; x++) {
        int xIndex = x / 10;
        // Take average of current and next points in x and y
        float val = (frame[yIndex + xIndex] + frame[yIndex + xIndex + 1] + frame[yNextIndex + xIndex] + frame[yNextIndex + xIndex + 1]) / 4.0;
        drawPixel(319 - x, y, val);
      }
    }
  } else if (interpolationMode == 2) {
    // Bilinear interpolation
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      float y_ratio = (y % 10) / 10.0;
      float y_opposite_ratio = 1 - y_ratio;
      for (int x = 0; x < 320; x++) {
        float x_ratio = (x % 10) / 10.0;
        float x_opposite_ratio = 1 - x_ratio;
        int x_over_10 = x / 10;
        float val = y_opposite_ratio * (x_opposite_ratio * frame[yIndex + x_over_10] + x_ratio * frame[yIndex + x_over_10 + 1]) + y_ratio * (x_opposite_ratio * frame[(yIndex + 32) + x_over_10] + x_ratio * frame[(yIndex + 32) + x_over_10 + 1]);
        drawPixel(319 - x, y, val);
      }
    }
  } else if (interpolationMode == 3) {
    // Bilinear interpolation
    int yIndex;
    int xIndex;
    int yNextIndex;
    for (int y = 0; y < 240; y++) {
      yIndex = (y / 10) * 32;
      yNextIndex = yIndex + 32;
      for (int x = 0; x < 320; x++) {
        xIndex = x / 10;
        float val = yOppositeRatios[y] * (xOppositeRatios[x] * frame[yIndex + xIndex] + xRatios[x] * frame[yIndex + xIndex + 1]) + yRatios[y] * (xOppositeRatios[x] * frame[yNextIndex + xIndex] + xRatios[x] * frame[yNextIndex + xIndex + 1]);
        drawPixel(319 - x, y, val);
      }
    }
  }
  else if (interpolationMode == 4) {
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      int yNextIndex = ((y / 10) + 1) * 32;  // Next row in original data
      for (int x = 0; x < 320; x++) {
        int xIndex = x / 10;
        // Determine which triangle the point is in and interpolate accordingly
        float t = (x % 10) / 10.0f;  // Horizontal distance from left pixel center
        float u = (y % 10) / 10.0f;  // Vertical distance from top pixel center
        float val;
        if (t > u) {  // Point is in lower-left triangle
          // Interpolate between bottom-left, top-right, and bottom-right
          val = (1 - t) * frame[yNextIndex + xIndex] + (1 - u) * frame[yIndex + xIndex + 1] + (t + u - 1) * frame[yNextIndex + xIndex + 1];
        } else {  // Point is in upper-right triangle
          // Interpolate between top-left, bottom-right, and top-right
          val = (1 - t) * frame[yIndex + xIndex] + (1 - u) * frame[yNextIndex + xIndex + 1] + (t + u - 1) * frame[yIndex + xIndex + 1];
        }
        drawPixel(319 - x, y, val);
      }
    }
  }
  // Display the maximum, minimum and center temperatures as overlay
  sprite.setTextColor(TFT_BLACK);
  sprite.setTextSize(1);
  sprite.setFreeFont(&Open_Sans_ExtraBold_10);
  sprite.drawString("T Min: " + String(tempMinRead) + " C", 15, 220);
  sprite.drawString("T Max: " + String(tempMaxRead) + " C", 220, 220);
  sprite.drawString("Tc: " + String(tempCenter, 1) + "C", 135, 220);
  // Display a small crosshair at the center
  sprite.drawLine(155, 120, 165, 120, TFT_WHITE);
  sprite.drawLine(160, 115, 160, 125, TFT_WHITE);
  detachInterrupt(digitalPinToInterrupt(downButton));
  batv = ((float)analogRead(VBAT_PIN) / 4095) * 2 * 1.07 * 3.3;
  attachInterrupt(digitalPinToInterrupt(downButton), downButton_ISR, CHANGE);
  int batpc = (uint8_t)(((batv - BATTV_MIN) / (BATTV_MAX - BATTV_MIN)) * 100);
  drawBattery(batpc);
  if (Menu == 1) {
    MenuChange = false;
    sprite.fillRect(60, 40, 200, 120, TFT_BLACK);
    sprite.fillRect(60, 40, 200, 10, TFT_WHITE);
    sprite.setTextFont(0);
    sprite.setTextSize(1);
    sprite.setTextColor(TFT_BLACK);
    sprite.drawString("Settings", 140, 41);
    sprite.setTextColor(TFT_WHITE);
    for (int i = 0; i < 7; i++) {
      if (i == Menuitem) sprite.setTextColor(TFT_GREEN);
      int y = 55 + (15 * i);
      sprite.drawString(menuItems[i], 65, y);
      if (i == 0) {
        sprite.drawString(ASindex[AutoScale], 160, y);
      } else if (i == 1) {
        sprite.drawString(String(MinT) + " C", 160, y);
      } else if (i == 2) {
        sprite.drawString(String(MaxT) + " C", 160, y);
      } else if (i == 3) {
        sprite.drawString(IPindex[interpolationMode], 160, y);
      } else if (i == 4) {
        sprite.drawString("Palette " + String(paletteIndex), 160, y);
      } else if (i == 5) {
        sprite.drawString(RRindex[RefreshRate], 160, y);
      } else if (i == 6) {
        sprite.drawString(String(BLPWM) + " %", 160, y);
      }
      sprite.setTextColor(TFT_WHITE);
    }
  }
  sprite.pushSprite(0, 0);
}
void drawPixel(int x, int y, float val) {
  int colorIndex;
  if (AutoScale == 1) {
    if (val <= tempMin) {
      colorIndex = 0;
    } else if (val >= tempMax) {
      colorIndex = 5;
    } else {
      // Map the value to a color index between 0 and 5
      colorIndex = int(map(val, tempMin, tempMax, 0, 5));
    }
  } else {
    if (val <= MinT) {
      colorIndex = 0;
    } else if (val >= MaxT) {
      colorIndex = 5;
    } else {
      // Map the value to a color index between 0 and 5
      colorIndex = int(map(val, MinT, MaxT, 0, 5));
    }
  }
  int color = colorPalettes[paletteIndex][colorIndex];
  sprite.drawPixel(x, y, color);
}
void drawBattery(int batpc) {
  int x = 290;
  int y = 10;
  int w = 20;
  int h = 10;
  int color = TFT_RED;  // Default color for the lowest level
  // Determine fill level and color based on batpc
  int fillLevel = 0;
  if (batpc > 75) {
    fillLevel = w;  // 100%
    color = TFT_GREEN;
  } else if (batpc > 50) {
    fillLevel = w * 3 / 4;  // 75%
    color = TFT_GREEN;
  } else if (batpc > 25) {
    fillLevel = w / 2;  // 50%
    color = TFT_BLUE;
  } else if (batpc > 0) {
    fillLevel = w / 4;  // 25%
    color = TFT_BLUE;
  }
  // Draw battery outline
  sprite.drawRect(x, y, w, h, TFT_WHITE);
  sprite.drawRect(x + w, y + h / 4, 2, h / 2, TFT_WHITE);
  // Draw fill level
  if (fillLevel > 0) {
    sprite.fillRect(x + 1, y + 1, fillLevel - 2, h - 2, color);
  }
}

The GIFDraw function is used to process and display the animated image file. It uses AnimatedGIF library for this purpose.

void GIFDraw(GIFDRAW *pDraw) {
  uint8_t *s;
  uint16_t *d, *usPalette;
  int x, y, iWidth;
  iWidth = pDraw->iWidth;
  if (iWidth + pDraw->iX > TFT_WIDTH)
    iWidth = TFT_WIDTH - pDraw->iX;
  usPalette = pDraw->pPalette;
  y = pDraw->iY + pDraw->y;
  if (y >= TFT_HEIGHT || pDraw->iX >= TFT_WIDTH || iWidth < 1)
    return;
  if (pDraw->ucDisposalMethod == 2) {
    for (x = 0; x < iWidth; x++) {
      if (s[x] == pDraw->ucTransparent)
        s[x] = pDraw->ucBackground;
    }
    pDraw->ucHasTransparency = 0;
  }
  s = pDraw->pPixels;
  if (pDraw->ucHasTransparency) {
    uint8_t *pEnd, c, ucTransparent = pDraw->ucTransparent;
    pEnd = s + iWidth;
    x = 0;
    while (x < iWidth) {
      c = ucTransparent - 1;
      d = &usTemp[0];
      while (c != ucTransparent && s < pEnd) {
        c = *s++;
        if (c == ucTransparent) {
          s--;
        } else {
          *d++ = usPalette[c];
        }
      }
      if (d > &usTemp[0]) {
        sprite.pushImage(pDraw->iX + x, y, d - &usTemp[0], 1, usTemp);  // Push the image to the sprite
        x += d - &usTemp[0];
      }
      c = ucTransparent;
      while (c == ucTransparent && s < pEnd) {
        c = *s++;
        if (c == ucTransparent)
          x++;
        else
          s--;
      }
    }
  } else {
    s = pDraw->pPixels;
    for (x = 0; x < iWidth; x++) {
      usTemp[x] = usPalette[*s++];
    }
    sprite.pushImage(pDraw->iX, y, iWidth, 1, usTemp);  // Push the image to the sprite
  }
}

Last but not least the setup and loop functions. As usual the setup function is used to initialise and setup all the libraries and configurations at the startup. The setup function will also recall all the configuration parameters from the namespace saved in the NVS flash area. The loop function will call the navigationUpdate and displayUpdate functions continuously for the smoother operation.

void setup() {
  Serial.begin(115200);
  tft.init();
#ifdef USE_DMA
  tft.initDMA();
#endif
  tft.setRotation(1);
  tft.fillScreen(TFT_BLACK);
  gif.begin(BIG_ENDIAN_PIXELS);
  sprite.createSprite(320, 240);
  pinMode(upButton, INPUT_PULLUP);
  attachInterrupt(digitalPinToInterrupt(upButton), upButton_ISR, CHANGE);
  pinMode(downButton, INPUT_PULLUP);
  attachInterrupt(digitalPinToInterrupt(downButton), downButton_ISR, CHANGE);
  pinMode(middleButton, INPUT_PULLUP);
  attachInterrupt(digitalPinToInterrupt(middleButton), middleButton_ISR, CHANGE);
  ReadConfig();
  MLXInit();
  if (MinT == 0 && MaxT == 0) {
    MinT = 20.0;
    MaxT = 32.0;
    WriteConfig();
  }
  if (BLPWM == 0) {
    BLPWM = 100;
    WriteConfig();
  }
  // mlx.setRefreshRate(MLX90640_32_HZ);
  pinMode(backlightPin, OUTPUT);
  analogWrite(backlightPin, map(BLPWM, 0, 100, 0, 255));  // Turn on backlight
  if (gif.open((uint8_t *)GIF_IMAGE, sizeof(GIF_IMAGE), GIFDraw)) {
    Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
    while (gif.playFrame(true, NULL)) {
      sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
      yield();
    }
    gif.close();
  }
  delay(1000);
  initSDcard();
  // Precompute ratios
  for (int x = 0; x < 320; x++) {
    xRatios[x] = (x % 10) / 10.0f;
    xOppositeRatios[x] = 1 - xRatios[x];
  }
  for (int y = 0; y < 240; y++) {
    yRatios[y] = (y % 10) / 10.0f;
    yOppositeRatios[y] = 1 - yRatios[y];
  }
}
void loop() {
  navigationUpdate();
  displayUpdate();
}

Supporting Files

You can download all the necessary files from the Circuit Digest GitHub repo, from the following link.

Complete Project Code

/*
 * Project Name: MLX90640 Thermal Camera
 * Project Brief: Firmware for Thermal camera built around ESP32 and MLX90640
 * Author: Jobit Joseph
 * Copyright © Jobit Joseph
 * Copyright © Semicon Media Pvt Ltd
 * Copyright © Circuitdigest.com
 * 
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, in version 3.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program. If not, see .
 *
 */
#include 
#include 
#include 
#include 
#include "FS.h"
#include 
#include 
#include 
#include   //Include a library of Fonts
#include "Open_Sans_ExtraBold_10.h"
#include "BootAnimation.h"
#include "success.h"
#include "Error.h"
SPIClass spiSD(HSPI);
#define SD_CS 15
//#define USE_DMA
#define NORMAL_SPEED
#define TFT_WIDTH 320
#define TFT_HEIGHT 240
#define BUFFER_SIZE 320
uint16_t usTemp[BUFFER_SIZE];
#define VBAT_PIN 33
#define BATTV_MAX 4.2  // maximum voltage of battery
#define BATTV_MIN 3.2  // what we regard as an empty battery
#define GIF_IMAGE BootAnimationIMG
#define SGIF_IMAGE success_GIF
#define EGIF_IMAGE Error_GIF
bool dmaBuf = 0;
Adafruit_MLX90640 mlx;
Preferences preferences;
AnimatedGIF gif;
TFT_eSPI tft = TFT_eSPI();
File bmpFile;
float frame[32 * 24];
float batv;
const int upButton = 35;
const int middleButton = 36;
const int downButton = 39;
const int backlightPin = 4;
int interpolationMode = 0;
int AutoScale = 0;
volatile bool MenuChange = false;
volatile bool _upShort = false;
volatile bool _downShort = false;
volatile bool middleShort = false;
volatile bool middlePressed = false;
unsigned long middlePressStartTime = 0;
int Menu = 0, Menuitem = 0;
int BLPWM = 0;
int RefreshRate = 0;
float MinT, MaxT;
String menuItems[7] = { "Auto Scale    : ", "Min Temp      : ", "Max Temp      : ", "Interpolation : ", "Palette       : ", "Refresh Rate  : ", "BackLight     : " };
String ASindex[2] = { "Off", "On" };
String IPindex[6] = { "Nearest Neighbor", "Average", "Bilinear", "Bilinear Fast", "Triangle" };
String RRindex[4] = { "4 Hz", "8 Hz", "16 Hz", "32 Hz" };
float xRatios[320];
float yRatios[240];
float xOppositeRatios[320];
float yOppositeRatios[240];
#define PALETTE_COUNT 10
uint16_t colorPalettes[PALETTE_COUNT][6] = {
  { TFT_BLUE, TFT_CYAN, TFT_GREEN, TFT_YELLOW, TFT_RED, TFT_MAGENTA },
  { TFT_BLACK, TFT_DARKGREY, TFT_LIGHTGREY, TFT_WHITE, TFT_ORANGE, TFT_PINK },
  { TFT_NAVY, TFT_OLIVE, TFT_DARKGREEN, TFT_DARKCYAN, TFT_MAROON, TFT_PURPLE },
  { TFT_BLUE, TFT_GREEN, TFT_DARKGREEN, TFT_ORANGE, TFT_MAROON, TFT_RED },
  { TFT_NAVY, TFT_DARKGREEN, TFT_GREEN, TFT_YELLOW, TFT_ORANGE, TFT_RED },
  { TFT_CYAN, TFT_BLUE, TFT_MAGENTA, TFT_YELLOW, TFT_GREEN, TFT_RED },
  { TFT_WHITE, TFT_ORANGE, TFT_RED, TFT_BLUE, TFT_GREEN, TFT_BLACK },
  { TFT_PURPLE, TFT_MAGENTA, TFT_RED, TFT_ORANGE, TFT_YELLOW, TFT_GREEN },
  { TFT_YELLOW, TFT_PINK, TFT_WHITE, TFT_BLUE, TFT_DARKCYAN, TFT_DARKGREEN },
  { TFT_RED, TFT_YELLOW, TFT_GREEN, TFT_CYAN, TFT_BLUE, TFT_MAGENTA }
};
int paletteIndex = 0;

float tempMin = 20.0;  // Minimum temperature
float tempMax = 32.0;  // Maximum temperature

TFT_eSprite sprite = TFT_eSprite(&tft);


void IRAM_ATTR upButton_ISR() {
  static unsigned long last_interrupt_time = 0;
  unsigned long interrupt_time = millis();

  if (interrupt_time - last_interrupt_time > 200) {  // simple debounce
    _upShort = true;
  }
  last_interrupt_time = interrupt_time;
}

void IRAM_ATTR downButton_ISR() {
  static unsigned long last_interrupt_time = 0;
  unsigned long interrupt_time = millis();

  if (interrupt_time - last_interrupt_time > 200) {  // simple debounce
    _downShort = true;
  }
  last_interrupt_time = interrupt_time;
}

void IRAM_ATTR middleButton_ISR() {
  if (digitalRead(middleButton) == LOW) {
    // Button press event
    if (!middlePressed) {  // If button was not already being pressed
      middlePressed = true;
      middlePressStartTime = millis();  // Save the start time of button press
    }
  } else {
    // Button release event
    if (middlePressed) {                             // If the button was being pressed
      if (millis() - middlePressStartTime < 1000) {  // If the button was pressed for less than 1 second
        middleShort = true;
      }
      middlePressed = false;
      middlePressStartTime = 0;  // Reset the start time of button press
    }
  }
}



void MLXInit() {

  Wire.begin();
  Wire.setClock(1000000);
  if (!mlx.begin()) {
    Serial.println("Failed to initialize MLX90640!");
  }
  mlx.setResolution(MLX90640_ADC_18BIT);
  ConfigRefreshrate();
}
void ConfigRefreshrate() {
  switch (RefreshRate) {
    case 1:
      mlx.setRefreshRate(MLX90640_4_HZ);
      break;
    case 2:
      mlx.setRefreshRate(MLX90640_8_HZ);
      break;
    case 3:
      mlx.setRefreshRate(MLX90640_16_HZ);
      break;
    case 4:
      mlx.setRefreshRate(MLX90640_32_HZ);
      break;
    default:
      break;
  }
}
void ReadConfig() {
  preferences.begin("Config", false);
  String temp;
  temp = preferences.getString("AutoScale", "");
  AutoScale = temp.toInt();
  temp = preferences.getString("MinTe", "");
  MinT = temp.toFloat();
  temp = preferences.getString("MaxTe", "");
  MaxT = temp.toFloat();
  temp = preferences.getString("Imode", "");
  interpolationMode = temp.toInt();
  temp = preferences.getString("PIndex", "");
  paletteIndex = temp.toInt();
  temp = preferences.getString("RefreshRate", "");
  RefreshRate = temp.toInt();
  temp = preferences.getString("BLPWM", "");
  BLPWM = temp.toInt();
  preferences.end();
}
void WriteConfig() {
  preferences.begin("Config", false);
  preferences.putString("AutoScale", String(AutoScale));
  preferences.putString("MinTe", String(MinT));
  preferences.putString("MaxTe", String(MaxT));
  preferences.putString("Imode", String(interpolationMode));
  preferences.putString("PIndex", String(paletteIndex));
  preferences.putString("RefreshRate", String(RefreshRate));
  preferences.putString("BLPWM", String(BLPWM));
  preferences.end();
}

void initSDcard() {
  //spiSD.begin(14, 12, 13, SD_CS);  //CLK,MISO,MOIS,SS
  if (!SD.begin(SD_CS, spiSD)) {
    Serial.println("Card Mount Failed");
    return;
  } else {
    Serial.println("Card Mount Successful");
  }
  uint8_t cardType = SD.cardType();

  if (cardType == CARD_NONE) {
    Serial.println("No SD card attached");
    return;
  }
}
String generateFilename(fs::FS &fs) {
  for (int i = 0; i <= 9999; i++) {
    char filename[23];                               // Allocate the char array
    sprintf(filename, "/ThermalCamera%04d.bmp", i);  // Print formatted string into char array

    if (!fs.exists(filename)) {
      return String(filename);  // Return a String object
    }
  }

  return "";  // return empty string if all filenames are taken
}
int writeBMP(fs::FS &fs, const char *path, TFT_eSprite *sprite) {
  const int width = sprite->width();
  const int height = sprite->height();

  // BMP file header (14 bytes)
  uint8_t bmpFileHeader[14] = { 'B', 'M', 0, 0, 0, 0, 0, 0, 0, 0, 54, 0, 0, 0 };

  // The size of the BMP file in bytes
  uint32_t fileSize = 54 + width * height * 2;
  bmpFileHeader[2] = (uint8_t)(fileSize);
  bmpFileHeader[3] = (uint8_t)(fileSize >> 8);
  bmpFileHeader[4] = (uint8_t)(fileSize >> 16);
  bmpFileHeader[5] = (uint8_t)(fileSize >> 24);

  // BMP info header (40 bytes)
  uint8_t bmpInfoHeader[40] = { 40, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 16, 0 };

  bmpInfoHeader[4] = (uint8_t)(width);
  bmpInfoHeader[5] = (uint8_t)(width >> 8);
  bmpInfoHeader[6] = (uint8_t)(width >> 16);
  bmpInfoHeader[7] = (uint8_t)(width >> 24);

  bmpInfoHeader[8] = (uint8_t)(height);
  bmpInfoHeader[9] = (uint8_t)(height >> 8);
  bmpInfoHeader[10] = (uint8_t)(height >> 16);
  bmpInfoHeader[11] = (uint8_t)(height >> 24);

  File file = fs.open(path, FILE_WRITE);
  if (!file) {
    Serial.println("Failed to open file for writing");
    return 0;
  } else {

    //showSavingImageMessage();
    // Write headers
    file.write(bmpFileHeader, 14);
    file.write(bmpInfoHeader, 40);

    // Write pixel data
    for (int y = height - 1; y >= 0; y--) {  // BMP is stored bottom-top
      for (int x = 0; x < width; x++) {
        uint16_t pixel = sprite->readPixel(x, y);
        // Swap red and green channels
        uint16_t r = (pixel >> 11) & 0x1F;
        uint16_t g = (pixel >> 5) & 0x3F;
        uint16_t b = pixel & 0x1F;
        pixel = (b << 11) | (r << 5) | g;
        file.write(pixel >> 8);  // high byte
        file.write(pixel);       // low byte
      }
    }
    file.close();
    return 1;
  }
}


void navigationUpdate() {
  if (_upShort == true) {
    _upShort = false;
    if (Menu == 0) {
      interpolationMode++;
      if (interpolationMode > 4) interpolationMode = 0;
    } else {
      if (Menuitem == 0) {
        AutoScale++;
        if (AutoScale > 1) AutoScale = 1;
      } else if (Menuitem == 1) {
        MinT++;
        if (MinT > 300) MinT = 300;
      } else if (Menuitem == 2) {
        MaxT++;
        if (MaxT > 300) MaxT = 300;
      } else if (Menuitem == 3) {
        interpolationMode++;
        if (interpolationMode > 4) interpolationMode = 4;
      } else if (Menuitem == 4) {
        paletteIndex++;
        if (paletteIndex > 9) paletteIndex = 9;
      } else if (Menuitem == 5) {
        RefreshRate++;
        if (RefreshRate > 3) RefreshRate = 3;
        ConfigRefreshrate();
      } else if (Menuitem == 6) {
        BLPWM = BLPWM + 10;
        if (BLPWM > 100) BLPWM = 100;
        analogWrite(backlightPin, map(BLPWM, 0, 100, 0, 255));  // Turn on backlight
      }
      MenuChange = true;
    }
    WriteConfig();
    Serial.println("Up Short Press");
  }
  if (_downShort == true) {
    _downShort = false;
    if (Menu == 0) {
      paletteIndex = (paletteIndex + 1) % PALETTE_COUNT;
    } else {
      if (Menuitem == 0) {
        AutoScale--;
        if (AutoScale < 0) AutoScale = 0;
      } else if (Menuitem == 1) {
        MinT--;
        if (MinT < 0) MinT = 0;
      } else if (Menuitem == 2) {
        MaxT--;
        if (MaxT < 5) MaxT = 5;
      } else if (Menuitem == 3) {
        interpolationMode--;
        if (interpolationMode < 0) interpolationMode = 0;
      } else if (Menuitem == 4) {
        paletteIndex--;
        if (paletteIndex < 0) paletteIndex = 0;
      } else if (Menuitem == 5) {
        RefreshRate--;
        if (RefreshRate < 0) RefreshRate = 0;
        ConfigRefreshrate();
      } else if (Menuitem == 6) {
        BLPWM = BLPWM - 10;
        if (BLPWM < 10) BLPWM = 10;
        analogWrite(backlightPin, map(BLPWM, 0, 100, 0, 255));  // Turn on backlight
      }
      MenuChange = true;
    }
    WriteConfig();
    Serial.println("down Short Press");
  }
  if (middleShort == true) {
    middleShort = false;
    if (Menu == 0) {
      if (!SD.begin(SD_CS, spiSD)) {
        Serial.println("Card Mount Failed");
        if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
          Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
          while (gif.playFrame(true, NULL)) {
            sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
            yield();
          }
          gif.close();
        }
        return;
      } else {
        Serial.println("Card Mount Successful");
      }
      uint8_t cardType = SD.cardType();

      if (cardType == CARD_NONE) {
        Serial.println("No SD card attached");

        if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
          Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
          while (gif.playFrame(true, NULL)) {
            sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
            yield();
          }
          gif.close();
        }
        return;
      } else {
        String filename = generateFilename(SD);
        if (filename != "") {
          if (writeBMP(SD, filename.c_str(), &sprite)) {

            if (gif.open((uint8_t *)SGIF_IMAGE, sizeof(SGIF_IMAGE), GIFDraw1)) {
              Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
              while (gif.playFrame(true, NULL)) {
                sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
                yield();
              }
              gif.close();
            }
          } else {

            if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
              Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
              while (gif.playFrame(true, NULL)) {
                sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
                yield();
              }
              gif.close();
            }
          }
        } else {
          Serial.println("Failed to create filename.");

          if (gif.open((uint8_t *)EGIF_IMAGE, sizeof(EGIF_IMAGE), GIFDraw1)) {
            Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
            while (gif.playFrame(true, NULL)) {
              sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
              yield();
            }
            gif.close();
          }
        }
      }
    } else {
      Menuitem++;
      if (Menuitem > 6) {
        Menuitem = 0;
      }
      MenuChange = true;
    }
    Serial.println("Middle Short Press");
    Serial.print(Menu);
    Serial.print("");
    Serial.println(Menuitem);
  }
  if (middlePressed && !middleShort && millis() - middlePressStartTime > 1000 && middlePressed && !middleShort && millis() - middlePressStartTime < 1500) {
    // If the button is being pressed, no short press has been registered, and it has been over 1 second
    Serial.println("Middle Long Press");
    middlePressed = false;     // Reset the pressed flag
    middlePressStartTime = 0;  // Reset the press start time
    if (Menu == 0) {
      Menu = 1;
      MenuChange = true;
      Menuitem = 0;
    } else {
      Menu = 0;
    }
  }
}

void displayUpdate() {

  if (!mlx.getFrame(frame)) {
    // Failed to get frame, so reinitialize
    MLXInit();
  }

  float tempMinRead = 1000;   // some high value
  float tempMaxRead = -1000;  // some low value
  float tempCenter = 0.0;     // Temperature at center

  // Update the minimum and maximum temperatures read from the sensor
  for (int i = 0; i < 32 * 24; i++) {
    if (frame[i] < tempMinRead) {
      tempMinRead = frame[i];
    }
    if (frame[i] > tempMaxRead) {
      tempMaxRead = frame[i];
    }
    // Get the temperature at center
    if (i == 32 * 12 + 16) {
      tempCenter = frame[i];
    }
  }


  sprite.fillSprite(TFT_BLACK);

  if (interpolationMode == 0) {
    // Nearest neighbor interpolation
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      for (int x = 0; x < 320; x++) {
        float val = frame[yIndex + (x / 10)];
        drawPixel(319 - x, y, val);
      }
    }
  } else if (interpolationMode == 1) {
    // Average Interpolation
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      int yNextIndex = ((y / 10) + 1) * 32;  // Next row in original data
      for (int x = 0; x < 320; x++) {
        int xIndex = x / 10;

        // Take average of current and next points in x and y
        float val = (frame[yIndex + xIndex] + frame[yIndex + xIndex + 1] + frame[yNextIndex + xIndex] + frame[yNextIndex + xIndex + 1]) / 4.0;

        drawPixel(319 - x, y, val);
      }
    }
  } else if (interpolationMode == 2) {
    // Bilinear interpolation
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      float y_ratio = (y % 10) / 10.0;
      float y_opposite_ratio = 1 - y_ratio;
      for (int x = 0; x < 320; x++) {
        float x_ratio = (x % 10) / 10.0;
        float x_opposite_ratio = 1 - x_ratio;
        int x_over_10 = x / 10;
        float val = y_opposite_ratio * (x_opposite_ratio * frame[yIndex + x_over_10] + x_ratio * frame[yIndex + x_over_10 + 1]) + y_ratio * (x_opposite_ratio * frame[(yIndex + 32) + x_over_10] + x_ratio * frame[(yIndex + 32) + x_over_10 + 1]);
        drawPixel(319 - x, y, val);
      }
    }
  } else if (interpolationMode == 3) {
    // Bilinear interpolation
    int yIndex;
    int xIndex;
    int yNextIndex;

    for (int y = 0; y < 240; y++) {
      yIndex = (y / 10) * 32;
      yNextIndex = yIndex + 32;
      for (int x = 0; x < 320; x++) {
        xIndex = x / 10;
        float val = yOppositeRatios[y] * (xOppositeRatios[x] * frame[yIndex + xIndex] + xRatios[x] * frame[yIndex + xIndex + 1]) + yRatios[y] * (xOppositeRatios[x] * frame[yNextIndex + xIndex] + xRatios[x] * frame[yNextIndex + xIndex + 1]);
        drawPixel(319 - x, y, val);
      }
    }
  }

  else if (interpolationMode == 4) {
    for (int y = 0; y < 240; y++) {
      int yIndex = (y / 10) * 32;
      int yNextIndex = ((y / 10) + 1) * 32;  // Next row in original data
      for (int x = 0; x < 320; x++) {
        int xIndex = x / 10;

        // Determine which triangle the point is in and interpolate accordingly
        float t = (x % 10) / 10.0f;  // Horizontal distance from left pixel center
        float u = (y % 10) / 10.0f;  // Vertical distance from top pixel center
        float val;
        if (t > u) {  // Point is in lower-left triangle
          // Interpolate between bottom-left, top-right, and bottom-right
          val = (1 - t) * frame[yNextIndex + xIndex] + (1 - u) * frame[yIndex + xIndex + 1] + (t + u - 1) * frame[yNextIndex + xIndex + 1];
        } else {  // Point is in upper-right triangle
          // Interpolate between top-left, bottom-right, and top-right
          val = (1 - t) * frame[yIndex + xIndex] + (1 - u) * frame[yNextIndex + xIndex + 1] + (t + u - 1) * frame[yIndex + xIndex + 1];
        }

        drawPixel(319 - x, y, val);
      }
    }
  }



  // Display the maximum, minimum and center temperatures as overlay
  sprite.setTextColor(TFT_BLACK);
  sprite.setTextSize(1);

  sprite.setFreeFont(&Open_Sans_ExtraBold_10);

  sprite.drawString("T Min: " + String(tempMinRead) + " C", 15, 220);
  sprite.drawString("T Max: " + String(tempMaxRead) + " C", 220, 220);
  sprite.drawString("Tc: " + String(tempCenter, 1) + "C", 135, 220);
  // Display a small crosshair at the center
  sprite.drawLine(155, 120, 165, 120, TFT_WHITE);
  sprite.drawLine(160, 115, 160, 125, TFT_WHITE);

  detachInterrupt(digitalPinToInterrupt(downButton));
  batv = ((float)analogRead(VBAT_PIN) / 4095) * 2 * 1.07 * 3.3;
  attachInterrupt(digitalPinToInterrupt(downButton), downButton_ISR, CHANGE);
  int batpc = (uint8_t)(((batv - BATTV_MIN) / (BATTV_MAX - BATTV_MIN)) * 100);
  drawBattery(batpc);
  if (Menu == 1) {
    MenuChange = false;
    sprite.fillRect(60, 40, 200, 120, TFT_BLACK);
    sprite.fillRect(60, 40, 200, 10, TFT_WHITE);
    sprite.setTextFont(0);
    sprite.setTextSize(1);
    sprite.setTextColor(TFT_BLACK);
    sprite.drawString("Settings", 140, 41);
    sprite.setTextColor(TFT_WHITE);
    for (int i = 0; i < 7; i++) {
      if (i == Menuitem) sprite.setTextColor(TFT_GREEN);
      int y = 55 + (15 * i);
      sprite.drawString(menuItems[i], 65, y);
      if (i == 0) {
        sprite.drawString(ASindex[AutoScale], 160, y);
      } else if (i == 1) {
        sprite.drawString(String(MinT) + " C", 160, y);
      } else if (i == 2) {
        sprite.drawString(String(MaxT) + " C", 160, y);
      } else if (i == 3) {
        sprite.drawString(IPindex[interpolationMode], 160, y);
      } else if (i == 4) {
        sprite.drawString("Palette " + String(paletteIndex), 160, y);
      } else if (i == 5) {
        sprite.drawString(RRindex[RefreshRate], 160, y);
      } else if (i == 6) {
        sprite.drawString(String(BLPWM) + " %", 160, y);
      }
      sprite.setTextColor(TFT_WHITE);
    }
  }
  sprite.pushSprite(0, 0);
}

void drawPixel(int x, int y, float val) {
  int colorIndex;
  if (AutoScale == 1) {
    if (val <= tempMin) {
      colorIndex = 0;
    } else if (val >= tempMax) {
      colorIndex = 5;
    } else {
      // Map the value to a color index between 0 and 5
      colorIndex = int(map(val, tempMin, tempMax, 0, 5));
    }
  } else {
    if (val <= MinT) {
      colorIndex = 0;
    } else if (val >= MaxT) {
      colorIndex = 5;
    } else {
      // Map the value to a color index between 0 and 5
      colorIndex = int(map(val, MinT, MaxT, 0, 5));
    }
  }
  int color = colorPalettes[paletteIndex][colorIndex];
  sprite.drawPixel(x, y, color);
}
void drawBattery(int batpc) {
  int x = 290;
  int y = 10;
  int w = 20;
  int h = 10;
  int color = TFT_RED;  // Default color for the lowest level


  // Determine fill level and color based on batpc
  int fillLevel = 0;
  if (batpc > 75) {
    fillLevel = w;  // 100%
    color = TFT_GREEN;
  } else if (batpc > 50) {
    fillLevel = w * 3 / 4;  // 75%
    color = TFT_GREEN;
  } else if (batpc > 25) {
    fillLevel = w / 2;  // 50%
    color = TFT_BLUE;
  } else if (batpc > 0) {
    fillLevel = w / 4;  // 25%
    color = TFT_BLUE;
  }


  // Draw battery outline
  sprite.drawRect(x, y, w, h, TFT_WHITE);
  sprite.drawRect(x + w, y + h / 4, 2, h / 2, TFT_WHITE);
  // Draw fill level
  if (fillLevel > 0) {
    sprite.fillRect(x + 1, y + 1, fillLevel - 2, h - 2, color);
  }
}


void GIFDraw(GIFDRAW *pDraw) {
  uint8_t *s;
  uint16_t *d, *usPalette;
  int x, y, iWidth;

  iWidth = pDraw->iWidth;
  if (iWidth + pDraw->iX > TFT_WIDTH)
    iWidth = TFT_WIDTH - pDraw->iX;

  usPalette = pDraw->pPalette;
  y = pDraw->iY + pDraw->y;

  if (y >= TFT_HEIGHT || pDraw->iX >= TFT_WIDTH || iWidth < 1)
    return;

  if (pDraw->ucDisposalMethod == 2) {
    for (x = 0; x < iWidth; x++) {
      if (s[x] == pDraw->ucTransparent)
        s[x] = pDraw->ucBackground;
    }
    pDraw->ucHasTransparency = 0;
  }

  s = pDraw->pPixels;
  if (pDraw->ucHasTransparency) {
    uint8_t *pEnd, c, ucTransparent = pDraw->ucTransparent;
    pEnd = s + iWidth;
    x = 0;
    while (x < iWidth) {
      c = ucTransparent - 1;
      d = &usTemp[0];
      while (c != ucTransparent && s < pEnd) {
        c = *s++;
        if (c == ucTransparent) {
          s--;
        } else {
          *d++ = usPalette[c];
        }
      }
      if (d > &usTemp[0]) {
        sprite.pushImage(pDraw->iX + x, y, d - &usTemp[0], 1, usTemp);  // Push the image to the sprite
        x += d - &usTemp[0];
      }
      c = ucTransparent;
      while (c == ucTransparent && s < pEnd) {
        c = *s++;
        if (c == ucTransparent)
          x++;
        else
          s--;
      }
    }
  } else {
    s = pDraw->pPixels;
    for (x = 0; x < iWidth; x++) {
      usTemp[x] = usPalette[*s++];
    }
    sprite.pushImage(pDraw->iX, y, iWidth, 1, usTemp);  // Push the image to the sprite
  }
}


void GIFDraw1(GIFDRAW *pDraw) {
  uint8_t *s;
  uint16_t *d, *usPalette;
  int x, y, iWidth;

  iWidth = pDraw->iWidth;
  if (iWidth + pDraw->iX > TFT_WIDTH)
    iWidth = TFT_WIDTH - pDraw->iX;

  usPalette = pDraw->pPalette;
  y = pDraw->iY + pDraw->y;

  if (y >= TFT_HEIGHT || pDraw->iX >= TFT_WIDTH || iWidth < 1)
    return;

  if (pDraw->ucDisposalMethod == 2) {
    for (x = 0; x < iWidth; x++) {
      if (s[x] == pDraw->ucTransparent)
        s[x] = pDraw->ucBackground;
    }
    pDraw->ucHasTransparency = 0;
  }

  s = pDraw->pPixels;
  if (pDraw->ucHasTransparency) {
    uint8_t *pEnd, c, ucTransparent = pDraw->ucTransparent;
    pEnd = s + iWidth;
    x = 0;
    while (x < iWidth) {
      c = ucTransparent - 1;
      d = &usTemp[0];
      while (c != ucTransparent && s < pEnd) {
        c = *s++;
        if (c == ucTransparent) {
          s--;
        } else {
          *d++ = usPalette[c];
        }
      }
      if (d > &usTemp[0]) {
        sprite.pushImage(pDraw->iX + x +80, y + 40, d - &usTemp[0], 1, usTemp);  // Push the image to the sprite
        x += d - &usTemp[0];
      }
      c = ucTransparent;
      while (c == ucTransparent && s < pEnd) {
        c = *s++;
        if (c == ucTransparent)
          x++;
        else
          s--;
      }
    }
  } else {
    s = pDraw->pPixels;
    for (x = 0; x < iWidth; x++) {
      usTemp[x] = usPalette[*s++];
    }
    sprite.pushImage(pDraw->iX + 80 , y + 40 , iWidth, 1, usTemp);  // Push the image to the sprite
  }
}



void setup() {
  Serial.begin(115200);
  tft.init();
#ifdef USE_DMA
  tft.initDMA();
#endif
  tft.setRotation(1);
  tft.fillScreen(TFT_BLACK);
  gif.begin(BIG_ENDIAN_PIXELS);
  sprite.createSprite(320, 240);

  pinMode(upButton, INPUT_PULLUP);
  attachInterrupt(digitalPinToInterrupt(upButton), upButton_ISR, CHANGE);

  pinMode(downButton, INPUT_PULLUP);
  attachInterrupt(digitalPinToInterrupt(downButton), downButton_ISR, CHANGE);

  pinMode(middleButton, INPUT_PULLUP);
  attachInterrupt(digitalPinToInterrupt(middleButton), middleButton_ISR, CHANGE);
  ReadConfig();
  MLXInit();
  if (MinT == 0 && MaxT == 0) {
    MinT = 20.0;
    MaxT = 32.0;
    WriteConfig();
  }
  if (BLPWM == 0) {
    BLPWM = 100;
    WriteConfig();
  }

  // mlx.setRefreshRate(MLX90640_32_HZ);
  pinMode(backlightPin, OUTPUT);
  analogWrite(backlightPin, map(BLPWM, 0, 100, 0, 255));  // Turn on backlight
  if (gif.open((uint8_t *)GIF_IMAGE, sizeof(GIF_IMAGE), GIFDraw)) {
    Serial.printf("Successfully opened GIF; Canvas size = %d x %d\n", gif.getCanvasWidth(), gif.getCanvasHeight());
    while (gif.playFrame(true, NULL)) {
      sprite.pushSprite(0, 0);  // Push the sprite to screen after every frame
      yield();
    }
    gif.close();
  }
  delay(1000);
  initSDcard();
  // Precompute ratios
  for (int x = 0; x < 320; x++) {
    xRatios[x] = (x % 10) / 10.0f;
    xOppositeRatios[x] = 1 - xRatios[x];
  }
  for (int y = 0; y < 240; y++) {
    yRatios[y] = (y % 10) / 10.0f;
    yOppositeRatios[y] = 1 - yRatios[y];
  }
}

void loop() {
  navigationUpdate();
  displayUpdate();
}
Video

Have any question realated to this Article?

Ask Our Community Members

Comments

a really lovely project. But I wish it wouldn't be on smd. The trade-off for size is something I would be willing to go for in terms of easier assembly by hand with cheaper equipment