mirror of
https://github.com/0015/ThatProject.git
synced 2026-01-12 09:17:42 +03:00
ESP32 | FLUTTER | BLE - Temperature & Humidity Check App
This commit is contained in:
@@ -0,0 +1,440 @@
|
||||
// BLE
|
||||
#include <BLEDevice.h>
|
||||
#include <BLEServer.h>
|
||||
#include <BLEUtils.h>
|
||||
#include <BLE2902.h>
|
||||
|
||||
BLEServer* pServer = NULL;
|
||||
BLECharacteristic* pCharacteristic = NULL;
|
||||
bool deviceConnected = false;
|
||||
bool oldDeviceConnected = false;
|
||||
|
||||
#define SERVICE_UUID "4fafc201-1fb5-459e-8fcc-c5c9c331914b"
|
||||
#define CHARACTERISTIC_UUID "beb5483e-36e1-4688-b7f5-ea07361b26a8"
|
||||
|
||||
class MyServerCallbacks: public BLEServerCallbacks {
|
||||
void onConnect(BLEServer* pServer) {
|
||||
deviceConnected = true;
|
||||
BLEDevice::startAdvertising();
|
||||
};
|
||||
|
||||
void onDisconnect(BLEServer* pServer) {
|
||||
deviceConnected = false;
|
||||
}
|
||||
};
|
||||
|
||||
// SCREEN, DHT
|
||||
|
||||
#include "SPIFFS.h"
|
||||
#include <JPEGDecoder.h>
|
||||
|
||||
#include <Adafruit_GFX.h>
|
||||
#include <Adafruit_SSD1351.h>
|
||||
#include <Adafruit_Sensor.h>
|
||||
#include <DHT.h>
|
||||
#include <DHT_U.h>
|
||||
|
||||
#define DHTPIN 19
|
||||
#define DHTTYPE DHT11
|
||||
|
||||
DHT_Unified dht(DHTPIN, DHTTYPE);
|
||||
|
||||
uint32_t delayMS;
|
||||
|
||||
#define SCREEN_WIDTH 128
|
||||
#define SCREEN_HEIGHT 128
|
||||
|
||||
#define SCLK_PIN 18
|
||||
#define MOSI_PIN 23
|
||||
#define DC_PIN 17
|
||||
#define CS_PIN 5
|
||||
#define RST_PIN 4
|
||||
|
||||
// Color definitions
|
||||
#define BLACK 0x0000
|
||||
#define BLUE 0x001F
|
||||
#define RED 0xF800
|
||||
#define GREEN 0x07E0
|
||||
#define CYAN 0x07FF
|
||||
#define MAGENTA 0xF81F
|
||||
#define YELLOW 0xFFE0
|
||||
#define WHITE 0xFFFF
|
||||
|
||||
const String fileTemperature = "/temperature.jpg";
|
||||
const String fileHumidity = "/humidity.jpg";
|
||||
|
||||
Adafruit_SSD1351 tft = Adafruit_SSD1351(SCREEN_WIDTH, SCREEN_HEIGHT, CS_PIN, DC_PIN, MOSI_PIN, SCLK_PIN, RST_PIN);
|
||||
|
||||
float prev_temp;
|
||||
float prev_humidity;
|
||||
|
||||
|
||||
// Thread
|
||||
#include <Thread.h>
|
||||
#include <ThreadController.h>
|
||||
|
||||
// ThreadController that will controll all threads
|
||||
ThreadController controll = ThreadController();
|
||||
|
||||
Thread* btThread = new Thread();
|
||||
Thread* drawingThread = new Thread();
|
||||
|
||||
void setup() {
|
||||
// put your setup code here, to run once:
|
||||
Serial.begin(115200);
|
||||
|
||||
initDrawing();
|
||||
initBT();
|
||||
|
||||
btThread->onRun(btCallback);
|
||||
btThread->setInterval(100);
|
||||
|
||||
drawingThread->onRun(drawingCallback);
|
||||
drawingThread->setInterval(delayMS);
|
||||
|
||||
controll.add(btThread);
|
||||
controll.add(drawingThread);
|
||||
}
|
||||
|
||||
void loop() {
|
||||
// put your main code here, to run repeatedly:
|
||||
controll.run();
|
||||
}
|
||||
|
||||
void drawingCallback(){
|
||||
sensors_event_t event;
|
||||
dht.temperature().getEvent(&event);
|
||||
if (isnan(event.temperature)) {
|
||||
Serial.println(F("Error reading temperature!"));
|
||||
}
|
||||
else {
|
||||
Serial.print(F("Temperature: "));
|
||||
Serial.print(event.temperature);
|
||||
Serial.println(F("°C"));
|
||||
updateTemp(event.temperature);
|
||||
}
|
||||
// Get humidity event and print its value.
|
||||
dht.humidity().getEvent(&event);
|
||||
if (isnan(event.relative_humidity)) {
|
||||
Serial.println(F("Error reading humidity!"));
|
||||
}
|
||||
else {
|
||||
Serial.print(F("Humidity: "));
|
||||
Serial.print(event.relative_humidity);
|
||||
Serial.println(F("%"));
|
||||
updateHumidity(event.relative_humidity);
|
||||
}
|
||||
}
|
||||
|
||||
void updateTemp(float temp){
|
||||
if(prev_temp != temp){
|
||||
tft.fillRect(64, 0, 128, 64, WHITE);
|
||||
tft.setCursor(70, 24);
|
||||
tft.setTextColor(BLACK);
|
||||
tft.setTextSize(3);
|
||||
String tempString = "";
|
||||
tempString += (int)temp;
|
||||
tempString += "C";
|
||||
tft.print(tempString);
|
||||
prev_temp = temp;
|
||||
}
|
||||
}
|
||||
|
||||
void updateHumidity(float humidity){
|
||||
if(prev_humidity != humidity){
|
||||
tft.fillRect(64, 64, 128, 128, WHITE);
|
||||
tft.setCursor(70, 88);
|
||||
tft.setTextColor(BLACK);
|
||||
tft.setTextSize(3);
|
||||
String humidityString = "";
|
||||
humidityString += (int)humidity;
|
||||
humidityString += "%";
|
||||
tft.print(humidityString);
|
||||
prev_humidity = humidity;
|
||||
}
|
||||
}
|
||||
|
||||
void btCallback(){
|
||||
// notify changed value
|
||||
if (deviceConnected) {
|
||||
|
||||
String str = "";
|
||||
str += prev_temp;
|
||||
str += ",";
|
||||
str += prev_humidity;
|
||||
|
||||
pCharacteristic->setValue((char*)str.c_str());
|
||||
pCharacteristic->notify();
|
||||
|
||||
}
|
||||
// disconnecting
|
||||
if (!deviceConnected && oldDeviceConnected) {
|
||||
//delay(500); // give the bluetooth stack the chance to get things ready
|
||||
pServer->startAdvertising(); // restart advertising
|
||||
Serial.println("start advertising");
|
||||
oldDeviceConnected = deviceConnected;
|
||||
}
|
||||
// connecting
|
||||
if (deviceConnected && !oldDeviceConnected) {
|
||||
// do stuff here on connecting
|
||||
oldDeviceConnected = deviceConnected;
|
||||
}
|
||||
}
|
||||
|
||||
void initBT(){
|
||||
// Create the BLE Device
|
||||
BLEDevice::init("ESP32 Test");
|
||||
|
||||
// Create the BLE Server
|
||||
pServer = BLEDevice::createServer();
|
||||
pServer->setCallbacks(new MyServerCallbacks());
|
||||
|
||||
// Create the BLE Service
|
||||
BLEService *pService = pServer->createService(SERVICE_UUID);
|
||||
|
||||
// Create a BLE Characteristic
|
||||
pCharacteristic = pService->createCharacteristic(
|
||||
CHARACTERISTIC_UUID,
|
||||
BLECharacteristic::PROPERTY_READ |
|
||||
BLECharacteristic::PROPERTY_WRITE |
|
||||
BLECharacteristic::PROPERTY_NOTIFY |
|
||||
BLECharacteristic::PROPERTY_INDICATE
|
||||
);
|
||||
|
||||
// https://www.bluetooth.com/specifications/gatt/viewer?attributeXmlFile=org.bluetooth.descriptor.gatt.client_characteristic_configuration.xml
|
||||
// Create a BLE Descriptor
|
||||
pCharacteristic->addDescriptor(new BLE2902());
|
||||
|
||||
// Start the service
|
||||
pService->start();
|
||||
|
||||
// Start advertising
|
||||
BLEAdvertising *pAdvertising = BLEDevice::getAdvertising();
|
||||
pAdvertising->addServiceUUID(SERVICE_UUID);
|
||||
pAdvertising->setScanResponse(false);
|
||||
pAdvertising->setMinPreferred(0x0); // set value to 0x00 to not advertise this parameter
|
||||
BLEDevice::startAdvertising();
|
||||
Serial.println("Waiting a client connection to notify...");
|
||||
|
||||
}
|
||||
|
||||
void initDrawing(){
|
||||
tft.begin();
|
||||
tft.fillRect(0,0,128,128,WHITE);
|
||||
|
||||
dht.begin();
|
||||
|
||||
// Print temperature sensor details.
|
||||
sensor_t sensor;
|
||||
dht.temperature().getSensor(&sensor);
|
||||
|
||||
// Print humidity sensor details.
|
||||
dht.humidity().getSensor(&sensor);
|
||||
|
||||
// Set delay between sensor readings based on sensor details.
|
||||
delayMS = sensor.min_delay / 1000;
|
||||
|
||||
if(!SPIFFS.begin()){
|
||||
while(1) yield();
|
||||
}
|
||||
|
||||
drawFSJpeg(fileTemperature.c_str(), 0, 0);
|
||||
drawFSJpeg(fileHumidity.c_str(), 0, 64);
|
||||
|
||||
tft.setCursor(72, 24);
|
||||
tft.setTextColor(BLACK);
|
||||
tft.setTextSize(3);
|
||||
tft.print("?");
|
||||
|
||||
tft.setCursor(72, 88);
|
||||
tft.setTextColor(BLACK);
|
||||
tft.setTextSize(3);
|
||||
tft.print("?");
|
||||
}
|
||||
|
||||
|
||||
/*====================================================================================
|
||||
This sketch contains support functions to render the Jpeg images.
|
||||
|
||||
Created by Bodmer 15th Jan 2017
|
||||
==================================================================================*/
|
||||
|
||||
// Return the minimum of two values a and b
|
||||
#define minimum(a,b) (((a) < (b)) ? (a) : (b))
|
||||
|
||||
//====================================================================================
|
||||
// This function opens the Filing System Jpeg image file and primes the decoder
|
||||
//====================================================================================
|
||||
void drawFSJpeg(const char *filename, int xpos, int ypos) {
|
||||
|
||||
Serial.println("=====================================");
|
||||
Serial.print("Drawing file: "); Serial.println(filename);
|
||||
Serial.println("=====================================");
|
||||
|
||||
// Open the file (the Jpeg decoder library will close it)
|
||||
fs::File jpgFile = SPIFFS.open( filename, "r"); // File handle reference for SPIFFS
|
||||
// File jpgFile = SD.open( filename, FILE_READ); // or, file handle reference for SD library
|
||||
|
||||
if ( !jpgFile ) {
|
||||
Serial.print("ERROR: File \""); Serial.print(filename); Serial.println ("\" not found!");
|
||||
return;
|
||||
}
|
||||
|
||||
// To initialise the decoder and provide the file, we can use one of the three following methods:
|
||||
//boolean decoded = JpegDec.decodeFsFile(jpgFile); // We can pass the SPIFFS file handle to the decoder,
|
||||
//boolean decoded = JpegDec.decodeSdFile(jpgFile); // or we can pass the SD file handle to the decoder,
|
||||
boolean decoded = JpegDec.decodeFsFile(filename); // or we can pass the filename (leading / distinguishes SPIFFS files)
|
||||
// The filename can be a String or character array
|
||||
if (decoded) {
|
||||
// print information about the image to the serial port
|
||||
jpegInfo();
|
||||
|
||||
// render the image onto the screen at given coordinates
|
||||
jpegRender(xpos, ypos);
|
||||
}
|
||||
else {
|
||||
Serial.println("Jpeg file format not supported!");
|
||||
}
|
||||
}
|
||||
|
||||
//====================================================================================
|
||||
// Decode and paint onto the TFT screen
|
||||
//====================================================================================
|
||||
void jpegRender(int xpos, int ypos) {
|
||||
|
||||
// retrieve infomration about the image
|
||||
uint16_t *pImg;
|
||||
uint16_t mcu_w = JpegDec.MCUWidth;
|
||||
uint16_t mcu_h = JpegDec.MCUHeight;
|
||||
uint32_t max_x = JpegDec.width;
|
||||
uint32_t max_y = JpegDec.height;
|
||||
|
||||
// Jpeg images are draw as a set of image block (tiles) called Minimum Coding Units (MCUs)
|
||||
// Typically these MCUs are 16x16 pixel blocks
|
||||
// Determine the width and height of the right and bottom edge image blocks
|
||||
uint32_t min_w = minimum(mcu_w, max_x % mcu_w);
|
||||
uint32_t min_h = minimum(mcu_h, max_y % mcu_h);
|
||||
|
||||
// save the current image block size
|
||||
uint32_t win_w = mcu_w;
|
||||
uint32_t win_h = mcu_h;
|
||||
|
||||
// record the current time so we can measure how long it takes to draw an image
|
||||
uint32_t drawTime = millis();
|
||||
|
||||
// save the coordinate of the right and bottom edges to assist image cropping
|
||||
// to the screen size
|
||||
max_x += xpos;
|
||||
max_y += ypos;
|
||||
|
||||
// read each MCU block until there are no more
|
||||
while ( JpegDec.read()) {
|
||||
|
||||
// save a pointer to the image block
|
||||
pImg = JpegDec.pImage;
|
||||
|
||||
// calculate where the image block should be drawn on the screen
|
||||
int mcu_x = JpegDec.MCUx * mcu_w + xpos;
|
||||
int mcu_y = JpegDec.MCUy * mcu_h + ypos;
|
||||
|
||||
// check if the image block size needs to be changed for the right edge
|
||||
if (mcu_x + mcu_w <= max_x) win_w = mcu_w;
|
||||
else win_w = min_w;
|
||||
|
||||
// check if the image block size needs to be changed for the bottom edge
|
||||
if (mcu_y + mcu_h <= max_y) win_h = mcu_h;
|
||||
else win_h = min_h;
|
||||
|
||||
// copy pixels into a contiguous block
|
||||
if (win_w != mcu_w)
|
||||
{
|
||||
for (int h = 1; h < win_h-1; h++)
|
||||
{
|
||||
memcpy(pImg + h * win_w, pImg + (h + 1) * mcu_w, win_w << 1);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// draw image MCU block only if it will fit on the screen
|
||||
if ( ( mcu_x + win_w) <= tft.width() && ( mcu_y + win_h) <= tft.height())
|
||||
{
|
||||
tft.drawRGBBitmap(mcu_x, mcu_y, pImg, win_w, win_h);
|
||||
}
|
||||
|
||||
// Stop drawing blocks if the bottom of the screen has been reached,
|
||||
// the abort function will close the file
|
||||
else if ( ( mcu_y + win_h) >= tft.height()) JpegDec.abort();
|
||||
|
||||
}
|
||||
|
||||
// calculate how long it took to draw the image
|
||||
drawTime = millis() - drawTime;
|
||||
|
||||
// print the results to the serial port
|
||||
Serial.print ("Total render time was : "); Serial.print(drawTime); Serial.println(" ms");
|
||||
Serial.println("=====================================");
|
||||
|
||||
}
|
||||
|
||||
//====================================================================================
|
||||
// Send time taken to Serial port
|
||||
//====================================================================================
|
||||
void jpegInfo() {
|
||||
Serial.println(F("==============="));
|
||||
Serial.println(F("JPEG image info"));
|
||||
Serial.println(F("==============="));
|
||||
Serial.print(F( "Width :")); Serial.println(JpegDec.width);
|
||||
Serial.print(F( "Height :")); Serial.println(JpegDec.height);
|
||||
Serial.print(F( "Components :")); Serial.println(JpegDec.comps);
|
||||
Serial.print(F( "MCU / row :")); Serial.println(JpegDec.MCUSPerRow);
|
||||
Serial.print(F( "MCU / col :")); Serial.println(JpegDec.MCUSPerCol);
|
||||
Serial.print(F( "Scan type :")); Serial.println(JpegDec.scanType);
|
||||
Serial.print(F( "MCU width :")); Serial.println(JpegDec.MCUWidth);
|
||||
Serial.print(F( "MCU height :")); Serial.println(JpegDec.MCUHeight);
|
||||
Serial.println(F("==============="));
|
||||
}
|
||||
|
||||
//====================================================================================
|
||||
// Open a Jpeg file and dump it to the Serial port as a C array
|
||||
//====================================================================================
|
||||
void createArray(const char *filename) {
|
||||
|
||||
fs::File jpgFile; // File handle reference for SPIFFS
|
||||
// File jpgFile; // File handle reference For SD library
|
||||
|
||||
if ( !( jpgFile = SPIFFS.open( filename, "r"))) {
|
||||
Serial.println(F("JPEG file not found"));
|
||||
return;
|
||||
}
|
||||
|
||||
uint8_t data;
|
||||
byte line_len = 0;
|
||||
Serial.println("// Generated by a JPEGDecoder library example sketch:");
|
||||
Serial.println("// https://github.com/Bodmer/JPEGDecoder");
|
||||
Serial.println("");
|
||||
Serial.println("#if defined(__AVR__)");
|
||||
Serial.println(" #include <avr/pgmspace.h>");
|
||||
Serial.println("#endif");
|
||||
Serial.println("");
|
||||
Serial.print("const uint8_t ");
|
||||
while (*filename != '.') Serial.print(*filename++);
|
||||
Serial.println("[] PROGMEM = {"); // PROGMEM added for AVR processors, it is ignored by Due
|
||||
|
||||
while ( jpgFile.available()) {
|
||||
|
||||
data = jpgFile.read();
|
||||
Serial.print("0x"); if (abs(data) < 16) Serial.print("0");
|
||||
Serial.print(data, HEX); Serial.print(",");// Add value and comma
|
||||
line_len++;
|
||||
if ( line_len >= 32) {
|
||||
line_len = 0;
|
||||
Serial.println();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
Serial.println("};\r\n");
|
||||
// jpgFile.seek( 0, SeekEnd);
|
||||
jpgFile.close();
|
||||
}
|
||||
//====================================================================================
|
||||
Reference in New Issue
Block a user