main/task_tft.c

changeset 0
b74b0e4902c3
child 1
ad2c8b13eb88
--- /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);
+    }
+}
+
+

mercurial