326 lines
10 KiB
C++
326 lines
10 KiB
C++
#include "lcd.h"
|
|
|
|
// This example draws an animated dial with a rotating needle.
|
|
|
|
// The dial is a jpeg image, the needle is created using a rotated
|
|
// Sprite. The example operates by reading blocks of pixels from the
|
|
// TFT, thus the TFT setup must support reading from the TFT CGRAM.
|
|
|
|
// The sketch operates by creating a copy of the screen block where
|
|
// the needle will be drawn, the needle is then drawn on the screen.
|
|
// When the needle moves, the original copy of the sreen area is
|
|
// pushed to the screen to over-write the needle graphic. A copy
|
|
// of the screen where the new position will be drawn is then made
|
|
// before drawing the needle in the new postion. This technique
|
|
// allows the needle to move over other screen graphics.
|
|
|
|
// The sketch calculates the size of the buffer memory required and
|
|
// reserves the memory for the TFT block copy.
|
|
|
|
// Created by Bodmer 17/3/20 as an example to the TFT_eSPI library:
|
|
// https://github.com/Bodmer/TFT_eSPI
|
|
|
|
#define NEEDLE_LENGTH 35 // Visible length
|
|
#define NEEDLE_WIDTH 5 // Width of needle - make it an odd number
|
|
#define NEEDLE_RADIUS 90 // Radius at tip
|
|
#define NEEDLE_COLOR1 TFT_MAROON // Needle periphery colour
|
|
#define NEEDLE_COLOR2 TFT_RED // Needle centre colour
|
|
#define DIAL_CENTRE_X 120
|
|
#define DIAL_CENTRE_Y 120
|
|
|
|
// Font attached to this sketch
|
|
#include "NotoSansBold36.h"
|
|
#define AA_FONT_LARGE NotoSansBold36
|
|
|
|
#include <TFT_eSPI.h>
|
|
TFT_eSPI tft = TFT_eSPI();
|
|
TFT_eSprite needle = TFT_eSprite(&tft); // Sprite object for needle
|
|
TFT_eSprite spr = TFT_eSprite(&tft); // Sprite for meter reading
|
|
TFT_eSprite nameSpr = TFT_eSprite(&tft);
|
|
TFT_eSprite unitSpr = TFT_eSprite(&tft);
|
|
|
|
// Jpeg image array attached to this sketch
|
|
#include "dial.h"
|
|
|
|
// Include the jpeg decoder library
|
|
#include <TJpg_Decoder.h>
|
|
|
|
uint16_t *tft_buffer;
|
|
bool buffer_loaded = false;
|
|
uint16_t spr_width = 0;
|
|
uint16_t name_spr_width = 0;
|
|
|
|
void createNeedle(void);
|
|
void plotNeedle(int16_t angle, uint16_t ms_delay);
|
|
|
|
// =======================================================================================
|
|
// This function will be called during decoding of the jpeg file
|
|
// =======================================================================================
|
|
bool tft_output(int16_t x, int16_t y, uint16_t w, uint16_t h, uint16_t *bitmap)
|
|
{
|
|
// Stop further decoding as image is running off bottom of screen
|
|
if (y >= tft.height())
|
|
return 0;
|
|
|
|
// This function will clip the image block rendering automatically at the TFT boundaries
|
|
tft.pushImage(x, y, w, h, bitmap);
|
|
|
|
// Return 1 to decode next block
|
|
return 1;
|
|
}
|
|
|
|
// =======================================================================================
|
|
// Setup
|
|
// =======================================================================================
|
|
void initLCD()
|
|
{
|
|
Serial.print("InitLCD:");
|
|
// The byte order can be swapped (set true for TFT_eSPI)
|
|
TJpgDec.setSwapBytes(true);
|
|
|
|
// The jpeg decoder must be given the exact name of the rendering function above
|
|
TJpgDec.setCallback(tft_output);
|
|
|
|
tft.begin();
|
|
tft.setRotation(0);
|
|
tft.fillScreen(TFT_BLACK);
|
|
|
|
// Draw the dial
|
|
TJpgDec.drawJpg(0, 0, dial, sizeof(dial));
|
|
tft.drawCircle(DIAL_CENTRE_X, DIAL_CENTRE_Y, NEEDLE_RADIUS - NEEDLE_LENGTH, TFT_DARKGREY);
|
|
|
|
// Load the font and create the Sprite for reporting the value
|
|
spr.loadFont(AA_FONT_LARGE);
|
|
spr_width = spr.textWidth("277");
|
|
spr.createSprite(spr_width, spr.fontHeight());
|
|
uint16_t bg_color = tft.readPixel(120, 120); // Get colour from dial centre
|
|
spr.fillSprite(bg_color);
|
|
spr.setTextColor(TFT_WHITE, bg_color);
|
|
spr.setTextDatum(MC_DATUM);
|
|
spr.setTextPadding(spr_width);
|
|
spr.drawNumber(0, spr_width / 2, spr.fontHeight() / 2);
|
|
spr.pushSprite(DIAL_CENTRE_X - spr_width / 2, DIAL_CENTRE_Y - spr.fontHeight() / 2);
|
|
|
|
// Plot the label text
|
|
nameSpr.setTextFont(3);
|
|
name_spr_width = nameSpr.textWidth("Temperature");
|
|
nameSpr.createSprite(name_spr_width, nameSpr.fontHeight() * 2 + 2);
|
|
nameSpr.fillSprite(bg_color);
|
|
nameSpr.setTextColor(TFT_WHITE, bg_color);
|
|
nameSpr.setTextDatum(MC_DATUM);
|
|
nameSpr.setTextPadding(name_spr_width);
|
|
nameSpr.drawString("Sensor Name", name_spr_width / 2, nameSpr.fontHeight() / 2);
|
|
nameSpr.drawString("Unit", name_spr_width / 2, nameSpr.fontHeight() / 2 * 3 + 2); nameSpr.pushSprite(DIAL_CENTRE_X - name_spr_width / 2, DIAL_CENTRE_Y + 40, 2);
|
|
nameSpr.pushSprite(DIAL_CENTRE_X - name_spr_width / 2, DIAL_CENTRE_Y + 40, 2);
|
|
|
|
// Define where the needle pivot point is on the TFT before
|
|
// creating the needle so boundary calculation is correct
|
|
tft.setPivot(DIAL_CENTRE_X, DIAL_CENTRE_Y);
|
|
|
|
// Create the needle Sprite
|
|
createNeedle();
|
|
|
|
// Reset needle position to 0
|
|
plotNeedle(0, 0, 0);
|
|
|
|
delay(2000);
|
|
Serial.println("OK");
|
|
}
|
|
|
|
// =======================================================================================
|
|
// Loop
|
|
// =======================================================================================
|
|
enum DISPLAY_STATE
|
|
{
|
|
DISPLAY_PM1P0,
|
|
DISPLAY_PM2P5,
|
|
DISPLAY_PM10P0,
|
|
DISPLAY_TEMP,
|
|
DISPLAY_HUM,
|
|
DISPLAY_CO2,
|
|
DISPLAY_LAST
|
|
};
|
|
|
|
DISPLAY_STATE displaystate = DISPLAY_PM1P0;
|
|
uint32_t display_last_update = 0;
|
|
|
|
#define MAXGUAGE 240
|
|
#define MINGUAGE 0
|
|
#define DISPLAY_ROTATE 10 //sec
|
|
|
|
void handleLCD()
|
|
{
|
|
static uint32_t value;
|
|
static uint16_t angle;
|
|
DISPLAY_STATE displaystate_next = DISPLAY_PM1P0;
|
|
sensor_e nextSensor = AE_1P0;
|
|
|
|
switch (displaystate)
|
|
{
|
|
case DISPLAY_PM1P0:
|
|
{
|
|
nextSensor = AE_1P0;
|
|
displaystate_next = DISPLAY_PM2P5;
|
|
}
|
|
break;
|
|
case DISPLAY_PM2P5:
|
|
{
|
|
nextSensor = AE_2P5;
|
|
displaystate_next = DISPLAY_PM10P0;
|
|
}
|
|
break;
|
|
case DISPLAY_PM10P0:
|
|
{
|
|
nextSensor = AE_10P0;
|
|
displaystate_next = DISPLAY_TEMP;
|
|
}
|
|
break;
|
|
case DISPLAY_TEMP:
|
|
{
|
|
nextSensor = SCD30_temp;
|
|
displaystate_next = DISPLAY_HUM;
|
|
}
|
|
break;
|
|
case DISPLAY_HUM:
|
|
{
|
|
nextSensor = SCD30_hum;
|
|
displaystate_next = DISPLAY_CO2;
|
|
}
|
|
break;
|
|
case DISPLAY_CO2:
|
|
{
|
|
nextSensor = SCD30_hum;
|
|
displaystate_next = DISPLAY_PM1P0;
|
|
}
|
|
break;
|
|
|
|
case DISPLAY_LAST:
|
|
default:
|
|
{
|
|
displaystate_next = DISPLAY_PM1P0;
|
|
}
|
|
break;
|
|
}
|
|
//rotate display
|
|
uint32_t timenow = millis();
|
|
if (timenow - display_last_update > (DISPLAY_ROTATE * 1000))
|
|
{
|
|
displaystate = displaystate_next;
|
|
display_last_update = timenow;
|
|
Serial.println("LCD next State");
|
|
}
|
|
//calculate value to angle
|
|
|
|
AQSSensor *sensor = getSensor(nextSensor);
|
|
if (sensor == NULL)
|
|
{
|
|
Serial.println("LCD: getSensor=NULL!");
|
|
return;
|
|
}
|
|
|
|
angle = map(value, sensor->getMin(), sensor->getMax(), MINGUAGE, MAXGUAGE);
|
|
|
|
plotNeedle(angle, 15, value);
|
|
|
|
nameSpr.drawString(sensor->getName().c_str(), name_spr_width / 2, nameSpr.fontHeight() / 2);
|
|
nameSpr.drawString(sensor->getUnit().c_str(), name_spr_width / 2, nameSpr.fontHeight() / 2 * 3 + 2);
|
|
nameSpr.pushSprite(DIAL_CENTRE_X - name_spr_width / 2, DIAL_CENTRE_Y + 40, 2);
|
|
}
|
|
|
|
// =======================================================================================
|
|
// Create the needle Sprite
|
|
// =======================================================================================
|
|
void createNeedle(void)
|
|
{
|
|
needle.setColorDepth(16);
|
|
needle.createSprite(NEEDLE_WIDTH, NEEDLE_LENGTH); // create the needle Sprite
|
|
|
|
needle.fillSprite(TFT_BLACK); // Fill with black
|
|
|
|
// Define needle pivot point relative to top left corner of Sprite
|
|
uint16_t piv_x = NEEDLE_WIDTH / 2; // pivot x in Sprite (middle)
|
|
uint16_t piv_y = NEEDLE_RADIUS; // pivot y in Sprite
|
|
needle.setPivot(piv_x, piv_y); // Set pivot point in this Sprite
|
|
|
|
// Draw the red needle in the Sprite
|
|
needle.fillRect(0, 0, NEEDLE_WIDTH, NEEDLE_LENGTH, TFT_MAROON);
|
|
needle.fillRect(1, 1, NEEDLE_WIDTH - 2, NEEDLE_LENGTH - 2, TFT_RED);
|
|
|
|
// Bounding box parameters to be populated
|
|
int16_t min_x;
|
|
int16_t min_y;
|
|
int16_t max_x;
|
|
int16_t max_y;
|
|
|
|
// Work out the worst case area that must be grabbed from the TFT,
|
|
// this is at a 45 degree rotation
|
|
needle.getRotatedBounds(45, &min_x, &min_y, &max_x, &max_y);
|
|
|
|
// Calculate the size and allocate the buffer for the grabbed TFT area
|
|
tft_buffer = (uint16_t *)malloc(((max_x - min_x) + 2) * ((max_y - min_y) + 2) * 2);
|
|
}
|
|
|
|
// =======================================================================================
|
|
// Move the needle to a new position
|
|
// =======================================================================================
|
|
void plotNeedle(int16_t angle, uint16_t ms_delay, uint32_t value)
|
|
{
|
|
static int16_t old_angle = -120; // Starts at -120 degrees
|
|
|
|
// Bounding box parameters
|
|
static int16_t min_x;
|
|
static int16_t min_y;
|
|
static int16_t max_x;
|
|
static int16_t max_y;
|
|
|
|
if (angle < 0)
|
|
angle = 0; // Limit angle to emulate needle end stops
|
|
if (angle > 240)
|
|
angle = 240;
|
|
|
|
angle -= 120; // Starts at -120 degrees
|
|
|
|
// Move the needle until new angle reached
|
|
while (angle != old_angle || !buffer_loaded)
|
|
{
|
|
|
|
if (old_angle < angle)
|
|
old_angle++;
|
|
else
|
|
old_angle--;
|
|
|
|
// Only plot needle at even values to improve plotting performance
|
|
if ((old_angle & 1) == 0)
|
|
{
|
|
if (buffer_loaded)
|
|
{
|
|
// Paste back the original needle free image area
|
|
tft.pushRect(min_x, min_y, 1 + max_x - min_x, 1 + max_y - min_y, tft_buffer);
|
|
}
|
|
|
|
if (needle.getRotatedBounds(old_angle, &min_x, &min_y, &max_x, &max_y))
|
|
{
|
|
// Grab a copy of the area before needle is drawn
|
|
tft.readRect(min_x, min_y, 1 + max_x - min_x, 1 + max_y - min_y, tft_buffer);
|
|
buffer_loaded = true;
|
|
}
|
|
|
|
// Draw the needle in the new postion, black in needle image is transparent
|
|
needle.pushRotated(old_angle, TFT_BLACK);
|
|
|
|
// Wait before next update
|
|
delay(ms_delay);
|
|
}
|
|
|
|
// Update the number at the centre of the dial
|
|
spr.drawNumber(value, spr_width / 2, spr.fontHeight() / 2);
|
|
spr.pushSprite(120 - spr_width / 2, 120 - spr.fontHeight() / 2);
|
|
|
|
// Slow needle down slightly as it approaches the new position
|
|
if (abs(old_angle - angle) < 10)
|
|
ms_delay += ms_delay / 5;
|
|
}
|
|
}
|
|
|
|
// =======================================================================================
|