小鹏浇花套件单机版程序V1.0.0

本程序是小鹏物联网智能浇花套件的单机版程序(不连接物联网),供同学们参考。

需求

1. 通过土壤湿度传感器收集湿度数据;

2. 根据湿度数据决定是否浇水;

3. 通过OLED显示相关信息;

4. 可手动进行浇水(不考虑土壤湿度);

硬件

小鹏物联网智能浇花套件

思路

1.需要使用多任务来实现,本程序使用ESP32 FreeRTOS

2.在主循环内读取土壤湿度数据并放入队列(任务0)

3.在数据接收任务内接收队列数据,并判断是否要浇水(任务1)

4.使用一个任务来处理按键(任务2)

5.使用一个任务来处理OLED显示(任务3)

完整程序

/**********************************************************************
 **  小鹏浇花系统
 * 
 **  version: 1.0.0
 **  author: 老张
 **  homesite: www.xpstem.com/product/intelligent-watering/index
 **********************************************************************/
#include <Arduino.h>
#include <SSD1306Wire.h>    // 4.6.1 (ThingPulse)
#include <OneButton.h>      // 2.6.1
#include <ArrayList.h>      // 1.0.2

//************** 常数数据 开始 **********************
// Pins
int soil_moisture_sensor_pin = 27;      // 土壤湿度传感器引脚
int pump_control_relay_pin = 15;        // 水泵继电器引脚
int oled_display_pin_scl = 22;          // OLED显示屏I2C引脚
int oled_display_pin_sda = 21;          // OLED显示屏I2C引脚
int oled_display_i2c_addr = 61;         // OLED显示屏I2C地址
int manual_button_pin = 26;             // 手动按钮引脚

//************* 配置变量 结束 *********************

//************* 运行时变量 开始 ********************
OneButton manualBtn(manual_button_pin, true);

int stateCode = 0; //1:config, 2:work, 3:sleep, 9:error
String stateInfo = "";

int collectCount = 0;
int sensorVal = 0;
int Threshold = 50;  // 湿度阀值

ArrayList<int> valueList; // 湿度值列表

QueueHandle_t msgQueue = xQueueCreate(10, sizeof(int)); // 消息队列

//************* 运行时变量 结束 *********************

/***********************************************************
 ***  控制水泵继电器
 **********************************************************/ 
void main_watering(int relayPin, int seconds) {
  pinMode(relayPin, OUTPUT);
  
  digitalWrite(relayPin, HIGH);
  Serial.printf("打开水泵继电器,引脚:%d,保持 %d 秒。\n", relayPin, seconds);
  
  for (int i=0; i<seconds; i++) { 
    Serial.print("."); 
    delay(1000);
  }
  Serial.println("");
        
  digitalWrite(relayPin, LOW);
  Serial.println("关闭水泵继电器。");
}

/***********************************************************
 ***  UI任务
 **********************************************************/
void task_display(void *pvParam) {

  SSD1306Wire display(/* address */ oled_display_i2c_addr, 
    /* sda */ oled_display_pin_sda, 
    /* scl */ oled_display_pin_scl); 

  display.init();
  //display.flipScreenVertically();
  display.setFont(ArialMT_Plain_16);
  
  while(1) {
    display.clear();
    if (stateCode==1) {
      display.drawString(0, 0, "[config]");
      
    } else if (stateCode==2) {
      display.drawString(0, 0, "[work]" );
      display.drawString(0, 20, "value: " + (sensorVal==0 ? "" : String(sensorVal)) );
      if (stateInfo.length()>0) {
        display.drawString(0, 40, stateInfo );
      }
      
    } else if (stateCode==3) {
      display.drawString(0, 0, "Zzz..." );
      
    } else if (stateCode==9) {
      display.drawString(0, 0, "[error]");
      display.drawString(0, 20, stateInfo );
    }
    display.display();
    
    vTaskDelay(1000 / portTICK_PERIOD_MS);
  };
}

/******************************************************** 
 ***  Button任务
 ********************************************************/
void task_buttonTick(void *pvParam) {
  while(1) {
    manualBtn.tick();
    vTaskDelay(10 / portTICK_PERIOD_MS); //10ms
  }
}

/******************************************************** 
 ***  数据接收任务
 ********************************************************/
void task_dataReceive(void *pvParam) {
  while (1) {
    int received;
    // 从队列接收数据
    if (xQueueReceive(msgQueue, &received, 100/portTICK_PERIOD_MS) == pdTRUE) {
      valueList.add(received);

      int flag = 0;
      if (valueList.size() > 5) {
        // 检查是否全部低于阀值
        for (int i=0; i<valueList.size(); i++) {
          if (valueList.get(i) > Threshold) { // 超过阀值
            flag = 1;
            break;
          }
        }

        if (flag == 0) {
          // 全部低于阀值,则浇水一次
          main_watering(pump_control_relay_pin, 5);
        }
        
        valueList.remove(0);
      }

    }
  }
}

/***********************************************************
 ***  进入休眠状态
 **********************************************************/
void main_deepSleep(int sleepSeconds) {
  stateCode = 3;
  delay(3000); //3s for ui update
  
  //digitalWrite(power_led_pin, LOW);

  Serial.printf("进入深度休眠状态(%d%s)。\n", 
    sleepSeconds<3600 ? sleepSeconds/60 : sleepSeconds/3600,
    sleepSeconds<3600 ? "分钟" : "小时");
  ESP.deepSleep(sleepSeconds * 1e6);
}

