/**The MIT License (MIT) Copyright (c) 2017 by Daniel Eichhorn Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. See more at https://blog.squix.org */ #include #include #include "MiniGrafx.h" #include "Carousel.h" #include "ILI9341_SPI.h" #include "ArialRounded.h" #include "moonphases.h" #include "weathericons.h" #include #include "settings.h" #include #include #include #include #include #include "CalendarParser.h" #define TFT_DC D2 #define TFT_CS D1 #define TFT_LED D8 #define MINI_BLACK 0 #define MINI_WHITE 1 #define MINI_YELLOW 2 #define MINI_BLUE 3 #define MAX_FORECASTS 12 // defines the colors usable in the paletted 16 color frame buffer uint16_t palette[] = {ILI9341_BLACK, // 0 ILI9341_WHITE, // 1 ILI9341_YELLOW, // 2 0x7E3C/*ILI9341_BLUE*/}; //3 int SCREEN_WIDTH = 240; int SCREEN_HEIGHT = 320; int BITS_PER_PIXEL = 2; // 2^2 = 4 colors // HOSTNAME for OTA update #define HOSTNAME "ESP8266-OTA-" /***************************** * Important: see settings.h to configure your settings!!! * ***************************/ ILI9341_SPI tft = ILI9341_SPI(TFT_CS, TFT_DC); MiniGrafx gfx = MiniGrafx(&tft, BITS_PER_PIXEL, palette); Carousel carousel(&gfx, 0, 0, 240, 100); TimeClient timeClient(UTC_OFFSET); WGConditions conditions; WGForecast forecasts[MAX_FORECASTS]; WGAstronomy astronomy; void updateData(); void drawProgress(uint8_t percentage, String text); void drawTime(); void drawCurrentWeather(); void drawForecast(); void drawForecastDetail(uint16_t x, uint16_t y, uint8_t dayIndex); void drawAstronomy(); void drawSeparator(uint16_t y); const char* getMeteoconIconFromProgmem(String iconText); const char* getMiniMeteoconIconFromProgmem(String iconText); void drawForecast1(MiniGrafx *display, CarouselState* state, int16_t x, int16_t y); void drawForecast2(MiniGrafx *display, CarouselState* state, int16_t x, int16_t y); FrameCallback frames[] = { drawForecast1, drawForecast2 }; int frameCount = 2; long lastDownloadUpdate = millis(); void updateCalendar(); void setup() { Serial.begin(115200); // The LED pin needs to set HIGH // Use this pin to save energy // Turn on the background LED pinMode(TFT_LED, OUTPUT); digitalWrite(TFT_LED, HIGH); gfx.init(); gfx.fillBuffer(MINI_BLACK); gfx.setFont(ArialRoundedMTBold_14); gfx.setColor(MINI_YELLOW); gfx.setTextAlignment(TEXT_ALIGN_CENTER); gfx.drawString(120, 160, "Connecting to WiFi"); gfx.commit(); carousel.setFrames(frames, frameCount); //WiFiManager Serial.print("Free heap: "); Serial.println(ESP.getFreeHeap()); //Manual Wifi WiFi.begin("yourssid", "yourpassw0rd"); while (WiFi.status() != WL_CONNECTED) { delay(500); Serial.print("."); } Serial.println(ESP.getFreeHeap()); //calendar.updateCalendar(G_SCRIPT_ID); //updateCalendar(); // load the weather information updateData(); } long lastDrew = 0; void loop() { gfx.fillBuffer(MINI_BLACK); drawTime(); drawCurrentWeather(); //drawForecast(); int remainingTimeBudget = carousel.update(); if (remainingTimeBudget > 0) { // You can do some work here // Don't do stuff if you are below your // time budget. delay(remainingTimeBudget); } drawAstronomy(); gfx.commit(); // Check if we should update weather information if (millis() - lastDownloadUpdate > 1000 * UPDATE_INTERVAL_SECS) { updateData(); lastDownloadUpdate = millis(); } } // Update the internet based information and update screen void updateData() { Serial.print("Free heap, before bit change: "); Serial.println(ESP.getFreeHeap()); gfx.changeBitDepth(1, palette); Serial.print("Free heap, after bit change: "); Serial.println(ESP.getFreeHeap()); gfx.fillBuffer(MINI_BLACK); gfx.setFont(ArialRoundedMTBold_14); free(conditions); free(forecasts); free(astronomy); drawProgress(10, "Updating calendar..."); /*CalendarParser calendar = new Calendar(); calendar->updateCalendar(G_SCRIPT_ID); calendar = nullptr;*/ //updateCalendar(); drawProgress(20, "Updating time..."); timeClient.updateTime(); drawProgress(50, "Updating conditions..."); conditions = (WGConditions *) malloc(sizeof(WGConditions)); Serial.println("Allocating"); delay(1000); conditions->currentTemp = "hello"; Serial.println("Allocated"); delay(1000); WundergroundConditions *conditionsClient = new WundergroundConditions(IS_METRIC); conditionsClient->updateConditions(conditions, WUNDERGRROUND_API_KEY, WUNDERGRROUND_LANGUAGE, WUNDERGROUND_COUNTRY, WUNDERGROUND_CITY); delete conditionsClient; conditionsClient = nullptr; drawProgress(70, "Updating forecasts..."); forecasts = (WGForecast*) malloc(sizeof(WGForecast) * MAX_FORECASTS); WundergroundForecast *forecastClient = new WundergroundForecast(IS_METRIC); forecastClient->updateForecast(forecasts, MAX_FORECASTS, WUNDERGRROUND_API_KEY, WUNDERGRROUND_LANGUAGE, WUNDERGROUND_COUNTRY, WUNDERGROUND_CITY); delete forecastClient; forecastClient = nullptr; drawProgress(80, "Updating astronomy..."); astronomy = (WGAstronomy*) malloc(sizeof(WGAstronomy)); WundergroundAstronomy *astronomyClient = new WundergroundAstronomy(USE_PM); astronomyClient->updateAstronomy(astronomy, WUNDERGRROUND_API_KEY, WUNDERGRROUND_LANGUAGE, WUNDERGROUND_COUNTRY, WUNDERGROUND_CITY); delete astronomyClient; astronomyClient = nullptr; drawProgress(100, "Done..."); Serial.print("Free heap, before bit change: "); Serial.println(ESP.getFreeHeap()); gfx.changeBitDepth(2, palette); Serial.print("Free heap, after bit change: "); Serial.println(ESP.getFreeHeap()); delay(1000); } // Progress bar helper void drawProgress(uint8_t percentage, String text) { gfx.fillBuffer(MINI_BLACK); gfx.setFont(ArialRoundedMTBold_14); gfx.setTextAlignment(TEXT_ALIGN_CENTER); gfx.setColor(MINI_YELLOW); gfx.drawString(120, 146, text); gfx.setColor(MINI_WHITE); gfx.drawRect(10, 165, 240 - 20, 15); gfx.setColor(MINI_BLUE); gfx.fillRect(12, 167, 216 * percentage / 100, 11); //ui.drawProgressBar(10, 165, 240 - 20, 15, percentage, ILI9341_WHITE, ILI9341_BLUE); gfx.commit(); } // draws the clock void drawTime() { gfx.setTextAlignment(TEXT_ALIGN_CENTER); gfx.setColor(MINI_WHITE); gfx.setFont(ArialRoundedMTBold_14); String date = conditions->date; gfx.drawString(120, 6, date); gfx.setFont(ArialRoundedMTBold_36); String time = timeClient.getHours() + ":" + timeClient.getMinutes() + ":" + timeClient.getSeconds(); gfx.drawString(120, 20, time); } void updateCalendar() { Serial.print("Free heap: "); Serial.println(ESP.getFreeHeap()); const char* host = "script.google.com"; // Replace with your own script id to make server side changes const char *GScriptId = "AKfycbwdOi6zab7cLU5fEr0AL6KrAMpygUoFHOtSrgnKfccyHHkpZPo"; const int httpsPort = 443; String url = String("/macros/s/") + GScriptId + "/exec"; // echo | openssl s_client -connect script.google.com:443 |& openssl x509 -fingerprint -noout const char* fingerprint = "08:9E:B2:B8:77:37:3E:85:26:09:CA:29:13:D2:B0:57:26:DE:C4:6D"; // Use HTTPSRedirect class to create a new TLS connection HTTPSRedirect *client = new HTTPSRedirect(httpsPort); client->setPrintResponseBody(true); client->setContentTypeHeader("application/json"); Serial.print("Connecting to "); Serial.println(host); // Try to connect for a maximum of 5 times bool flag = false; for (int i=0; i<5; i++){ Serial.println("."); delay(1000); int retval = client->connect(host, httpsPort); if (retval == 1) { flag = true; break; } else Serial.println("Connection failed. Retrying..."); } if (!flag){ Serial.print("Could not connect to server: "); Serial.println(host); Serial.println("Exiting..."); return; } if (client->verify(fingerprint, host)) { Serial.println("Certificate match."); } else { Serial.println("Certificate mis-match"); } // fetch spreadsheet data client->GET(url, host); delete client; client = nullptr; } // draws current weather information void drawCurrentWeather() { // Weather Icon gfx.setTransparentColor(MINI_BLACK); //gfx.drawBmpFromFile(weatherIcon + ".bmp", 0, 55); gfx.drawPalettedBitmapFromPgm(0, 55, getMeteoconIconFromProgmem(conditions->weatherIcon)); // Weather Text gfx.setFont(ArialRoundedMTBold_14); gfx.setColor(MINI_YELLOW); gfx.setTextAlignment(TEXT_ALIGN_RIGHT); gfx.drawString(220, 76, conditions->weatherText); gfx.setFont(ArialRoundedMTBold_36); gfx.setColor(MINI_WHITE); gfx.setTextAlignment(TEXT_ALIGN_RIGHT); String degreeSign = "°F"; if (IS_METRIC) { degreeSign = "°C"; } String temp = conditions->currentTemp + degreeSign; gfx.drawString(220, 89, temp); } void drawForecast1(MiniGrafx *display, CarouselState* state, int16_t x, int16_t y) { drawForecastDetail(x + 10, y + 165, 0); drawForecastDetail(x + 95, y + 165, 2); drawForecastDetail(x + 180, y + 165, 4); } void drawForecast2(MiniGrafx *display, CarouselState* state, int16_t x, int16_t y) { drawForecastDetail(x + 10, y + 165, 6); drawForecastDetail(x + 95, y + 165, 8); drawForecastDetail(x + 180, y + 165, 10); } // draws the three forecast columns void drawForecast() { } // helper for the forecast columns void drawForecastDetail(uint16_t x, uint16_t y, uint8_t dayIndex) { gfx.setColor(MINI_YELLOW); gfx.setFont(ArialRoundedMTBold_14); gfx.setTextAlignment(TEXT_ALIGN_CENTER); String day = forecasts[dayIndex].forecastTitle.substring(0, 3); day.toUpperCase(); gfx.drawString(x + 25, y - 15, day); gfx.setColor(MINI_WHITE); gfx.drawString(x + 25, y, forecasts[dayIndex].forecastLowTemp + "|" + forecasts[dayIndex].forecastHighTemp); gfx.drawPalettedBitmapFromPgm(x, y + 15, getMiniMeteoconIconFromProgmem(forecasts[dayIndex].forecastIcon)); } // draw moonphase and sunrise/set and moonrise/set void drawAstronomy() { char moonAgeImage = 65 + 26 * astronomy->moonAge.toInt() / 30.0; gfx.setFont(MoonPhases_Regular_36); gfx.setColor(MINI_WHITE); gfx.setTextAlignment(TEXT_ALIGN_CENTER); gfx.drawString(120, 270, String(moonAgeImage)); //gfx.drawBmpFromFile("/moon" + String(moonAgeImage) + ".bmp", 120 - 30, 255); gfx.setColor(MINI_WHITE); gfx.setFont(ArialRoundedMTBold_14); gfx.setTextAlignment(TEXT_ALIGN_CENTER); gfx.setColor(MINI_YELLOW); gfx.drawString(120, 245, astronomy->moonPhase); gfx.setTextAlignment(TEXT_ALIGN_LEFT); gfx.setColor(MINI_YELLOW); gfx.drawString(10, 245, "Sun"); gfx.setColor(MINI_WHITE); astronomy->sunriseTime.trim(); astronomy->sunriseTime.trim(); gfx.drawString(10, 276, astronomy->sunriseTime); gfx.drawString(10, 291, astronomy->sunsetTime); gfx.setTextAlignment(TEXT_ALIGN_RIGHT); gfx.setColor(MINI_YELLOW); gfx.drawString(230, 245, "Moon"); gfx.setColor(MINI_WHITE); astronomy->moonriseTime.trim(); astronomy->moonsetTime.trim(); gfx.drawString(230, 276, astronomy->moonriseTime); gfx.drawString(230, 291, astronomy->moonsetTime); } // Helper function, should be part of the weather station library and should disappear soon const char* getMeteoconIconFromProgmem(String iconText) { if (iconText == "chanceflurries") return chanceflurries; if (iconText == "chancerain") return chancerain; if (iconText == "chancesleet") return chancesleet; if (iconText == "chancesnow") return chancesnow; if (iconText == "chancetstorms") return chancestorms; if (iconText == "cloudy") return clear; if (iconText == "cloudy") return cloudy; if (iconText == "flurries") return flurries; if (iconText == "fog") return fog; if (iconText == "hazy") return hazy; if (iconText == "mostlycloudy") return mostlycloudy; if (iconText == "mostlysunny") return mostlysunny; if (iconText == "partlycloudy") return partlycloudy; if (iconText == "partlysunny") return partlysunny; if (iconText == "sleet") return sleet; if (iconText == "rain") return rain; if (iconText == "snow") return snow; if (iconText == "sunny") return sunny; if (iconText == "tstorms") return tstorms; return unknown; } const char* getMiniMeteoconIconFromProgmem(String iconText) { if (iconText == "chanceflurries") return minichanceflurries; if (iconText == "chancerain") return minichancerain; if (iconText == "chancesleet") return minichancesleet; if (iconText == "chancesnow") return minichancesnow; if (iconText == "chancetstorms") return minichancestorms; if (iconText == "clear") return miniclear; if (iconText == "cloudy") return minicloudy; if (iconText == "flurries") return miniflurries; if (iconText == "fog") return minifog; if (iconText == "hazy") return minihazy; if (iconText == "mostlycloudy") return minimostlycloudy; if (iconText == "mostlysunny") return minimostlysunny; if (iconText == "partlycloudy") return minipartlycloudy; if (iconText == "partlysunny") return minipartlysunny; if (iconText == "sleet") return minisleet; if (iconText == "rain") return minirain; if (iconText == "snow") return minisnow; if (iconText == "sunny") return minisunny; if (iconText == "tstorms") return minitstorms; return miniunknown; } // if you want separators, uncomment the tft-line void drawSeparator(uint16_t y) { //tft.drawFastHLine(10, y, 240 - 2 * 10, 0x4228); }