diff -r 000000000000 -r b74b0e4902c3 main/task_tft.c --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/main/task_tft.c Sat Oct 20 13:23:15 2018 +0200 @@ -0,0 +1,795 @@ +/** + * @file task_tft.c + * @brief BrewBoard TFT and Touch screen driver. + * But because the application is controlled using the touch screen, + * all the processing of menus is also found here. + * It's the first started task, but it does nothing until the + * Main_Screen variable is set. + * + */ + +#include "config.h" + +spi_lobo_device_handle_t spi; +spi_lobo_device_handle_t tsspi = NULL; +extern sButton Buttons[MAXBUTTONS]; // 40 buttons on a screen. +time_t now; +struct tm timeinfo; +char s_timer[10]; +char s_top_msg[64]; + +extern float stageTemp; +extern uint16_t stageTime; +extern uint16_t TimeWhirlPool; +extern uint32_t TimeLeft; +extern uint32_t TimeSpent; +extern uint32_t SecsCount; +extern uint32_t pumpTime; +extern uint32_t TimeBrewing; +extern uint16_t Steady; +esp_timer_handle_t timerHandle; +extern bool _NewMinute; +extern bool _UseHLT; +extern bool System_TimeOk; + +static const char *TAG = "task_tft"; + +// Define which spi bus to use TFT_VSPI_HOST or TFT_HSPI_HOST +#define SPI_BUS TFT_HSPI_HOST + +extern int Main_Screen; +extern int Old_Screen; +extern int MLT_pin; +extern int HLT_pin; +extern int Pump_pin; +extern DS18B20_State *ds18b20_state; +extern DRIVER_State *driver_state; +extern JSON_log *json_log; +extern SemaphoreHandle_t xSemaphoreDS18B20; +extern SemaphoreHandle_t xSemaphoreDriver; +extern SemaphoreHandle_t xSemaphoreWiFi; +extern WIFI_State *wifi_state; +extern double Output; +extern sButton Buttons[MAXBUTTONS]; + +extern int BoilPower, LastMashStep; +extern char temp_buf[], logline[], strftime_buf[64]; +extern bool loop, CoolBeep, Resume, pumpRest, updateRuntime; +extern bool NewMinute, TempReached; +extern uint8_t MashState; +extern float temp_MLT, MinMash, MaxMash; +extern uint32_t power_MLT, power_HLT, counts; + + +#ifdef CONFIG_TEMP_SENSORS_SIMULATOR +extern float Fake_MLT; +extern float Fake_HLT; +#endif + + + +/** + * @brief Seconds timer callback. + */ +void TimerCallback(void *arg); + + +/***************************************************************************/ + + + +int init_tft_display(void) +{ + esp_err_t ret; + esp_timer_create_args_t timerSecond = { + .callback = &TimerCallback, + .name = "SecondsTimer" + }; + + ESP_LOGI(TAG, "Initialize TFT"); + + max_rdclock = 8000000; + TFT_PinsInit(); + + spi_lobo_bus_config_t buscfg = { + .miso_io_num=PIN_NUM_MISO, // set SPI MISO pin + .mosi_io_num=PIN_NUM_MOSI, // set SPI MOSI pin + .sclk_io_num=PIN_NUM_CLK, // set SPI CLK pin + .quadwp_io_num=-1, + .quadhd_io_num=-1, + .max_transfer_sz = 6*1024, + }; + spi_lobo_device_interface_config_t devcfg={ + .clock_speed_hz=8000000, // Initial clock out at 8 MHz + .mode=0, // SPI mode 0 + .spics_io_num=-1, // we will use external CS pin + .spics_ext_io_num=PIN_NUM_CS, // external CS pin + .flags=LB_SPI_DEVICE_HALFDUPLEX, // ALWAYS SET to HALF DUPLEX MODE!! for display spi + }; + spi_lobo_device_interface_config_t tsdevcfg={ + .clock_speed_hz=2500000, //Clock out at 2.5 MHz + .mode=0, //SPI mode 0 + .spics_io_num=PIN_NUM_TCS, //Touch CS pin + .spics_ext_io_num=-1, //Not using the external CS + }; + + ESP_LOGI(TAG, "TFT pins: miso=%d, mosi=%d, sck=%d, cs=%d", PIN_NUM_MISO, PIN_NUM_MOSI, PIN_NUM_CLK, PIN_NUM_CS); + + ret = spi_lobo_bus_add_device(SPI_BUS, &buscfg, &devcfg, &spi); + assert(ret == ESP_OK); + disp_spi = spi; + + // ==== Test select/deselect ==== + ret = spi_lobo_device_select(spi, 1); + assert(ret == ESP_OK); + ret = spi_lobo_device_deselect(spi); + assert(ret == ESP_OK); + + ESP_LOGI(TAG, "SPI: attached display, spi bus: %d, speed: %u, bus uses native pins: %s", + SPI_BUS, spi_lobo_get_speed(spi), spi_lobo_uses_native_pins(spi) ? "true" : "false"); + + ESP_LOGI(TAG, "TS pins : miso=%d, mosi=%d, sck=%d, cs=%d", PIN_NUM_MISO, PIN_NUM_MOSI, PIN_NUM_CLK, PIN_NUM_TCS); + + ret=spi_lobo_bus_add_device(SPI_BUS, &buscfg, &tsdevcfg, &tsspi); + assert(ret == ESP_OK); + ts_spi = tsspi; + + // ==== Test select/deselect ==== + ret = spi_lobo_device_select(tsspi, 1); + assert(ret == ESP_OK); + ret = spi_lobo_device_deselect(tsspi); + assert(ret == ESP_OK); + + ESP_LOGI(TAG, "SPI: attached TS device, spi bus: %d, speed: %u", SPI_BUS, spi_lobo_get_speed(tsspi)); + + // ==== Initialize the Display ==== + TFT_display_init(); + + // ---- Detect maximum read speed ---- + max_rdclock = find_rd_speed(); + + // ==== Set SPI clock used for display operations ==== + spi_lobo_set_speed(spi, DEFAULT_SPI_CLOCK); + ESP_LOGI(TAG, "SPI: Max rd speed: %u, changed speed to %u", max_rdclock, spi_lobo_get_speed(spi)); + + font_rotate = 0; + text_wrap = 0; + font_transparent = 0; + font_forceFixed = 0; + gray_scale = 0; + TFT_setGammaCurve(DEFAULT_GAMMA_CURVE); + TFT_setRotation(LANDSCAPE); + TFT_setFont(DEFAULT_FONT, NULL); + TFT_resetclipwin(); + + /* + * Create a one second periodic timer. + */ + ret = esp_timer_create(&timerSecond, &timerHandle); + assert(ret == ESP_OK); + ret = esp_timer_start_periodic(timerHandle, 1000000); + assert(ret == ESP_OK); + + return ret; +} + + + +void TimerCallback(void *arg) +{ + TimeSpent++; + SecsCount++; + Steady++; + TimeBrewing++; + runtime.TimeBrewing++; + if ((SecsCount % 60) == 0) + _NewMinute = true; + + if (TimeLeft) { + TimeLeft--; + if (TimeLeft == 5) { + SoundPlay(SOUND_TimeOut); + } + if ((TimeLeft % 60) == 0) { + pumpTime++; + } + } +} + + + +void TimerSet(uint32_t seconds) +{ + Steady = TimeSpent = SecsCount = 0; + TimeLeft = seconds; +} + + + +void TimerShow(uint32_t Time, int X, int Y) +{ + uint8_t Hours = (uint8_t)(Time / 3600); + uint8_t Minutes = (uint8_t)((Time % 3600) / 60); + uint8_t Seconds = (uint8_t)(Time % 60); + char msg[32]; + static uint32_t _oldTime = 0; + + if (Time != _oldTime) { + _fg = TFT_GREEN; + TFT_setFont(FONT_7SEG, NULL); + set_7seg_font_atrib(12, 2, 1, TFT_DARKGREY); + snprintf(s_timer, 9, "%02d:%02d:%02d", Hours, Minutes, Seconds); + TFT_print(s_timer, X, Y); + _oldTime = Time; + snprintf(msg, 31, "{\"timer\":\"%s\"}", s_timer); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } +} + + + +void TopMessage(char *text) +{ + char msg[64]; + + snprintf(s_top_msg, 63, "%s", text); + _fg = TFT_YELLOW; + font_transparent = 1; + TFT_setFont(DEJAVU24_FONT, NULL); + TFT_fillRect(0, 0, 319, 25, TFT_NAVY); + TFT_print(s_top_msg, CENTER, 2); + font_transparent = 0; + snprintf(msg, 63, "{\"top_msg\":\"%s\"}", s_top_msg); + ws_server_send_text_clients("/ws", msg, strlen(msg)); +} + + + +void MLT_info(int x, int y, bool update) +{ + char ctemp[16], csp[16], cpower[16], msg[32]; + static char ltemp[16], lsp[16], lpower[16]; + bool con, cpwr, cpump = false; + static bool lon, lpwr, lpump; + + _bg = (color_t){ 48, 48, 48 }; + _fg = TFT_WHITE; + color_t _led = { 31,255, 31}; + color_t _pump = {127,175,255}; + color_t _pwr = {255, 47, 47}; + + if (! update) { + TFT_fillRect(x, y, 178, 90, _bg); + TFT_drawRect(x, y, 178, 90, _fg); + TFT_drawFastHLine(x, y + 21, 178, _fg); + TFT_setFont(DEJAVU18_FONT, NULL); + TFT_print("MLT", x + 67, y + 3); + } + + if (xSemaphoreTake(xSemaphoreDriver, 10) == pdTRUE) { + sprintf(ctemp, "%7.3f", driver_state->mlt_pv); + if (driver_state->mlt_mode) { + sprintf(csp, "%6.2f sp", driver_state->mlt_sp); + } else { + csp[0] = '\0'; + } + if ((driver_state->mlt_mode == MLT_MODE_BANG) || (driver_state->mlt_mode == MLT_MODE_PID) || (driver_state->mlt_mode == MLT_MODE_EXT)) { + sprintf(cpower, "%3d%%", driver_state->mlt_power); + } else { + cpower[0] = '\0'; + } + xSemaphoreGive(xSemaphoreDriver); + } + + con = (MLT_pin) ? true : false; + if ((con != lon) || (! update)) { + if (con) { + TFT_fillCircle(x + 166, y + 11, 8, _led); + } else { + TFT_fillCircle(x + 166, y + 11, 8, _bg); + } + lon = con; + snprintf(msg, 31, "{\"mlt_led\":\"%s\"}", con ? "1":"0"); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + + cpump = (Pump_pin) ? true : false; + if ((cpump != lpump) || (! update)) { + if (cpump) { + TFT_fillCircle(x + 146, y + 11, 8, _pump); + } else { + TFT_fillCircle(x + 146, y + 11, 8, _bg); + } + lpump = cpump; + snprintf(msg, 31, "{\"pump_led\":\"%s\"}", cpump ? "1":"0"); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + + if (equipment.SSR2 == SSR2_ON_IDLE) { + cpwr = (HLT_pin) ? true : false; + if ((cpwr != lpwr) || (! update)) { + if (cpwr) { + TFT_fillCircle(x + 126, y + 11, 8, _pwr); + } else { + TFT_fillCircle(x + 126, y + 11, 8, _bg); + } + lpwr = cpwr; + snprintf(msg, 31, "{\"hlt_led\":\"%s\"}", cpwr ? "1":"0"); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + } + + if (strcmp(ctemp, ltemp) || (! update)) { + TFT_setFont(USER_FONT, "/spiffs/fonts/Grotesk24x48.fon"); + TFT_print(ctemp, x + 5, y + 23); + strncpy(ltemp, ctemp, 16); + snprintf(msg, 31, "{\"mlt_pv\":\"%s\"}", ctemp); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + + TFT_setFont(DEJAVU18_FONT, NULL); + if (strcmp(csp, lsp) || (! update)) { + TFT_clearStringRect(x + 5, y + 70, "123.45 sp"); + TFT_print(csp, x + 5, y + 70); + strncpy(lsp, csp, 16); + snprintf(msg, 31, "{\"mlt_sp\":\"%s\"}", csp); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + if (strcmp(cpower, lpower) || (! update)) { + TFT_clearStringRect(x + 120, y + 70, "100%"); + TFT_print(cpower, x + 120, y + 70); + strncpy(lpower, cpower, 16); + snprintf(msg, 31, "{\"mlt_power\":\"%s\"}", cpower); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } +} + + + +void HLT_info(int x, int y, bool update, bool small) +{ + char ctemp[16], csp[16], cpower[16], msg[32]; + static char ltemp[16], lsp[16], lpower[16]; + bool con = false; + static bool lon; + uint8_t H; + + _bg = (color_t){ 63, 63, 64 }; + _fg = TFT_YELLOW; + color_t _led = {255, 47, 47}; + H = (small) ? 70 : 90; + + if (! update) { + TFT_fillRect(x, y, 178, H, _bg); + TFT_drawRect(x, y, 178, H, _fg); + TFT_drawFastHLine(x, y + 21, 178, _fg); + TFT_setFont(DEJAVU18_FONT, NULL); + TFT_print("HLT", x + 67, y + 3); + } + + if (xSemaphoreTake(xSemaphoreDriver, 10) == pdTRUE) { + sprintf(ctemp, "%7.3f", driver_state->hlt_pv); + if (driver_state->hlt_mode == HLT_MODE_BANG) { + sprintf(cpower, "%3d%%", driver_state->hlt_power); + } else { + cpower[0] = '\0'; + } + if (driver_state->hlt_mode == HLT_MODE_BANG || driver_state->hlt_mode == HLT_MODE_OFF) { + sprintf(csp, "%6.2f sp", driver_state->hlt_sp); + } else { + csp[0] = '\0'; + } + xSemaphoreGive(xSemaphoreDriver); + } + + con = (HLT_pin) ? true : false; + if ((con != lon) || (! update)) { + if (con) { + TFT_fillCircle(x + 166, y + 11, 8, _led); + } else { + TFT_fillCircle(x + 166, y + 11, 8, _bg); + } + lon = con; + snprintf(msg, 31, "{\"hlt_led\":\"%s\"}", con ? "1":"0"); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + + if (strcmp(ltemp, ctemp) || (! update)) { + if (small) { + TFT_setFont(USER_FONT, "/spiffs/fonts/DejaVuSans24.fon"); + TFT_print(ctemp, x + 40, y + 25); + } else { + TFT_setFont(USER_FONT, "/spiffs/fonts/Grotesk24x48.fon"); + TFT_print(ctemp, x + 5, y + 23); + } + strncpy(ltemp, ctemp, 16); + snprintf(msg, 31, "{\"hlt_pv\":\"%s\"}", ctemp); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + + H = (small) ? 50 : 70; + TFT_setFont(DEJAVU18_FONT, NULL); + if (strcmp(csp, lsp) || (! update)) { + TFT_clearStringRect(x + 5, y + H, "123.45 sp"); + TFT_print(csp, x + 5, y + H); + strncpy(lsp, csp, 16); + snprintf(msg, 31, "{\"hlt_sp\":\"%s\"}", csp); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } + if (strcmp(cpower, lpower) || (! update)) { + TFT_clearStringRect(x + 120, y + H, "100%"); + TFT_print(cpower, x + 120, y + H); + strncpy(lpower, cpower, 16); + snprintf(msg, 31, "{\"hlt_power\":\"%s\"}", cpower); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + } +} + + + +void update_json(void) +{ + int Hour = (TimeBrewing / 3600); + int Minute = ((TimeBrewing % 3600) / 60); + + if (counts == 0) + counts = 1; // Prevent division by zero. + + if (xSemaphoreTake(xSemaphoreDriver, 10) == pdTRUE) { + snprintf(json_log->time, 8, "%02d:%02d", Hour, Minute); + json_log->mlt_sp = driver_state->mlt_sp; + json_log->mlt_pv = driver_state->mlt_pv; + json_log->mlt_power = power_MLT / counts; + json_log->mlt_tempreached = TempReached ? 1:0; + json_log->pump_run = driver_state->pump_run; + json_log->hlt_sp = driver_state->hlt_sp; + json_log->hlt_pv = driver_state->hlt_pv; + json_log->hlt_power = power_HLT / counts; + json_log->event[0] = '\0'; + xSemaphoreGive(xSemaphoreDriver); + } +} + + + +void TFTstartWS(int client) +{ + char msg[1024]; + char mlt_sp[16], mlt_power[16], hlt_sp[16], hlt_power[16]; + + if (xSemaphoreTake(xSemaphoreDriver, 10) == pdTRUE) { + if (driver_state->mlt_sp) { + snprintf(mlt_sp, 15, "%6.2f sp", driver_state->mlt_sp); + snprintf(mlt_power, 15, "%3d%%", driver_state->mlt_power); + } else { + mlt_sp[0] = '\0'; + mlt_power[0] = '\0'; + } + if (driver_state->hlt_sp) { + snprintf(hlt_sp, 15, "%6.2f sp", driver_state->hlt_sp); + snprintf(hlt_power, 15, "%3d%%", driver_state->hlt_power); + } else { + hlt_sp[0] = '\0'; + hlt_power[0] = '\0'; + } + snprintf(msg, 1023, "{\"main\":\"%d\",\"mlt_led\":\"%d\",\"mlt_pv\":\"%7.3f\",\"mlt_sp\":\"%s\",\"mlt_power\":\"%s\"" \ + ",\"pump_led\":\"%d\",\"hlt_led\":\"%d\",\"hlt_pv\":\"%7.3f\",\"hlt_sp\":\"%s\",\"hlt_power\":\"%s\"" \ + ",\"timer\":\"%s\",\"top_msg\":\"%s\"}", + Main_Screen, (MLT_pin) ? 1:0, driver_state->mlt_pv, mlt_sp, mlt_power, + (Pump_pin) ? 1:0, (HLT_pin) ? 1:0, driver_state->hlt_pv, hlt_sp, hlt_power, + s_timer, s_top_msg); + + xSemaphoreGive(xSemaphoreDriver); + ws_server_send_text_client(client, msg, strlen(msg)); + } + +} + + + +void task_tft(void *pvParameter) +{ + char msg[32]; + + ESP_LOGI(TAG, "Initialize TFT/Touch task"); + + /* + * Task loop. Read touchscreen events. + */ + while (1) { + /* + * Build new screen. + */ +startover: + + updateRuntime = false; + if (_NewMinute) { + _NewMinute = false; + NewMinute = true; + } + + /* + * Timekeeping. In the WiFi task sntp is started if there + * is a valid internet connection. + */ + time(&now); + localtime_r(&now, &timeinfo); + // Is time set? If not, tm_year will be (1970 - 1900). + if ((timeinfo.tm_year > (2016 - 1900)) && (! System_TimeOk)) { + System_TimeOk = true; + strftime(strftime_buf, sizeof(strftime_buf), "%c", &timeinfo); + ESP_LOGI(TAG, "System time is set: %s", strftime_buf); + } + + if (Old_Screen != Main_Screen) { + + if ((Main_Screen == MAIN_MODE_FREE) && ((config.ts_xleft == 0) || (config.ts_ybottom == 0))) { + Main_Screen = MAIN_MODE_CALIBRATION; + } + + /* + * With each screenchange, remove the timer too. + */ + snprintf(msg, 31, "{\"main\":\"%d\",\"timer\":\"\"}", Main_Screen); + ws_server_send_text_clients("/ws", msg, strlen(msg)); + + ESP_LOGI(TAG, "Change screen %d to %d", Old_Screen, Main_Screen); + _bg = TFT_BLACK; + TFT_fillScreen(_bg); + TFT_resetclipwin(); + Buttons_Clear(); + Old_Screen = Main_Screen; + + switch (Main_Screen) { + case MAIN_MODE_FREE: + TopMessage("Hoofdmenu"); + MLT_info(71, 26, false); + if ((equipment.SSR2 == SSR2_HLT_SHARE) || (equipment.SSR2 == SSR2_HLT_IND)) { + HLT_info(71,150, false, false); + } + Buttons_Add( 5, 26, 60, 40, "Hand", 0); + Buttons_Add(255, 26, 60, 40, "Auto", 1); + Buttons_Add( 5, 200, 60, 40, "Info", 2); + Buttons_Add(255, 200, 60, 40, "Tools", 3); + Buttons_Show(); + break; + + case MAIN_MODE_CALIBRATION: + Calibration_Init(); + break; + + case MAIN_INFO: + sprintf(temp_buf, "BrewBoard %s", VERSION); + TopMessage(temp_buf); + _fg = TFT_YELLOW; + TFT_setFont(UBUNTU16_FONT, NULL); + TFT_print("Written by Michiel Broek (C) 2018\r\n\n", 0, 50); + // ------------------------------------- + _fg = TFT_ORANGE; + TFT_print("Parts are written by Chris Morgan,\r\n", 0, LASTY); + TFT_print("Brett Beauregard, Chris Garry, LoBo,\r\n", 0, LASTY); + TFT_print("and David Antliff.\r\n", 0, LASTY); + ShowInteger(1,140, "Free memory", " bytes", esp_get_free_heap_size()); + ShowText(1,158, "IDF version", (char *)esp_get_idf_version()); + Buttons_Add(130, 200, 60, 40, "Ok", 0); + Buttons[0].dark = true; + Buttons_Show(); + break; + + case MAIN_TOOLS: + TopMessage("Tools menu"); + Buttons_Add( 20, 40,120, 40, "Setup", 0); + Buttons_Add( 20,120,120, 40, "Bestanden", 1); + Buttons_Add(180, 40,120, 40, "Recepten", 2); + Buttons_Add(180,120,120, 40, "Updates", 3); + Buttons_Add(130, 200, 60, 40, "Ok", 4); + Buttons[4].dark = true; + Buttons_Show(); + break; + + case MAIN_TOOLS_SETUP: + case MAIN_TOOLS_SETUP_CONFIG: + case MAIN_TOOLS_SETUP_CO_EDIT: + case MAIN_TOOLS_SETUP_EQUIPMENT: + case MAIN_TOOLS_SETUP_EQ_EDIT: + case MAIN_TOOLS_SETUP_CALIBRATION: + Setup_Init(); + break; + + case MAIN_TOOLS_SETUP_WIFI: + case MAIN_TOOLS_SETUP_WIFI_CUR: + case MAIN_TOOLS_SETUP_WIFI_CON: + case MAIN_TOOLS_SETUP_WIFI_NEW: + if (WiFi_Init()) + goto startover; + break; + + case MAIN_TOOLS_RECIPE: + case MAIN_TOOLS_RECIPE_EDIT: + Recipes_Init(); + break; + + case MAIN_TOOLS_FILES: + Files_Init(); + break; + + case MAIN_TOOLS_UPDATES: + Updates_Init(); + break; + + case MAIN_AUTO_INIT: + case MAIN_AUTO_DELAYSTART: + case MAIN_AUTO_HEATUP: + case MAIN_AUTO_MASH_IN: + case MAIN_AUTO_MASH_1: + case MAIN_AUTO_MASH_2: + case MAIN_AUTO_MASH_3: + case MAIN_AUTO_MASH_4: + case MAIN_AUTO_MASH_5: + case MAIN_AUTO_MASH_6: + case MAIN_AUTO_MASH_OUT: + case MAIN_AUTO_TOBOIL: + case MAIN_AUTO_BOILING: + case MAIN_AUTO_COOLING_H: + case MAIN_AUTO_COOLING_M: + case MAIN_AUTO_COOLING_C: + case MAIN_AUTO_WHIRLPOOL9: + case MAIN_AUTO_WHIRLPOOL7: + case MAIN_AUTO_WHIRLPOOL6: + case MAIN_AUTO_WHIRLPOOL2: + case MAIN_AUTO_DONE: + case MAIN_AUTO_ABORT: + if (Automation_Init()) + goto startover; + break; + + case MAIN_MANUAL_INIT: + case MAIN_MANUAL_MAIN: + if (Manual_Init()) + goto startover; + break; + + default: + break; + } + } + + /* + * Update screen + */ + switch (Main_Screen) { + case MAIN_MODE_FREE: + MLT_info(71, 26, true); + if ((equipment.SSR2 == SSR2_HLT_SHARE) || (equipment.SSR2 == SSR2_HLT_IND)) { + HLT_info(71, 150, true, false); + } + switch (Buttons_Scan()) { + case 0: Main_Screen = MAIN_MANUAL_INIT; break; + case 1: Main_Screen = MAIN_AUTO_INIT; break; + case 2: Main_Screen = MAIN_INFO; break; + case 3: Main_Screen = MAIN_TOOLS; break; + default: break; + } + break; + + case MAIN_MODE_CALIBRATION: + Calibration_Loop(); + Main_Screen = MAIN_MODE_FREE; + break; + + case MAIN_TOOLS: + switch (Buttons_Scan()) { + case 0: Main_Screen = MAIN_TOOLS_SETUP; break; + case 1: Main_Screen = MAIN_TOOLS_FILES; break; + case 2: Main_Screen = MAIN_TOOLS_RECIPE; break; + case 3: Main_Screen = MAIN_TOOLS_UPDATES; break; + case 4: Main_Screen = MAIN_MODE_FREE; break; + default: break; + } + break; + + case MAIN_TOOLS_SETUP: + case MAIN_TOOLS_SETUP_CONFIG: + case MAIN_TOOLS_SETUP_CO_EDIT: + case MAIN_TOOLS_SETUP_EQUIPMENT: + case MAIN_TOOLS_SETUP_EQ_EDIT: + case MAIN_TOOLS_SETUP_CALIBRATION: + Setup_Loop(); + break; + + case MAIN_TOOLS_SETUP_WIFI: + case MAIN_TOOLS_SETUP_WIFI_CUR: + case MAIN_TOOLS_SETUP_WIFI_CON: + case MAIN_TOOLS_SETUP_WIFI_NEW: + if (WiFi_Loop()) + goto startover; + break; + + case MAIN_TOOLS_RECIPE: + case MAIN_TOOLS_RECIPE_EDIT: + Recipes_Loop(); + break; + + case MAIN_TOOLS_FILES: + Files_Loop(); + break; + + case MAIN_TOOLS_UPDATES: + Updates_Loop(); + break; + + case MAIN_INFO: + if (Buttons_Scan() == 0) { + Main_Screen = MAIN_MODE_FREE; + } + break; + + case MAIN_AUTO_INIT: + case MAIN_AUTO_DELAYSTART: + case MAIN_AUTO_HEATUP: + case MAIN_AUTO_MASH_IN: + case MAIN_AUTO_MASH_1: + case MAIN_AUTO_MASH_2: + case MAIN_AUTO_MASH_3: + case MAIN_AUTO_MASH_4: + case MAIN_AUTO_MASH_5: + case MAIN_AUTO_MASH_6: + case MAIN_AUTO_MASH_OUT: + case MAIN_AUTO_TOBOIL: + case MAIN_AUTO_BOILING: + case MAIN_AUTO_COOLING_H: + case MAIN_AUTO_COOLING_M: + case MAIN_AUTO_COOLING_C: + case MAIN_AUTO_WHIRLPOOL9: + case MAIN_AUTO_WHIRLPOOL7: + case MAIN_AUTO_WHIRLPOOL6: + case MAIN_AUTO_WHIRLPOOL2: + case MAIN_AUTO_DONE: + case MAIN_AUTO_ABORT: + if (Automation_Loop()) + goto startover; + break; + + case MAIN_MANUAL_INIT: + case MAIN_MANUAL_MAIN: + if (Manual_Loop()) + goto startover; + break; + + default: + break; + } + + if (updateRuntime) { + write_runtime(); + } + + /* + * Count power average during brewing. + */ + if ((Main_Screen >= MAIN_AUTO_MASH_IN) && (Main_Screen < MAIN_AUTO_DONE)) { + if (xSemaphoreTake(xSemaphoreDriver, 10) == pdTRUE) { + power_MLT += driver_state->mlt_power; + power_HLT += driver_state->hlt_power; + counts++; + xSemaphoreGive(xSemaphoreDriver); + } + } + + if (NewMinute) { + /* + * Brew logging. + */ + if ((Main_Screen >= MAIN_AUTO_MASH_IN) && (Main_Screen < MAIN_AUTO_DONE)) { + update_json(); + log_json(); + power_MLT = power_HLT = counts = 0; + } + } + + NewMinute = false; + vTaskDelay(50 / portTICK_PERIOD_MS); + } +} + +