/*
************** UPDATED ON DEC 2017 *************
*************** GETTING BETTER THAN BEFORE *************
#############################################
FOR UPDATED 6.5 VERSION WATCH/READ DESCRIPTION:
##############################################
*** MAKING A MODULAR SYSTEM FOR D.I.Y. GROWBOX / GREENHOUSE
***Independent Grow Box Controller that can be used to Control all the Important devices to maintain
*
*
the ""Grow Environment"" for example Lights, Heater,Cooler,Fan for blowing air inside the box ,Fan for exhaust,
Humidifier or mist maker,Water pump ,Solenoids etc
*****i WILL improve it as i learn more :)
//**********************//**********************//**********************
//**********************//**********************//**********************//**********************//**********************//**********************
// Arduino Nano Grow-box / Green-House Controller v 5.0 :)
//**********************//**********************//**********************//**********************//**********************//**********************
//###############################################################
//***************************************************************
// PLEASE SUBSCRIBE
// comment for any question or help with this sketch
// My youtube Channel: https://youtube.com/CrazyGuyOfficial
//****************************************************************
//################################################################
// ***Made by Ali khan from Pakistan :)
//**********************//**********************//**********************
// **PARTS required (for this sketch to work as it is )
//**********************//**********************//**********************
1.Arduino Nano
2.A thermistor (i used A 10K WATERPROOF ONE)
3.A resistor (matching the value of the thermistor's "nominal resistance at 25 DEGREES C" for example: 5k resistor for 5k thermistor)
4.SPI OLED display ssd1306 (mine is ssd1306 but got it working by using sh1106 constructor also got rid of the "white line problem")
5. DHT 22 (WITH PULL UP RESISTOR I USED 10k OHM)
6.RELAY MODULE LOW LEVEL TRIGGER TYPE (6 Relays required in total for all the pins, can be separate 4 CAN WORK TOO )
7. DS3231 RTC MODULE (RESISTOR from charging circuit REMOVED and charging DISABLED..BECAUSE I USED CR2032(NON-rechargeable)Cell
8. AN LED WITH APPROPRIATE RESISTOR FOR USE AS A STATUS LED
9. Soil Humidity Sensor
10. Water Level Sensor
11. Two Stable 5 Volts DC Power Sources with Separate Grounds For Complete Optical Isolation (one can work)
*/
//**********************//**********************//**********************
// INCLUDE LIBRARIES make sure you have them all installed
//**********************//**********************//**********************
// Get them from within you Arduino IDE 1.6.9 that i used
// get the correct version
// by correct i mean the version that i used and
// because i can't guarantee that this sketch will work with
// other latest versions of the same libraries
#include "DHT.h" // use DHTsensorLibrary // Version 1.2.3 //latest is the AdafruitUnifiedSensorLibrary
#include <U8glib.h> // OLED // Version 1.19.1
#include <Wire.h> // I2C // Version 1.0.0
#include <Time.h> // Time Manipulation // Version 1.5.0
#include <DS1307RTC.h> // DS1307 RTC // Version 1.4.0
#include <TimeAlarms.h> // time alarm library // Version 1.5.0
// Changed Number of alarms " dtNBR_ALARMS to 8 "
// BY EDITING TIMEALARM.H FILE IN LIBRARIES FOLDER.
// increasing this number increases size of the sketch too
// and the use of ram
// if you don't want to change the number then remove alarms and keep only
// that number of Alarms that is specified in YOUR TimeAlarms Library
// I have Left 8 Alarms BELOW that Call a Function to Sync Arduino's Clock with RTC
// if you Uncomment the Rest you will get some Errors
// if you increase the Number in TimeAlarms.H file then
// Don't UN-comment Void SPrintStatusTwo
// That prints out Switch Status
// OR you will get Low Memory Error for Nano/UNO
#define DS1307_ID 0x68 // I2c Address of the RTC
//######################################################################
//******************************************************************/
// TEMPERATURE and HUMIDITY DEFAULT "CONTROL PARAMETERS"
//******************************************************************\
//######################################################################
// Default Temperature and Humidity Settings
byte Tmax = 29 ; // Set temperature Maximum point at which "Cooler" will turn ON
byte Tnor = 25 ; // Set temperature point at which "heater" and "cooler" will turn OFF
byte Tmin = 19 ; // Set temperature Minimum point at which Heater will turn ON
byte Hmax = 60 ; // set humidity max at which "bigfan" will turn ON
byte Hnor = 45 ; // Set humidity normal at which "bigfan" and "spray" will turn OFF
byte Hmin = 25 ; // Set humidity low at which "spray will turn ON
// Separate Temperature and Humidity Settings for Day and Night time can be done by Alarms function belowwww
// but anything done by Alarms is lost if a Reset happens or power up/down cycle
// and so in that case everything will return to their default state
//######################################
//*********************************/
// PUMP TURN ON/OFF Settings ####
//*********************************\
//######################################
byte pumpT = 30 ; // Time in Seconds for which the Pump will run if Soil goes Dry
// after the Time is up the pump will Turn Off
byte dryVal = 50 ; // Soil Humidity percentage Value below which the Pump will be Turned ON
// when the Alarm Calls the WateringA function
// pump will turn OFF after the Time "pumpT" is up
int DV = 1023 ; // Dry Value the MAX Analog Read Value Above which the Soil gets DRY //max 1023
int HV = 0 ; // Humid / Wet Value below which the Soil should not be watered // min 0
// these numbers are used to "map" analog read values in SoilHumCalc Function
// can be used for Calibration
byte waterOne = 50 ; // Water Level Sensor 1 percentage Value below which the water reservoir will be considered empty
int FV1 = 800 ; // MAX value when Tank is FULL // i used it because my sensor doesn't give out 5V/max 1023 when 100% wet
int EV1 = 0 ; // MIN Value Below Which Tank is Empty // min 0
// Separate Function for watering using the Digital/Analog Read method for Soil Humidity Sensor's Digital/Analog output Pins
// can be called from alarm functions belowwwww
//##################################################
//***********************************************/
// Light Timer ON/OFF Time Settings DEFAULT
//***********************************************\
//##################################################
const byte OnTime = 14 ; // Hour when Light will turn ON (24 hr format NO zeros before single digits)
const byte OffTime = 8 ; // Hour when Light will turn OFF (24 hr format NO zeros before single digits)
// SET RTC'S TIME USING DS1307 EXAMPLE SKETCH FROM ABOVE FILE > EXAMPLE > DS1307 > SETTIME
// THEN UPLOAD THIS SKETCH
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//***************************************************
// MAKING CHARACTER ARRAYS or C strings for OLED
//***************************************************
char str[10]; // for Display
//#################################################################################################################
//*****************************************************************************************************************
//************ some important variables *************** Values NOT TO BE CHANGED MANUALLY *************************
//*****************************************************************************************************************
//#################################################################################################################
float t2 = 0; // for Temperature 2 from Thermistor
float t =0; // floats for Temperature1 and Humidity from DHT
float h =0;
byte TempSts = 0 ; // used for updating the oled display
byte HumSts = 0 ;
float TmaxR = 0; // for Recording Max and Min Values from DHT
float HmaxR = 0;
float TminR = 0;
float HminR = 0;
bool DHTerr = 0 ;
bool RTCerr = 0 ;
bool LightSts = 0 ; // THESE ARE FOR STATUS UPDATES IN SERIAL MONITOR
bool CoolerSts = 0 ; // and ESP8266 to show on Html Webpage
bool HeaterSts = 0 ;
bool BigFanSts = 0 ;
bool SpraySts = 0 ;
bool PumpSts = 0 ;
int soilHumVal = 0 ; // for reading analog value from Soil Humidity Sensor's Analog output
bool waterTimeA = 0 ;
bool waterTimeD = 0 ;
int waterLvlOneVal = 0 ; // for reading analog value from Water Level Sensor's Analog output
bool REmpty = 0 ; // water reservoir empty
unsigned long Secs = 0;
unsigned long SecsTwo = 0;
unsigned long SecsThree = 0;
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//##################################
//********************************/
// PINS AND CONNECTIONS
//********************************\
//##################################
const byte Bigfan = 2 ; // Bigfan pin turns on at high humidity should be Two fans (inside,exhaust)
const byte cooler = 3 ; // Cooler pin turns on at high temp (T1 from DHT)
const byte spray = 4 ; // Spray turns on at low humidity
const byte lightone = 5 ; // Main Lights(for plants)
const byte heater = 6 ; // Heater pin turns ON at low temp (T1 from DHT)
const byte lighttwo = 7 ; // pin for "status LED'
const byte pumpPin = 8 ; // pin for turning the Water Pump ON/OFF
const byte waterLvlOne = A2 ; // pin for water level Sensor's Signal output pin it gives it gives raw voltage values
const byte soilHumD = A3 ; // pin for Soil Humidity Sensor's DIGITAL Output Signal Pin it goes High/Low
const byte soilHumA = A7 ; // pin for Soil Humidity Sensor's ANALOG Output Signal Pin it gives raw voltage values
// ******** THERMISTOR AND DHT PINS BELOW *******
// Connect the RTC according to it's pinout
// DS3231 RTC "SDA to A4", "SCL to A5"(uno/nano) "Vcc to 5V arduino pin Gnd to Gnd"
//####################################
//********************************/
// THERMISTOR PIN and SETTINGS
//********************************\
//####################################
#define THERMISTORPIN A0 // analog pin to connect to thermistor and the resistor "joint"
#define THERMISTORNOMINAL 10000 // resistance at 25 degrees C
#define TEMPERATURENOMINAL 25 // temp. for nominal resistance (almost always 25 C)
#define NUMSAMPLES 5 // how many samples to take and average,takes longer gets 'smooth'
#define BCOEFFICIENT 3500 // The beta coefficient of the thermistor (usually 3000-4000)
#define SERIESRESISTOR 9700 // the value of the 'other' resistor should be close to the thermistors nominal value
int samples[NUMSAMPLES];
//################################
//****************************/
// DHT PIN and SETTINGS
//****************************\
//################################
#define DHTPIN A1 // pin connected to DHT output
#define DHTTYPE DHT22 // DHT 22 (AM2302)
DHT dht(DHTPIN, DHTTYPE, 6); // the number 6 is for slower chips like the one used in Arduino Nano/UNO
//######################################################################
//*****************************************************************/
// OLED CONSTRUCTOR / OLED PINS that are connected to Arduino
//*****************************************************************\
//######################################################################
U8GLIB_SH1106_128X64 u8g(13, 11, 10, 9); // SW SPI Com: SCK = 13, MOSI = 11, CS = 10, A0 = 9 // OLED "constructor" and pins info.
// the version of oled i have has a different chip inside so the above one works for me
// before it i used the one that was supposedly for my version of oled and it was not working as expected
// if this doesn't work for you then go to Example Sketches for U8glib Library it has All the Constructors
// try them for your version of OLED
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//**************************
// SETUP BEGINS
//**************************
void setup()
{
digitalWrite(Bigfan, HIGH); // When using "LOW level trigger" type Relay module, turning a pin "HIGH" will
digitalWrite(cooler, HIGH); // turn the relay "OFF" and turning the pin "LOW" turns the relay "ON"
digitalWrite(heater, HIGH); // THIS WILL PREVENT STARTUP RELAY TRIGGERING PROBLEM
digitalWrite(spray, HIGH);
digitalWrite(pumpPin,HIGH);
digitalWrite(lightone,HIGH);
// add more if you have more relays or pins connected to something
Serial.begin(9600); // for initializing serial monitor
pinMode(Bigfan, OUTPUT) ; // set output relay pins // add more if you have more relays or pins connected to something
pinMode(cooler, OUTPUT) ;
pinMode(heater, OUTPUT) ;
pinMode(spray, OUTPUT) ;
pinMode(lightone,OUTPUT) ;
pinMode(lighttwo,OUTPUT) ;
pinMode(pumpPin,OUTPUT) ;
pinMode(soilHumD, INPUT) ;
pinMode(soilHumA,INPUT) ;
pinMode(waterLvlOne,INPUT) ;
// add more if you have more relays or pins connected to something
analogReference(EXTERNAL); // AREF connected to 3.3 ON BOARD for accuracy of thermistor
//(go to adafruit using a thermistor tutorial )
//******************************************
//********* Display at Startup ********
//******************************************
Start();
Alarm.delay(3000);
LightSet();
Alarm.delay(5000);
TempSet();
Alarm.delay(5000);
HumSet();
Alarm.delay(5000);
//**********************//**********************//***********************//**********************
//Check to see if the RTC is present.if yes then Set the ARDUINO's INTERNAL clock accordingly
//**********************//**********************//***********************//**********************
// when using timer library the Arduino's "internal clock " is used for getting "current time"
// Arduino's internal clock is NOT so accurate and gets off time so we use time from RTC (which IS
// accurate at-least for our use) and we sync Arduino's internal Clock with the time from RTC and we
// have to do this periodically (at-least once in every 24 hrs )so our tasks can be completed according
// to the correct time we can put the sync command in the Loop and Alarm Functions too
Wire.beginTransmission(DS1307_ID); // i2c devices have addresses which are used to communicate with them
Wire.write((uint8_t)0x00); // because same interface can be used to connect multiple i2c devices
setSyncProvider(RTC.get); // Configure Time to automatically called the getTimeFunction() regularly.
setSyncInterval(60);
//#####################
Alarm.timerRepeat(10, Repeats); // timer for every 10 seconds blinking LIGHT TWO (status LED)
// and Serial Prints Data
// TIMER ALARM THAT RUNS OVER AND OVER AFTER SECONDS // the Alarm in the Loop are different
// see the Example Sketches with the TimeAlarms Lib
dhtCalc() ; // call DHT,Thermistor calculation and Soil Humidity functions at setup
thermCalc() ; // so the relays don't trigger accidentally in those two seconds after which the loop calls those functions again
soilHumCalc() ;
waterLvlCalc() ;
Secs=SecsTwo=millis()/1000; // reset the timers and then in the loop it will be used again and again
}
//**************************************************************
//********* SETUP END ****************************************
//**************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*************************************************
//*** Function that Syncs Arduino's Clock with Rtc
//*************************************************
void SyncTime()
{
if (Wire.endTransmission() == 0) //Did the RTC respond?
{
setSyncProvider(RTC.get); // it did, get the time from the RTC
RTCerr=0; // set RTC error status boolean to false
// Serial.println("Got Time");
}
else
{
RTCerr=1;
}
}
//**************************************************
//*** Function For Display on the OLED *************
//**************************************************
void Start()
{
u8g.firstPage(); // START SCREEN "Picture loop" for OLED
do
{
u8g.setFont(u8g_font_helvB08); // set font for oled display
u8g.drawFrame(10, 12, 108, 31); // Draw Frame and it's Size
u8g.drawFrame(0, 0, 128, 64); // Draw Frame and it's Size
u8g.drawStr(15, 25, "Grow Box"); // Draw a string mentioned in quotes " "
u8g.setPrintPos(79,32); // Set print position for the next line
u8g.print("V 5.0"); // Print on the oled
u8g.drawStr( 16, 40, "Controller");
u8g.drawStr(53, 60, "Made by:AMK");
}
while ( u8g.nextPage() );
}
//***************************************************************
//*** Function For Light Control Settings Display on the OLED ***
//***************************************************************
void LightSet()
{
u8g.firstPage(); // START SCREEN "Picture loop" for OLED
do
{
DrawFrame();
u8g.drawStr(4, 14, "Light Control Hours");
u8g.drawStr( 4, 34,"ON:");
u8g.setPrintPos(31,34);
u8g.print(OnTime);
u8g.drawStr( 4, 46,"OFF:");
u8g.setPrintPos(37,46);
u8g.print(OffTime);
}
while ( u8g.nextPage() );
}
//*********************************************************************
//*** Function For Temperature Control Settings Display on the OLED ***
//*********************************************************************
void TempSet()
{
u8g.firstPage(); // //START SCREEN "Picture loop" for OLED
do
{
DrawFrame();
u8g.drawStr(4, 14, "Temp Control Settings");
u8g.drawStr( 4, 34,"Max:");
u8g.setPrintPos(39,34);
u8g.print(Tmax);
u8g.drawStr( 58, 34, "\260C"); // make Degree Centigrade Symbol
u8g.drawStr( 4, 46,"Nor:");
u8g.setPrintPos(39,46);
u8g.print(Tnor);
u8g.drawStr( 58, 46, "\260C");
u8g.drawStr( 4, 58,"Min:");
u8g.setPrintPos(39,58);
u8g.print(Tmin);
u8g.drawStr( 58, 58, "\260C");
}
while ( u8g.nextPage() );
}
//*********************************************************************
//*** Function For Humidity Control Settings Display on the OLED ***
//*********************************************************************
void HumSet()
{
u8g.firstPage(); // //START SCREEN "Picture loop" for OLED
do
{
DrawFrame();
u8g.drawStr(4, 14, "Humidity Settings");
u8g.drawStr( 4, 34,"Max:");
u8g.setPrintPos(39,34);
u8g.print(Hmax);
u8g.drawStr( 58, 34, "%");
u8g.drawStr( 4, 46,"Nor:");
u8g.setPrintPos(39,46);
u8g.print(Hnor);
u8g.drawStr( 58, 46, "%");
u8g.drawStr( 4, 58,"Min:");
u8g.setPrintPos(39,58);
u8g.print(Hmin);
u8g.drawStr( 58, 58, "%");
}
while ( u8g.nextPage() );
}
void DrawFrame()
{ u8g.drawFrame(0, 0, 128, 20); // Draw frame
u8g.drawFrame(0, 22, 128, 42); // Draw frame
}
//########################################################################################
//***************************************************************************************/
//************ Function Called When Repeating Timer alarm triggers ******************
//***************************************************************************************\
//########################################################################################
void Repeats()
{
if(!DHTerr && !RTCerr) // blink the light only when DHTerr is not 1 because then status LED will constantly stay ON
{
digitalWrite(lighttwo, HIGH); // it is also used to indicate Dht read failed and lights up continuously below in dht part
Alarm.delay(30);
digitalWrite(lighttwo, LOW);
}
//TimeDateDisplayOne(); // Update the Clock Display on Tx line
// this calls the function TimeDateDisplayOne
// includes time,date,day,year only numbers
// Formatted to be read by ESP8266 using Serial.Read
//SPrintStatusOne(); // SerialPrints data on the Tx line of Arduino
// by connecting the Rx line of Esp8266 / Wemos D1 mini
// that is running an html webserver
// that data is read and categorized in Arrays and then displayed
// on a simple html page
// *************
// * IMPORTANT * : if anything is changed Regarding serial.print/Serial.println
// ************* anywhere in the Sketch it WILL effect the Results of the sketch
// running in Esp8266 because of the changed position/content of the
// data printed on Tx line and Read by Rx of Esp8266 so keep it in mind
TimeDateDisplayTwo(); // For Humans Only
SPrintStatusTwo(); // For Humans Only
}
//##################################################################################################################
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//##################################################################################################################
//########################################################################
//***********************************************************************/
//********** LOOP BEGINS ******************************************
//***********************************************************************\
//########################################################################
void loop()
{
if (hour() == 0 && minute() == 0 && second() == 0) //If it is 00:00:00 MIDNIGHT UPDATE the Arduino's INTERNAL Clock
{
SyncTime();
}
//####################################################################################################
//####################################################################################################
//###### LET THERE BE LIGHT ########################################################################
//####################################################################################################
//####################################################################################################
// This will make sure that the light one (for plants) stays ON after reset or power down/up cycle
// because Alarms will trigger on a given time and will turn the pin HIGH but if after the alarm is
// triggered AND there is a reset the pin will turn off until next alarm that turns it ON again
// which can interrupt photo-period for plants So, this makes sure that LIGHTS turn ON immediately
// after reset which happens DURING the LIGHT period for plants
//####################################################################################################
// Turn Lights On/Off
//####################################################################################################
// if(!RTCerr) // only if RTC error is 0 or False
// {
// IF Turning ON the Light is Required on one day and Turning OFF on the same day
// within the 24 hrs of that day 00:00 --- 11:59
if (OffTime > OnTime) // Same Day
{ // GETS THE JOB DONE :)
if (hour() >= OnTime && hour () <= OffTime - 1)
{
// if current hour is equal to or greater than ON time
// AND equal to or less than OFF time minus 1 then turn the
// lights ON (by turning pins LOW because of using LOW level Relays)
// Minus 1 from OFF time because we are not using Minutes here
// For Example turn off time is "16" (4 PM) the hour number will be "16" from
// 16:00 till 16:59 (4:00 till 4:59)
if (!LightSts)
{
digitalWrite(lightone, LOW);
Alarm.delay(2000);
LightSts = 1 ;
}
}
else
{
digitalWrite(lightone, HIGH);
LightSts = 0 ;
}
}
//####################################################################################################
if (OffTime < OnTime) // Different Day
{
// IF Turning ON the Light is Required on ONE DAY and
// Turning OFF on the SECOND DAY
if (hour() >= OnTime || hour () <= OffTime - 1)
{
// if current hour is equal or greater than ON time
// OR equal to or less than OFF time minus 1 then turn the
// lights ON (by turning pins LOW because of using LOW level Relays)
// Minus 1 from OFF time because we are not using Minutes here
// For Example turn off time is "16" (4 PM) the hour number will be "16" from
// 16:00 till 16:59 (4:00 till 4:59)
// Minutes,Seconds even day,date,year can be used by adding code
if (!LightSts)
{
digitalWrite(lightone, LOW);
Alarm.delay(2000);
LightSts = 1 ;
}
}
else
{
digitalWrite(lightone, HIGH);
LightSts = 0 ;
}
}
// }
// else
// {
// digitalWrite (lighttwo,HIGH); // turn status LED indicate RTC error
// //Serial.println("Something Wrong with RTC.So, NOT Controlling Lights Now");
// }
if (RTCerr)
{
digitalWrite (lighttwo, HIGH); // turn status LED indicate RTC error
}
//###############################################################################################################
// *************************************************************************************************************/
// DAILY TIMED ALARMS THAT TRIGGER ON SPECIFIED TIMES
// these alarms will trigger once and will do "stuff" mentioned in the alarm functions at the end of the loop
// *************************************************************************************************************\
//###############################################################################################################
Alarm.alarmRepeat(3, 0, 0, ONAlarm); // daily Alarm 1 for example // for triggering an alarm on ""03:00:00"" in the morning AM
// ONAlarm is the name of the function that will be called when the current time matches
// the Time you have set for the specified alarm
// the ONAlarm function is defined below and the name ONAlarm can be changed
// the fuctions will be called only at the moment when the time matches the alarm time
// and if the state of switch is changed and then Arduino is reset the state will change back to default
Alarm.alarmRepeat(6, 0, 0, OFFAlarm); // daily Alarm 2
Alarm.alarmRepeat(9, 0, 0, OFFAlarm); // daily Alarm 3
Alarm.alarmRepeat(12, 0, 0, OFFAlarm); // daily Alarm 4
Alarm.alarmRepeat(15, 0, 0, OFFAlarm); // daily Alarm 5
Alarm.alarmRepeat(18, 0, 0, OFFAlarm); // daily Alarm 6
Alarm.alarmRepeat(21, 0, 0, OFFAlarm); // daily Alarm 7
/*
Alarm.alarmRepeat(15, 35, 0, ON2Alarm); // daily Alarm 5
Alarm.alarmRepeat(18, 37, 0, wateringD); // daily Alarm 6
Alarm.alarmRepeat(21, 38, 0, wateringA); // daily Alarm 7 */
//############################################################
//#####################################
// Get Sensor Values after every 2 Secs
//#####################################
if (millis() / 1000 - Secs > 2 ) // Call dht and thermistor calculation fuction every 2 secs
{
dhtCalc() ;
thermCalc() ;
soilHumCalc() ;
waterLvlCalc() ;
if (waterLvlOneVal < waterOne)
{
REmpty = 1 ;
u8g.firstPage();
do
{
DrawFrame();
u8g.drawStr(4, 14, "Water Tank");
u8g.drawStr( 14, 44, "* EMPTY *");
}
while ( u8g.nextPage() );
Alarm.delay(1300);
}
else
{
REmpty = 0 ;
}
Secs = millis() / 1000; // then reset the timer
}
//#########################################################################################################
//###############################################################################################################
// CONTROL PART turning things ON and OFF according to Temperature and Humidity (THE GOOD STUFF) "ifs"
//###############################################################################################################
//#########################################################################################################
if (!DHTerr) // only proceed to controlling things according to Temp1 and humidity
{ // if there is no DHT read error
if (t >= Tmax ) // TURN COOLER ON IF TEMPERATURE IS EQUAL TO OR EXCEEDS "Tmax" (HEATER remains OFF)
{
if (!CoolerSts)
{
digitalWrite(cooler, LOW);// Turn Cooler Pin LOW // ON because of LOW Level trigger Relay
Alarm.delay(2000);
CoolerSts = 1 ; // Cooler Status is ON // Status is used for Serial Print
}
TempSts = 4; // High Temp // Status is used by Oled
}
if (t <= Tnor ) // Turn OFF the COOLER when TEMP GETS DOWN TO "setT" OR GOES BELOW
{
digitalWrite(cooler, HIGH);
CoolerSts = 0 ;
}
if (t <= Tmin) // TURN HEATER "ON" WHEN TEMPERATURE GETS TO "Tmin" OR BELOW (cooler remains OFF)
{
if (!HeaterSts)
{
digitalWrite(heater, LOW);
Alarm.delay(2000);
HeaterSts = 1 ;
}
TempSts = 1; // Low Temp
}
if (t >= Tnor ) // TURN OFF HEATER WHEN TEMP GETS TO "setT" (and REMAINS LOWER THAN "Tmax")
{
digitalWrite (heater, HIGH);
HeaterSts = 0 ;
}
//######################################/
//######################################/
//# Just for Setting Status of things #
//######################################/
//######################################\
if (t <= Tmax && t > Tnor)
{
TempSts = 3; // Upper Range
}
if (t <= Tnor && t > Tmin)
{
TempSts = 2; // normal range
}
if ( t > TmaxR)
{
TmaxR = t; // Update Max Recorded Temperature For the Session
}
if ( t < TminR)
{
TminR = t; // Update Minimum Recorded Temperature For the Session
}
else if (TminR == 0)
{
TminR = t;
}
//#######################################################
//############### Control for HUMIDITY ###############
//#######################################################
if (h >= Hmax)
{
if (!BigFanSts)
{
digitalWrite(Bigfan, LOW); // Turn fan on when humidity is high
Alarm.delay(2000);
BigFanSts = 1 ;
}
HumSts = 4; // humidity max/high
}
if (h <= Hnor )
{
digitalWrite(Bigfan, HIGH);
BigFanSts = 0;
}
if (h <= Hmin)
{
if (!SpraySts)
{
digitalWrite(spray, LOW); // Turn Spray on when humidity is low
Alarm.delay(2000);
SpraySts = 1 ;
}
HumSts = 1; // humidity low
}
if (h >= Hnor)
{
digitalWrite(spray, HIGH); // Turn Spray off when humidity is normal or high
SpraySts = 0 ;
}
if (h > Hnor && h < Hmax)
{
HumSts = 3; // upper range
}
if (h > Hmin && h <= Hnor)
{
HumSts = 2; // normal range
}
if ( h > HmaxR)
{
HmaxR = h; // Update Max Recorded Humidity For the Session
}
if ( h < HminR)
{
HminR = h; // Update Max Recorded Humidity For the Session
}
else if (HminR == 0)
{
HminR = h;
}
}
else
{
digitalWrite (lighttwo, HIGH); // turn status LED indicate DHT error
//Serial.println("Something Wrong with DHT.So, NOT Controlling Things Now");
digitalWrite(Bigfan, HIGH);
digitalWrite(cooler, HIGH);
digitalWrite(heater, HIGH);
digitalWrite(spray, HIGH);
}
//##############################################################
//************************************************************//
//*************** Control for Water Pump ***********
//***********************************************************//
//#############################################################
//###############################################/
// Using Soil Humidity Sensor's Digital OUTPUT
//###############################################\
if(waterTimeD) // using Digital OUT Pin from Soil Humidity Sensor
{
if(digitalRead (soilHumD) == HIGH && !REmpty && !PumpSts)
{
digitalWrite(pumpPin,LOW);
Alarm.delay(2000);
PumpSts = 1 ;
}
else if (millis()/1000 -SecsThree > pumpT && PumpSts)
{
digitalWrite(pumpPin,HIGH);
PumpSts = 0 ;
waterTimeD = 0;
}
}
//###############################################/
// Using Soil Humidity Sensor's Analog OUTPUT
//###############################################\
if(waterTimeA) // Using Analog Pin from Soil Humidity Sensor
{
if(soilHumVal < dryVal && !REmpty && !PumpSts)
{
digitalWrite(pumpPin,LOW);
PumpSts = 1 ;
}
else if (millis()/1000 -SecsThree > pumpT && PumpSts)
{
digitalWrite(pumpPin,HIGH);
PumpSts = 0 ;
waterTimeA = 0;
}
}
//###################################
//###################################
Display(); // Call the Main Display Loop function (beloww) for OLED
//#######################################
// Call Recorded Max & Min Display loops
//#######################################
if(millis()/1000 -Secs > 60 ) // Call Max & Min Recorded Temp & Hum Display Loop every 60 secs
{
RecMax();
Alarm.delay(2000); // and show them for 5 secs each
RecMin();
Alarm.delay(2000);
Secs=millis()/1000; // reset the timer "secs"
}
}
//**********************//**********************//**********************
// LOOP END
//**********************//**********************//**********************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//*****************************************************************************************************************
//********************//**********************//************//
// Function for DHT 22 TEMPERATURE HUMIDITY CALCULATIONS
//********************//**********************//************//
void dhtCalc()
{
h = dht.readHumidity(); // READ HUMIDITY
t = dht.readTemperature(); // Read temperature as Celsius
if (isnan(h) || isnan(t) )
{
Alarm.delay(500);
DHTerr=1;
}
else
{
DHTerr=0;
}
}
//********************************************
// Function for THERMISTOR CALCULATIONS
//********************************************
void thermCalc()
{
byte i;
float average =0 ;
for (i = 0; i < NUMSAMPLES; i++) // take "N" number of samples in a row, with a slight delay
{
samples[i] = analogRead(THERMISTORPIN);
Alarm.delay(10);
}
for (i = 0; i < NUMSAMPLES; i++) // average all the samples out
{
average += samples[i];
}
average /= NUMSAMPLES;
average = 1023 / average - 1; // convert the value to resistance
average = SERIESRESISTOR / average;
t2= average / THERMISTORNOMINAL; // (R/Ro)
t2 = log(t2); // ln(R/Ro)
t2 /= BCOEFFICIENT; // 1/B * ln(R/Ro)
t2 += 1.0 / (TEMPERATURENOMINAL + 273.15); // + (1/To)
t2 = 1.0 / t2; // Invert
t2 -= 273.15; // convert to C
}
//***************************************
// Function for Reading Soil Humidity
//***************************************
void soilHumCalc()
{
soilHumVal= analogRead(soilHumA) ;
//Serial.print("Soil Hum: ");
//Serial.println(soilHumVal);
soilHumVal = map(soilHumVal,DV,HV,0,100);
}
//***************************************
// Function for Reading Water Level
//***************************************
void waterLvlCalc()
{
waterLvlOneVal= analogRead(waterLvlOne) ;
//Serial.print("Water Lv: ");
//Serial.println(waterLvlOneVal);
waterLvlOneVal = map(waterLvlOneVal,EV1,FV1,0,100);
}
//**************************************************
// Main Loop FUNCTION FOR Main DISPLAY ON THE OLED
//**************************************************
void Display()
{
char timebuf[5]; // array for Time
tmElements_t tm; //formatting time and date before displaying
if (RTC.read(tm))
{
sprintf(timebuf, "%02d:%02d", tm.Hour, tm.Minute); // format time
}
u8g.firstPage();
do
{
DrawFrame();
if(DHTerr) // if DHT error is 1 or true then display dht error on oled
{
u8g.drawStr( 35, 14, "* DHT ERROR *");
}
else // else display values
{
switch (HumSts) // Display According to Humidity Status on OLED
{
case 4:
u8g.drawStr( 91, 14, "*H MX*"); // humidity is equal to max or more than max
break;
case 3:
u8g.drawStr( 91, 14, "*H AN*"); // above normal
break;
case 2:
u8g.drawStr( 91, 14, ">H NR<"); // Normal
break;
case 1:
u8g.drawStr( 91, 14, "*H LW*"); // Lower than Minimum Humidity
break;
}
//*********************************************************************
switch (TempSts) // Display According to Temperature Status on OLED
{
case 4:
u8g.drawStr( 34, 14, "*HI TMP*");
break;
case 3:
u8g.drawStr( 34, 14, "*ABV NR*");
break;
case 2:
u8g.drawStr( 34, 14, ">NR TMP<");
break;
case 1:
u8g.drawStr( 34, 14, "*LO TMP*");
break;
}
}
//************// Display Time on OLED//**************//
if(!RTCerr)
{
u8g.setPrintPos(2, 14); // set position for displaying time
u8g.print(timebuf); //display time
}
else
{
u8g.drawStr( 2, 14, "RtcEr");
}
//****** Display CURRENT Temperature 1 & 2 , humidity,Set Tmax,Set Tnor and Set Tmin,Soil Humidity and Water Level on OLED *********************
u8g.drawStr(2, 32, "T :"); // first digit is screen coordinate for X-axis and second for Y-axis
// (it starts on left bottom corner of the text which is to be shown on the oled)
// T for temperature from DHT22 and T2 is from Thermistor
u8g.drawStr( 21, 32, dtostrf(t, 5, 2, str));
u8g.drawStr( 51, 32, "\260C");
u8g.drawStr( 2, 42, "M :");
u8g.drawStr( 21, 42, dtostrf(Tmax, 5, 2, str));
u8g.drawStr( 51, 42, "\260C");
u8g.drawStr( 2, 52, "N ");
u8g.drawStr( 15, 52, ":");
u8g.drawStr( 21, 52, dtostrf(Tnor, 5, 2, str));
u8g.drawStr( 51, 52, "\260C");
u8g.drawStr( 2, 62, "m ");
u8g.drawStr( 15, 62, ":");
u8g.drawStr( 21, 62, dtostrf(Tmin, 5, 2, str));
u8g.drawStr( 51, 62, "\260C");
u8g.drawStr( 65, 32, "T2:");
u8g.drawStr( 83, 32, dtostrf(t2, 5, 2, str));
u8g.drawStr( 113, 32, "\260C");
u8g.drawStr( 65, 42, "H :");
u8g.drawStr( 83, 42, dtostrf(h, 5, 2, str));
u8g.drawStr( 117, 42, "%");
u8g.drawStr( 67, 52, "SH:"); // Soil Humidity Value mapped to 0-100 %
u8g.setPrintPos(92, 52);
u8g.print(soilHumVal);
u8g.drawStr( 117, 52, "%");
u8g.drawStr( 67, 62, "WL:"); // water level mapped to 0-100 %
u8g.setPrintPos( 92, 62);
u8g.print(waterLvlOneVal);
u8g.drawStr( 117,62, "%");
}
while ( u8g.nextPage() );
}
//********************************
// Display Loop for Max Recorded
//********************************
void RecMax()
{
u8g.firstPage();
do
{
DrawFrame();
u8g.drawStr(4, 14, "Max Recorded");
u8g.drawStr( 4, 34,"T Max:");
u8g.setPrintPos(47,34);
u8g.print(TmaxR);
u8g.drawStr( 75, 34, "\260C");
u8g.drawStr( 4, 46,"H Max:");
u8g.setPrintPos(47,46);
u8g.print(HmaxR);
u8g.drawStr( 79, 46, "%");
}
while ( u8g.nextPage() );
}
//********************************
// Display Loop for Min Recorded
//********************************
void RecMin()
{
u8g.firstPage();
do
{
DrawFrame();
u8g.drawStr(4, 14, "Min Recorded");
u8g.drawStr( 4, 34,"T Min:");
u8g.setPrintPos(47,34);
u8g.print(TminR);
u8g.drawStr( 75, 34, "\260C");
u8g.drawStr( 4, 46,"H Min:");
u8g.setPrintPos(47,46);
u8g.print(HminR);
u8g.drawStr( 79, 46, "%");
}
while ( u8g.nextPage() );
}
//**********************//**********************//**********************//**********************
//Functions called when daily time specific alarms (above in the loop) triggers:
//***********************************************//**********************//**********************
/*
void daySett()
{
//Tmax = 29; // set day time temperatures
//setT = 25;
//Tmin = 19;
SyncTime();
}
void nightSett()
{
//Tmax = 26; // set night time temperatures
//setT = 23;
//Tmin = 19;
SyncTime();
}
*/
void OFFAlarm() // when this fuction is called by the alarm it calls another fuction Synctime which syncs arduino's internal clock with RTC
{
SyncTime();
}
void ONAlarm() // this fuction calls two other fuctions
{
SyncTime();
}
void ON2Alarm() // this fuction calls two other fuctions
{
wateringA();
SyncTime();
}
//*******************************************************************
// Function for Watering using soil Humidity Sensor's Digital Output
//*******************************************************************
void wateringD()
{
SecsThree= millis()/1000; // reset the timer
waterTimeD= 1 ;
//Serial.println("water time D Flag set");
}
//*******************************************************************
// Function for Watering using soil Humidity Sensor's Analog Output
//*******************************************************************
void wateringA()
{
SecsThree= millis()/1000; // reset the timer
waterTimeA = 1 ;
//Serial.println("water time A Flag set");
}
///**************************************************************************************
//*** Function for Serial Printing status of switches and other values ** FOR ESP8266 *
//**************************************************************************************
/*
void SPrintStatusOne()
{
Serial.print ('L');
Serial.print (LightSts);
Serial.print ('C');
Serial.print (CoolerSts);
Serial.print('H');
Serial.print(HeaterSts);
Serial.print('B');
Serial.print(BigFanSts);
Serial.print('S');
Serial.print(SpraySts);
Serial.print('E');
Serial.print(DHTerr);
Serial.print(RTCerr);
Serial.print('t');
Serial.print(t);
Serial.print('h');
Serial.print(h);
Serial.print('r');
Serial.println(t2);
}
*/
//*************************************************************************************
//*** Function for Serial Printing status of switches and other values ** FOR HUMAN ***
//*************************************************************************************
void SPrintStatusTwo()
{
Serial.print("T1: ");
Serial.println(t);
Serial.print("H: ");
Serial.println(h);
Serial.print("T2: ");
Serial.println(t2);
Serial.print("Water Lv: ");
Serial.println(waterLvlOneVal);
Serial.print("Soil Hum: ");
Serial.println(soilHumVal);
if(LightSts)
{
Serial.println ("Light ON");
}
else
{
Serial.println("Light OFF");
}
if(CoolerSts)
{
Serial.println("Cooler ON");
}
else
{
Serial.println("Cooler OFF");
}
if(HeaterSts)
{
Serial.println("Heater ON");
}
else
{
Serial.println("Heater OFF");
}
if(BigFanSts)
{
Serial.println("Fan ON");
}
else
{
Serial.println("Fan OFF");
}
if(SpraySts)
{
Serial.println("Spray ON");
}
else
{
Serial.println("Spray OFF");
}
if(PumpSts)
{
Serial.println("Pump ON");
}
else
{
Serial.println("Pump OFF");
}
}
//**********************//**********************//********************************
//Function for digital clock display (in the SERIAL MONITOR *** FOR ESP8266 ***
//**********************//**********************//********************************
/*
void TimeDateDisplayOne() // digital clock display 24hr format // for ESP-8266 and HUM-an too :)
{
Serial.print("Z");
PrintZD(hour()); // Print Hour Digit and add a zero before single digits
PrintCD(minute()); // Print a colon sign ":" and add zero before single digits
PrintCD(second());
Serial.print('W');
Serial.print(weekday()); // there are no double digits only 7 days in a week
Serial.print('D');
PrintZD(day()); // Print the Day of the month and add zero before single digits
Serial.print('M'); // add zero before single digit month numbers
PrintZD(month());
Serial.print('Y');
Serial.print(year());
}
*/
//**********************//**********************//**********************//*****
//Function for digital clock display (in the SERIAL MONITOR *** FOR HUMAN ***
//**********************//**********************//**********************//*****
void TimeDateDisplayTwo() // digital clock display of the time 24hr format // for HUM-an
{
Serial.println("****>> 10 second timer<< ****");
Serial.println("TIME");
PrintZD(hour());
PrintCD(minute());
PrintCD(second());
Serial.println();
Serial.println("DATE");
Serial.print(dayStr(weekday()));
Serial.print("-");
Serial.print(day());
Serial.print("-");
Serial.print(monthStr(month()));
Serial.print("-");
Serial.println(year());
}
//***********************************************************************
//** Function for adding Zero before single digits before Serial print **
//***********************************************************************
void PrintZD(int D)
{
if(D < 10) // don't add brackets for this if statement it will not work
Serial.print('0');
Serial.print(D);
}
//*********************************************************************************
//** Function for adding a Colon & Zero before single digits before Serial print **
//*********************************************************************************
void PrintCD(int digits)
{
Serial.print(":");
if (digits < 10) // don't add brackets for this if statement it will not work
Serial.print('0');
Serial.print(digits);
}
//******************************
// THE END
//******************************
Do-it-yourself,DIY,How-To,Electronics,Arduino,Esp,Raspberry-Pi,Programming,Hacking Stuff,Gardening,Photography,Hobby,Fun
Arduino Grow Box Green House Controller 5.0 [OUT-DATED] Cooler,Heater,Lights,Humidifier,Fan,Water Level,Soil Humidity,Pump
Subscribe to:
Posts (Atom)