466 lines
11 KiB
C++
466 lines
11 KiB
C++
/*
|
|
* CanGrow - simply DIY automatic plant grow system (for cannabis).
|
|
*
|
|
*/
|
|
|
|
|
|
/*
|
|
* Includes
|
|
*
|
|
*/
|
|
|
|
// Libraries
|
|
// https://github.com/arduino/ArduinoCore-avr/tree/master/libraries/SPI
|
|
#include <SPI.h>
|
|
// https://github.com/arduino/ArduinoCore-avr/tree/master/libraries/Wire
|
|
#include <Wire.h>
|
|
// https://github.com/arduino/ArduinoCore-avr/tree/master/libraries/EEPROM
|
|
#include <EEPROM.h>
|
|
// https://github.com/esp8266/Arduino/tree/master/libraries/ESP8266WiFi
|
|
#include <ESP8266WiFi.h>
|
|
#include <WiFiUdp.h>
|
|
// https://github.com/esp8266/Arduino/tree/master/libraries/ESP8266WebServer
|
|
#include <ESP8266WebServer.h>
|
|
// OTA update
|
|
#include <ESP8266HTTPUpdateServer.h>
|
|
// https://github.com/adafruit/Adafruit-GFX-Library
|
|
#include <Adafruit_GFX.h>
|
|
// https://github.com/adafruit/Adafruit_SSD1306
|
|
#include <Adafruit_SSD1306.h>
|
|
// https://github.com/adafruit/Adafruit_BME280_Library/
|
|
#include <Adafruit_Sensor.h>
|
|
#include <Adafruit_BME280.h>
|
|
// https://github.com/bblanchon/ArduinoJson
|
|
#include <ArduinoJson.h>
|
|
// https://github.com/arduino-libraries/NTPClient
|
|
#include <NTPClient.h>
|
|
// https://github.com/PaulStoffregen/Time
|
|
#include <TimeLib.h>
|
|
// DHT support dropped
|
|
// https://github.com/adafruit/DHT-sensor-library
|
|
// #include "DHT.h"
|
|
// SHT30/31
|
|
// https://github.com/adafruit/Adafruit_SHT31/
|
|
#include "Adafruit_SHT31.h"
|
|
|
|
|
|
|
|
/*
|
|
* CanGrow header files
|
|
*/
|
|
|
|
#include "CanGrow_PinAssignments.h"
|
|
#include "CanGrow_Init.h"
|
|
#include "CanGrow_Logo.h"
|
|
#include "CanGrow_Sensors.h"
|
|
#include "CanGrow_HTML.h"
|
|
#include "CanGrow_SysFunctions.h"
|
|
#include "CanGrow_WebFunctions.h"
|
|
|
|
|
|
|
|
/*
|
|
* Setup
|
|
*
|
|
*/
|
|
void setup() {
|
|
|
|
// setup pins
|
|
//pinMode(PINdht, INPUT);
|
|
pinMode(PINwaterlevel, OUTPUT);
|
|
pinMode(PINsoilmoisture, OUTPUT);
|
|
pinMode(PinFAN2, OUTPUT);
|
|
|
|
pinMode(PinWIPE, OUTPUT);
|
|
// set all OUTPUT to low
|
|
digitalWrite(PINwaterlevel, LOW);
|
|
|
|
// its better to leave them untuched until we know from EEPROM if they have to be inverted or not
|
|
//~ digitalWrite(PinFAN, HIGH);
|
|
//~ digitalWrite(PinLED, HIGH);
|
|
//~ digitalWrite(PinPUMP, HIGH);
|
|
// except PINsoilmoisture
|
|
// PINsoilmoisture is always HIGH and gets LOW in moment of waterlevel measurement
|
|
digitalWrite(PINsoilmoisture, HIGH);
|
|
// set FAN2 to off with digitalWrite LOW
|
|
analogWrite(PinFAN2, PinFAN2PWM);
|
|
|
|
// Start EEPROM
|
|
EEPROM.begin(512);
|
|
|
|
// Start Serial
|
|
Serial.begin(115200);
|
|
|
|
// Write a line before doing serious output, because before there is some garbage in serial
|
|
// whats get the cursor somewhere over the place
|
|
Serial.println("420");
|
|
Serial.print(".:: CanGrow firmware v");
|
|
Serial.print(CANGROW_VER);
|
|
Serial.print(" build ");
|
|
Serial.print(CANGROW_BUILD);
|
|
Serial.println(" starting ::.");
|
|
|
|
Serial.println(":: initialise I2C ::");
|
|
// initialise Wire for I2C
|
|
Wire.begin();
|
|
// just for testing
|
|
//Wire.setClockStretchLimit(2500);
|
|
Serial.println(":: initialise display ::");
|
|
// initialise I2C display
|
|
display.begin(SSD1306_SWITCHCAPVCC, 0x3C); // Address 0x3C for 128x64
|
|
display.clearDisplay();
|
|
display.display();
|
|
|
|
// set display settings
|
|
display.setTextSize(1);
|
|
display.setTextColor(SSD1306_WHITE, SSD1306_BLACK);
|
|
|
|
// display Logo
|
|
display.drawBitmap(0, 0, bmpCanGrow_Logo, 128, 32, WHITE);
|
|
display.display();
|
|
|
|
Serial.println(":: initialise chirp sensor if present ::");
|
|
// reset chirp
|
|
writeI2CRegister8bit(0x20, 6); //TODO: Do only, when configured
|
|
|
|
// initialise DHT11
|
|
// dht support dropped
|
|
// dht.begin(); //TODO: Do only, when configured
|
|
|
|
// initialise BME280
|
|
// dirty way of supporting multiple addresses, DRY? :p
|
|
Serial.println(":: initialise BME280 sensor, address 0x76 ::");
|
|
// ToDo: let the user configure somewhere the ID of the BME280 sensor
|
|
if(!bme_0x76.begin(0x76)) {
|
|
Serial.println("!! Cannot find BME280 on I2C bus at address 0x76. Please check connection or ID");
|
|
}
|
|
Serial.println(":: initialise BME280 sensor, address 0x77 ::");
|
|
// ToDo: let the user configure somewhere the ID of the BME280 sensor
|
|
if(!bme_0x77.begin(0x77)) {
|
|
Serial.println("!! Cannot find BME280 on I2C bus at address 0x77. Please check connection or ID");
|
|
}
|
|
|
|
// initialise SHT31
|
|
Serial.println(":: initialise SHT31 sensor, address 0x44 ::");
|
|
if (! sht31_0x44.begin(0x44)) { // Set to 0x45 for alternate i2c addr
|
|
Serial.println("!! Cannot find SHT31 on I2C bus at address 0x45. Please check connection or ID");
|
|
}
|
|
Serial.println(":: initialise SHT31 sensor, address 0x45 ::");
|
|
if (! sht31_0x45.begin(0x45)) { // Set to 0x45 for alternate i2c addr
|
|
Serial.println("!! Cannot find SHT31 on I2C bus at address 0x45. Please check connection or ID");
|
|
}
|
|
|
|
Serial.print(":: SHT31 (0x44) heater enable state ::");
|
|
if (sht31_0x44.isHeaterEnabled())
|
|
Serial.println("ENABLED");
|
|
else
|
|
Serial.println("DISABLED");
|
|
|
|
Serial.print(":: SHT31 (0x45) heater enable state ::");
|
|
if (sht31_0x45.isHeaterEnabled())
|
|
Serial.println("ENABLED");
|
|
else
|
|
Serial.println("DISABLED");
|
|
|
|
|
|
|
|
Serial.println("To wipe the EEPROM saved data, set D4 (PinWIPE) to LOW - NOW! (2 seconds left)");
|
|
// wait a few seconds to let the user pull D4 down to wipe EEPROM
|
|
// and we can enjoy the boot screen meanwhile :p
|
|
// meanwhile blink with the led onboad :)
|
|
// 333 * 6 =~ 2 seconds
|
|
|
|
display.fillRect(0,36,128,64-36, 0);
|
|
display.setCursor(0,36);
|
|
display.println("To wipe EEPROM pull");
|
|
display.println("D4 (PinWIPE) to GND");
|
|
display.display();
|
|
|
|
// blink with the onboard LED on D4 (PinWIPE)
|
|
for(byte i = 0; i <= 6 ; i++) {
|
|
if(i % 2) {
|
|
digitalWrite(PinWIPE, LOW);
|
|
} else {
|
|
digitalWrite(PinWIPE, HIGH);
|
|
}
|
|
delay(333);
|
|
}
|
|
// set back to HIGH because thats the default
|
|
digitalWrite(PinWIPE, HIGH);
|
|
//delay(2000);
|
|
|
|
// read status from PinWIPE to WIPE
|
|
// when PinWIPE is set to LOW, wipe EEPROM
|
|
if(digitalRead(PinWIPE) == LOW) {
|
|
// wipe EEPROM
|
|
wipeEEPROM();
|
|
}
|
|
|
|
/*
|
|
* load EEPROM and Setup WiFi
|
|
*
|
|
* call loadEEPROM() which returns a bool
|
|
* When true, CanGrow is already configured and EEPROM values are applied
|
|
* When false, CanGrow is unconfigured and we need to run the setup assistant
|
|
*/
|
|
|
|
|
|
// load stored values from EEPROM and check what var configured is returned
|
|
if(loadEEPROM()) {
|
|
|
|
// connect to wifi
|
|
wifiConnect();
|
|
|
|
// configured is 0, setup Access Point
|
|
} else {
|
|
|
|
// start an wifi accesspoint
|
|
wifiAp();
|
|
|
|
}
|
|
// set web handler
|
|
WebHandler();
|
|
// start webserver
|
|
webserver.begin();
|
|
|
|
Serial.println(".:: CanGrow Ready ::.");
|
|
delay(1000);
|
|
if(strlen(GrowName) > 0 ) {
|
|
display.clearDisplay();
|
|
display.display();
|
|
}
|
|
|
|
// at the end of setup, set the outputs, when configured true
|
|
// we do this here because otherwise on inverted
|
|
// boards like CanGrow PCB v0.6 it would be turned on
|
|
if(configured == true) {
|
|
initOutputs();
|
|
}
|
|
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
*
|
|
* Loop
|
|
*
|
|
*
|
|
*/
|
|
void loop() {
|
|
// var definition
|
|
unsigned long currentRuntime = millis();
|
|
|
|
|
|
// first we call webserver handle client
|
|
webserver.handleClient();
|
|
|
|
|
|
// do every second when everything is configured and grow is started
|
|
if( (configured == true) && (strlen(GrowName) > 0) && (currentRuntime - outputPrevTime >= 1000) ){
|
|
|
|
// refresh all sensor values
|
|
refreshSensors();
|
|
|
|
// calculate VPD - https://www.grower.ch/forum/threads/diy-grow-controller-cangrow-projektvorstellung.163654/page-4#post-4294197
|
|
valVPD = (((100 - valHumidity) / 100) * (610.7 * (pow(10, (7.5 * valTemperature / (237.3 + valTemperature))))))/1000;
|
|
|
|
// calculate acutal DayOfGrow
|
|
DayOfGrow = int(ceil(float((timeClient.getEpochTime() - GrowStart) / 60 / 60 / 24)));
|
|
// decide if we are in Veg or Bloom phase of grow
|
|
// when DayOfGrow is larger then DaysVeg we must be in Bloom
|
|
|
|
// set the actual state of the Grow LED
|
|
// when being in Maintenance Mode and UseRelaisLED not true,
|
|
// dimm the light
|
|
if(MaintenanceMode == true) {
|
|
if((currentRuntime - MaintenanceStarted <= MaintenanceDuration * 1000 ) && (UseLEDrelais == false)) {
|
|
// in case of being in Maintenance Mode , dimm the grow light when not a relais is used
|
|
setOutput(1, 15);
|
|
} else {
|
|
MaintenanceMode = false;
|
|
}
|
|
} else {
|
|
controlLED();
|
|
}
|
|
|
|
controlPUMP();
|
|
|
|
controlFAN();
|
|
|
|
displayScreens();
|
|
|
|
// current time gets previous time for new interval
|
|
outputPrevTime = currentRuntime;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
*
|
|
* TODO LIST / NOTES
|
|
*
|
|
*
|
|
* - when PWM for fan is set, set fan speed to regulate humidity and
|
|
* temperature, depending on which phase of grow the plant is
|
|
* (https://www.royalqueenseeds.de/blog-cannabisanbau-im-grow-room-relative-luftfeuchtigkeit-und-temperaturen-n243)
|
|
* - re-organize EEPROM saved values.
|
|
* - prevent GrowStart to be in the future
|
|
* - maybe let the user configure some screens to display.
|
|
* - put EEPROM adresses into DEFINEs
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
* Fan control
|
|
*
|
|
* Vars:
|
|
* - FanVent (byte) Fan1 or Fan2
|
|
* - FanExhaust (byte) Fan1 or Fan2
|
|
* -
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
*
|
|
*
|
|
* PLAYGROUND / TRASH
|
|
*
|
|
*
|
|
*/
|
|
|
|
/*
|
|
|
|
unsigned long currentTime = millis();
|
|
|
|
int valSoilmoisture0 = getSoilmoisture(0);
|
|
int valSoilmoisture1 = getSoilmoisture(1);
|
|
|
|
float valTemperature0 = getTemperature(0);
|
|
float valTemperature1 = getTemperature(1);
|
|
|
|
float valHumidity = getHumidity();
|
|
|
|
int valWaterlevel = getWaterlevel();
|
|
|
|
switch(valWaterlevel) {
|
|
|
|
case 0:
|
|
digitalWrite(PinLED, HIGH);
|
|
digitalWrite(PinPUMP, LOW);
|
|
digitalWrite(PinFAN, LOW);
|
|
break;
|
|
case 1:
|
|
digitalWrite(PinLED, LOW);
|
|
digitalWrite(PinPUMP, HIGH);
|
|
digitalWrite(PinFAN, LOW);
|
|
break;
|
|
case 2:
|
|
digitalWrite(PinLED, LOW);
|
|
digitalWrite(PinPUMP, LOW);
|
|
digitalWrite(PinFAN, HIGH);
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
// OUTPUT
|
|
if(currentTime - outputPrevTime >= 1000) {
|
|
|
|
|
|
// set display cursor to top left
|
|
display.setCursor(0,0);
|
|
// display text
|
|
display.print("I2C: ");
|
|
display.print(valSoilmoisture1);
|
|
display.print(", ");
|
|
display.println(valTemperature1);
|
|
|
|
Serial.print("I2C: ");
|
|
Serial.print(valSoilmoisture1);
|
|
Serial.print(", ");
|
|
Serial.println(valTemperature1);
|
|
|
|
|
|
display.print("DHT11: ");
|
|
display.print(valTemperature0);
|
|
display.print(", ");
|
|
display.println(valHumidity);
|
|
|
|
Serial.print("DHT11: ");
|
|
Serial.print(valTemperature0);
|
|
Serial.print(", ");
|
|
Serial.println(valHumidity);
|
|
|
|
|
|
display.print("Water Status: ");
|
|
display.println(valWaterlevel);
|
|
|
|
Serial.print("Water Status: ");
|
|
Serial.println(valWaterlevel);
|
|
|
|
display.print("ASM: ");
|
|
display.print(valSoilmoisture0);
|
|
display.println(", ");
|
|
|
|
Serial.print("ASM: ");
|
|
Serial.println(valSoilmoisture0);
|
|
|
|
// print everything on the display
|
|
display.display();
|
|
|
|
Serial.println("Test");
|
|
|
|
outputPrevTime = currentTime;
|
|
*/
|
|
|
|
|
|
/* if(D6status == true) {
|
|
digitalWrite(PinLED, LOW);
|
|
digitalWrite(PinPUMP, LOW);
|
|
digitalWrite(PinFAN, LOW);
|
|
D6status = false;
|
|
Serial.println("D6 is off now");
|
|
} else {
|
|
digitalWrite(PinLED, HIGH);
|
|
digitalWrite(PinPUMP, HIGH);
|
|
digitalWrite(PinFAN, HIGH);
|
|
D6status = true;
|
|
Serial.println("D6 is ON now");
|
|
}
|
|
*/
|
|
|
|
/*
|
|
for(int dutyCycle = 0; dutyCycle < 255; dutyCycle++){
|
|
// changing the LED brightness with PWM
|
|
analogWrite(PinLED, dutyCycle);
|
|
delay(1);
|
|
}
|
|
|
|
// decrease the LED brightness
|
|
for(int dutyCycle = 255; dutyCycle > 0; dutyCycle--){
|
|
// changing the LED brightness with PWM
|
|
analogWrite(PinLED, dutyCycle);
|
|
delay(1);
|
|
}
|
|
*/
|
|
|
|
|