1.在此練習中,設定當溫度介於15度以下呈現「黃色」,15度到28度呈現「綠色」,28度以上呈現「紅色」。

2.打開之前已完成的OLED程式,並複製所需要的MQTT程式加入到OLED程式中,並儲存為「OLED_Temp_MQTT.ino」。範例檔「6.OLED_Temp_MQTT.ino」。

程式碼

#include "Wire.h"  //i2c 21或22接腳
#include "U8g2lib.h" //請下載u8g2程式庫
#include <SimpleDHT.h>
#include <WiFi.h>
#include <PubSubClient.h> //請先安裝PubSubClient程式庫
#include <ESP32Servo.h> //請先安裝ESP32Servo程式庫
// ------ 以下修改成你自己的WiFi帳號密碼 ------
char* ssid = "test";
char* password = "12345678";
//------ 以下修改成你DHT11腳位 ------
int pinDHT11 = 4;
SimpleDHT11 dht11(pinDHT11);
// ------ 以下修改成你MQTT設定 ------
char* MQTTServer = "mqttgo.io";//免註冊MQTT伺服器
int MQTTPort = 1883;//MQTT Port 除非加密,否則不用更改port
char* MQTTUser = "";//不須帳密
char* MQTTPassword = "";//不須帳密
//推播主題1:推播溫度(記得改Topic)
char* MQTTPubTopic1 = "李小保/ESP32/環境";
long MQTTLastPublishTime;//此變數用來記錄推播時間
long MQTTPublishInterval = 10000;//每10秒推撥一次
WiFiClient WifiClient;
PubSubClient MQTTClient(WifiClient);

//OLED 螢幕解析度為128*64
U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);

void setup()
{
  pinMode(15,OUTPUT); // 紅R
  pinMode(16,OUTPUT); // 黃Y
  pinMode(17,OUTPUT); // 綠G
  
  u8g2.begin();//初始化
  u8g2.enableUTF8Print();//啟用 UTF8字集
  u8g2.setFont(u8g2_font_unifont_t_chinese1);//設定使用中文字形
  u8g2.setDrawColor(1);//設定顏色,我們是單色只有1
  u8g2.setFontPosTop();//座標從上開始
  u8g2.setFontDirection(0);//0不旋轉、1->90、2->180、3->270

  Serial.begin(115200);
  pinMode(15, OUTPUT);//rLED燈
  pinMode(16, OUTPUT);//yLED燈
  pinMode(17, OUTPUT);//gLED燈
  
  //開始WiFi連線
  WifiConnecte();

  //開始MQTT連線
  MQTTConnecte();
}
//long i = 0;  //用來顯示目前更新的次數
void loop()
{
  // read without samples.
  byte temperature = 0; //宣告為byte,範圍為0-255
  byte humidity = 0;
  int err = SimpleDHTErrSuccess;

  if ((err = dht11.read(&temperature, &humidity, NULL)) != SimpleDHTErrSuccess) {
    Serial.print("Read DHT11 failed, err="); Serial.print(SimpleDHTErrCode(err));
    Serial.print(","); Serial.println(SimpleDHTErrDuration(err)); delay(1000);
    return;
  }

  //i = i + 1;
  u8g2.clear();//顯示前清除螢幕
  u8g2.setCursor(0, 20);//移動游標
  u8g2.print(String("溫度:").c_str());   //寫入文字
  u8g2.print(String((int)temperature).c_str()); //c_str將String字串轉為OLED可以使用的character字元
  u8g2.print(String("C").c_str());

  u8g2.setCursor(0, 40);//移動游標
  /*這一段是用來顯示更新的次數
    u8g2.print(String("目前數字:").c_str());//寫入文字
    u8g2.print(String(i).c_str());//寫入文字
  */

  u8g2.print(String("溼度:").c_str());   //寫入文字
  u8g2.print(String((int)humidity).c_str()); //c_str將String字串轉為OLED可以使用的character字元
  u8g2.print(String("%").c_str());

  u8g2.drawLine(0, 11, 90, 11);//劃線從0,11->90,60
  u8g2.drawLine(0, 60, 90, 60);//劃線從0,60->90,60

  u8g2.sendBuffer();//送到螢幕顯示

  Serial.print("溫度:"); Serial.print((int)temperature); Serial.print("C,"); //(int)強制轉型為int整數
  Serial.print("溼度:"); Serial.print((int)humidity); Serial.println("%。"); //(int)強制轉型為int整數
  Serial.println("==========================");

  //如果WiFi連線中斷,則重啟WiFi連線
  if (WiFi.status() != WL_CONNECTED) { WifiConnecte(); }

  //如果MQTT連線中斷,則重啟MQTT連線
  if (!MQTTClient.connected()) {  MQTTConnecte(); }

  //如果距離上次傳輸已經超過10秒,則Publish溫溼度
  if ((millis() - MQTTLastPublishTime) >= MQTTPublishInterval ) {
    // ------ 將DHT11溫度送到MQTT主題 ------
    String json="{\\"temp\\":"+String(temperature)+",\\"humi\\":"+String(humidity)+"}";
    MQTTClient.publish(MQTTPubTopic1, json.c_str());
    Serial.println("環境參數已推播到MQTT Broker");
    MQTTLastPublishTime = millis(); //更新最後傳輸時間
  }
  MQTTClient.loop();//更新訂閱狀態

  if ((int)temperature > 28){
    digitalWrite(15,HIGH);
    digitalWrite(16,LOW);
    digitalWrite(17,LOW); 
  } else if ((int)temperature < 15){
    digitalWrite(15,LOW);
    digitalWrite(16,HIGH);
    digitalWrite(17,LOW); 
  } else {
    digitalWrite(15,LOW);
    digitalWrite(16,LOW);
    digitalWrite(17,HIGH); 
  }

  delay(5000);
}

//開始WiFi連線
void WifiConnecte() {
  //開始WiFi連線
  WiFi.begin(ssid, password);
  while (WiFi.status() != WL_CONNECTED) {
    delay(500);
    Serial.print(".");
  }
  Serial.println("WiFi連線成功");
  Serial.print("IP Address:");
  Serial.println(WiFi.localIP());
}

//開始MQTT連線
void MQTTConnecte() {
  MQTTClient.setServer(MQTTServer, MQTTPort);
  //MQTTClient.setCallback(MQTTCallback); //收到訂閱資料時,要處理MQTTCALLBACK
  while (!MQTTClient.connected()) {
    //以亂數為ClietID
    String  MQTTClientid = "esp32-" + String(random(1000000, 9999999));//建立一個郵差,為了要不一樣,所以使用亂數
    if (MQTTClient.connect(MQTTClientid.c_str(), MQTTUser, MQTTPassword)) {
      //連結成功,顯示「已連線」。
      Serial.println("MQTT已連線");
    } else {
      //若連線不成功,則顯示錯誤訊息,並重新連線
      Serial.print("MQTT連線失敗,狀態碼=");
      Serial.println(MQTTClient.state());
      Serial.println("五秒後重新連線");
      delay(5000);
    }
  }
}

範例結果

341854452_243134374928722_1919320014116728418_n.jpg

下一節