/***********************************************************
 ***  收集传感器数据
 **********************************************************/
void main_collectData(int sensorPin) {
  Serial.printf("获取土壤湿度数据,引脚:%d\n", sensorPin);
  int originVal = analogRead(sensorPin);
  if (originVal > 0) {
    sensorVal = map(originVal, 1, 4095, 100, 1);
    // 加入队列
    xQueueSend(msgQueue, &sensorVal, portMAX_DELAY);
    
  } else { 
    // 数据无效,不做处理。
  }
}

/***********************************************************
 ***  初始化设置
 **********************************************************/
void setup() {
  Serial.begin(115200);
  Serial.println();
  Serial.println("-------------------------------");
  Serial.println("--------- 小鹏浇花系统 ----------");
  Serial.println("-------------------------------");

  Serial.println("启动UI任务程序。");
  // UI任务
  xTaskCreate(
    task_display,     // 任务函数
    "Display_Task",  // 任务名称
    4096,       // 堆栈大小(字节)
    NULL,       // 参数
    1,          // 优先级
    NULL        // 任务句柄
  );

  Serial.println("启动主程序。");
  
  stateCode = 2;

  // 等传感器稳定。
  Serial.print("等待传感器初始化(3sec).");
  for (int i=0;i<6;i++) {
    Serial.print(".");
    delay(500);
  }
  Serial.println("");

  // 手动按钮
  manualBtn.attachDoubleClick([]() {
    main_watering(pump_control_relay_pin, 5);
  });
  
  xTaskCreate(task_buttonTick, "ButtonTick_Task", 2048, NULL, 1, NULL);
  
  xTaskCreate(task_dataReceive, "DataReceive_Task", 2048, NULL, 1, NULL);
}

/***********************************************************
 ***  主循环
 ***  1.获取传感器数据并发送
 **********************************************************/
void loop() {

  // 采集数据
  main_collectData(soil_moisture_sensor_pin);
  collectCount++;
  
  vTaskDelay(120000 / portTICK_PERIOD_MS); //120s

  if (collectCount > 30) {
    main_deepSleep(23 * 3600); //23小时
  }
  
} 
- 本文为本站原创文章,转载请保留出处。
- 文章链接:https://www.xpstem.com/article/2000406

08-22   阅读(19)   评论(0)
 标签: 创客 Arduino ESP32 FreeRTOS

涨知识
数码管

也称作LED数码管,是一种可以显示数字和其他信息的电子设备。玻璃管中包括一个金属丝网制成的阳极和多个阴极。大部分数码管阴极的形状为数字。

评论:
相关文章
Arduino-ESP32与ESP-IDF的版本对应表

Arduino-ESP32与ESP-IDF的版本对应表。


Arduino-ESP32文件系统全解析:SPIFFS、LittleFS、SD卡操作

Arduino-ESP32提供了多种文件系统解决方案,本文将深入解析SPIFFS、LittleFS和SD卡三种主流存储方案,帮助你做出最佳选择。


ESP32-P4-WIFI6开发板

ESP32-P4-WIFI6-DEV-KIT是一款微雪(Waveshare)设计的基于 ESP32-P4 的多媒体开发板,并集成 ESP32-C6,支持 Wi-Fi 6 和 BLE 5 无线连接。它提供丰富的人机交互接口,包括 MIPI-CSI (集成图像信号处理器 ISP)、MIPI-DSI、SPI、I2S、I2C、LED PWM、MCPWM、RMT、ADC、UART 和 TWAI 等。


ESP-Hosted 入门介绍 &使用指南

ESP-Hosted 解决方案提供了将 ESP 板用作 Wi-Fi 和 Bluetooth/BLE 连接的通信处理器的方法。


设备上云太麻烦?ESP-Hosted一站触达!

ESP-Hosted 提供了一种将ESP芯片和模组用作通信协处理器的解决方案,该解决方案为主机微处理器或微控制器提供无线连接,使主机能够与其他设备通信。简单来说为网卡方案。


ESP32 + Arduino使用TFT_eSPI库

Arduino+ESP32上使用TFT_eSPI库快速点亮这个屏幕,驱动芯片ST7789


ESP32 利用 SPI 连通 TFT 彩屏

本文给出了一个ESP32与SPI 接口TFT显示屏接线的详细说明,供大家参考。


在Micropython下使用ESPNow功能进行数据传输

本文讲解如何在Micropython环境下使用ESP32的ESPNow功能进行数据传输。


用 ESP32-S3 打造多功能 USB Dongle

ESP-Dongle 是一款基于 ESP32-S3 芯片开发的多功能 USB Device 解决方案。它不仅外形小巧,功能齐全,更集成了无线 U 盘、SD 卡读取以及 USB 无线网卡等多项功能。


利用 ESP32-S3 和 CSI 技术打造智能家居

ESP32 系列芯片可以利用 CSI 数据实现动作检测和存在检测。无论是自动调节灯光、风扇,还是节能控制,CSI 技术为智能家居带来了新的可能性。随着 CSI 技术的发展,未来的智能家居将能够更精确地感知和响应我们的行为,实现更高效、更人性化的控制。