/* * Copyright (c) 2018-2019 CTCaer * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * This program is distributed in the hope it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include "gui.h" #include "gui_emummc_tools.h" #include "gui_tools.h" #include "gui_info.h" #include "../libs/lvgl/lv_themes/lv_theme_hekate.h" #include "../libs/lvgl/lvgl.h" #include "../gfx/logos-gui.h" #include "../config/config.h" #include "../config/ini.h" #include "../gfx/di.h" #include "../gfx/gfx.h" #include "../input/touch.h" #include "../libs/fatfs/ff.h" #include "../mem/minerva.h" #include "../power/bq24193.h" #include "../power/max17050.h" #include "../rtc/max77620-rtc.h" #include "../soc/bpmp.h" #include "../soc/hw_init.h" #include "../soc/t210.h" #include "../thermal/tmp451.h" #include "../utils/dirlist.h" #include "../utils/sprintf.h" #include "../utils/types.h" #include "../utils/util.h" extern hekate_config h_cfg; extern volatile nyx_storage_t *nyx_str; extern volatile boot_cfg_t *b_cfg; extern lv_res_t launch_payload(lv_obj_t *list); extern bool get_sd_card_removed(); extern bool sd_mount(); extern void sd_unmount(bool deinit); void *sd_file_read(const char *path, u32 *fsize); int sd_save_to_file(void *buf, u32 size, const char *filename); static bool disp_init_done = false; typedef struct _gui_status_bar_ctx { lv_obj_t *mid; lv_obj_t *time_temp; lv_obj_t *temp_symbol; lv_obj_t *temp_degrees; lv_obj_t *battery; lv_obj_t *battery_more; } gui_status_bar_ctx; static gui_status_bar_ctx status_bar; static void _nyx_disp_init() { display_backlight_brightness(0, 1000); display_init_framebuffer(); display_backlight_brightness(h_cfg.backlight - 20, 1000); } static void _save_fb_to_bmp() { const u32 file_size = 0x384000 + 0x36; u8 *bitmap = malloc(file_size); u32 *fb = malloc(0x384000); u32 *fb_ptr = gfx_ctxt.fb; // Reconstruct FB for bottom-top, landscape bmp. for (u32 x = 0; x < 1280; x++) { for (int y = 719; y > -1; y--) fb[y * 1280 + x] = *fb_ptr++; } // Create notification box. lv_obj_t * mbox = lv_mbox_create(lv_layer_top(), NULL); lv_mbox_set_recolor_text(mbox, true); lv_mbox_set_text(mbox, SYMBOL_CAMERA" #FFDD00 Saving screenshot...#"); lv_obj_set_width(mbox, LV_DPI * 4); lv_obj_set_top(mbox, true); lv_obj_align(mbox, NULL, LV_ALIGN_IN_TOP_LEFT, 0, 0); manual_system_maintenance(true); memcpy(bitmap + 0x36, fb, 0x384000); typedef struct _bmp_t { u16 magic; u32 size; u32 rsvd; u32 data_off; u32 hdr_size; u32 width; u32 height; u16 planes; u16 pxl_bits; u32 comp; u32 img_size; u32 res_h; u32 res_v; u64 rsvd2; } __attribute__((packed)) bmp_t; bmp_t *bmp = (bmp_t *)bitmap; bmp->magic = 0x4D42; bmp->size = file_size; bmp->rsvd = 0; bmp->data_off = 0x36; bmp->hdr_size = 40; bmp->width = 1280; bmp->height = 720; bmp->planes = 1; bmp->pxl_bits = 32; bmp->comp = 0; bmp->img_size = 0x384000; bmp->res_h = 2834; bmp->res_v = 2834; bmp->rsvd2 = 0; sd_mount(); char path[0x80]; strcpy(path, "bootloader/screenshots"); f_mkdir(path); s_printf(path + strlen(path), "/screen_%08X.bmp", get_tmr_us()); sd_save_to_file(bitmap, file_size, path); sd_unmount(false); free(bitmap); free(fb); lv_mbox_set_text(mbox, SYMBOL_CAMERA" #96FF00 Screenshot saved!#"); manual_system_maintenance(true); lv_mbox_start_auto_close(mbox, 4000); } static void _disp_fb_flush(int32_t x1, int32_t y1, int32_t x2, int32_t y2, const lv_color_t *color_p) { // Draw to framebuffer. gfx_set_rect_land_pitch(gfx_ctxt.fb, (u32 *)color_p, x1, y1, x2, y2); //pitch // Check if display init was done. If it's the first big draw, init. if (!disp_init_done && ((x2 - x1 + 1) > 600)) { disp_init_done = true; _nyx_disp_init(); } lv_flush_ready(); } touch_event touchpad; static bool _fts_touch_read(lv_indev_data_t *data) { touch_poll(&touchpad); if (touchpad.fingers > 2) _save_fb_to_bmp(); switch (touchpad.type & STMFTS_MASK_EVENT_ID) { case STMFTS_EV_MULTI_TOUCH_ENTER: case STMFTS_EV_MULTI_TOUCH_MOTION: data->state = LV_INDEV_STATE_PR; data->point.x = touchpad.x; data->point.y = touchpad.y; break; case STMFTS_EV_MULTI_TOUCH_LEAVE: data->state = LV_INDEV_STATE_REL; data->point.x = touchpad.x; data->point.y = touchpad.y; break; case STMFTS_EV_NO_EVENT: default: if (touchpad.touch) { data->state = LV_INDEV_STATE_PR; } else { data->point.x = touchpad.x; data->point.y = touchpad.y; data->state = LV_INDEV_STATE_REL; //return true; } break; } return false; // No buffering so no more data read. } typedef struct _system_maintenance_tasks_t { union { lv_task_t *tasks[2]; struct { lv_task_t *status_bar; lv_task_t *dram_periodic_comp; } task; }; } system_maintenance_tasks_t; static system_maintenance_tasks_t system_tasks; void manual_system_maintenance(bool refresh) { for (u32 task_idx = 0; task_idx < (sizeof(system_maintenance_tasks_t) / sizeof(lv_task_t *)); task_idx++) { lv_task_t *task = system_tasks.tasks[task_idx]; if(task && (lv_tick_elaps(task->last_run) >= task->period)) { task->last_run = lv_tick_get(); task->task(task->param); } } if (refresh) lv_refr_now(); } lv_img_dsc_t *bmp_to_lvimg_obj(const char *path) { u8 *bitmap = sd_file_read(path, NULL); if (!bitmap) return NULL; struct _bmp_data { u32 size; u32 size_x; u32 size_y; u32 offset; }; struct _bmp_data bmpData; // Get values manually to avoid unaligned access. bmpData.size = bitmap[2] | bitmap[3] << 8 | bitmap[4] << 16 | bitmap[5] << 24; bmpData.offset = bitmap[10] | bitmap[11] << 8 | bitmap[12] << 16 | bitmap[13] << 24; bmpData.size_x = bitmap[18] | bitmap[19] << 8 | bitmap[20] << 16 | bitmap[21] << 24; bmpData.size_y = bitmap[22] | bitmap[23] << 8 | bitmap[24] << 16 | bitmap[25] << 24; // Sanity check. if (bitmap[0] == 'B' && bitmap[1] == 'M' && bitmap[28] == 32) // Only 32 bit BMPs allowed. { // Check if non-default Bottom-Top. bool flipped = false; if (bmpData.size_y & 0x80000000) { bmpData.size_y = ~(bmpData.size_y) + 1; flipped = true; } lv_img_dsc_t *img_desc = (lv_img_dsc_t *)bitmap; u32 offset_copy = ALIGN((u32)bitmap + sizeof(lv_img_dsc_t), 0x10); img_desc->header.always_zero = 0; img_desc->header.w = bmpData.size_x; img_desc->header.h = bmpData.size_y; img_desc->header.cf = (bitmap[28] == 32) ? LV_IMG_CF_TRUE_COLOR_ALPHA : LV_IMG_CF_TRUE_COLOR; img_desc->data_size = bmpData.size - bmpData.offset; img_desc->data = (u8 *)offset_copy; u32 *tmp = malloc(bmpData.size); u32 *tmp2 = (u32 *)offset_copy; // Copy the unaligned data to an aligned buffer. memcpy((u8 *)tmp, bitmap + bmpData.offset, img_desc->data_size); u32 j = 0; if (!flipped) { for (u32 y = 0; y < bmpData.size_y; y++) { for (u32 x = 0; x < bmpData.size_x; x++) tmp2[j++] = tmp[(bmpData.size_y - 1 - y ) * bmpData.size_x + x]; } } else { for (u32 y = 0; y < bmpData.size_y; y++) { for (u32 x = 0; x < bmpData.size_x; x++) tmp2[j++] = tmp[y * bmpData.size_x + x]; } } free(tmp); } else { free(bitmap); return NULL; } return (lv_img_dsc_t *)bitmap; } #pragma GCC push_options #pragma GCC target ("thumb") lv_res_t nyx_generic_onoff_toggle(lv_obj_t *btn) { lv_obj_t *label_btn = lv_obj_get_child(btn, NULL); lv_obj_t *label_btn2 = lv_obj_get_child(btn, label_btn); char label_text[64]; if (!label_btn2) { strcpy(label_text, lv_label_get_text(label_btn)); label_text[strlen(label_text) - 15] = 0; if (!(lv_btn_get_state(btn) & LV_BTN_STATE_TGL_REL)) { strcat(label_text, "#D0D0D0 OFF#"); lv_label_set_array_text(label_btn, label_text, 64); } else { strcat(label_text, "#00FFC9 ON #"); lv_label_set_array_text(label_btn, label_text, 64); } } else { if (!(lv_btn_get_state(btn) & LV_BTN_STATE_TGL_REL)) lv_label_set_static_text(label_btn, "#D0D0D0 OFF#"); else lv_label_set_static_text(label_btn, "#00FFC9 ON #"); } return LV_RES_OK; } static lv_res_t auto_hos_poweroff_toggle(lv_obj_t *btn) { h_cfg.autohosoff = !h_cfg.autohosoff; if (!h_cfg.autohosoff) lv_btn_set_state(btn, LV_BTN_STATE_REL); else lv_btn_set_state(btn, LV_BTN_STATE_TGL_REL); nyx_generic_onoff_toggle(btn); return LV_RES_OK; } static lv_res_t auto_nogc_toggle(lv_obj_t *btn) { h_cfg.autonogc = !h_cfg.autonogc; if (!h_cfg.autonogc) lv_btn_set_state(btn, LV_BTN_STATE_REL); else lv_btn_set_state(btn, LV_BTN_STATE_TGL_REL); nyx_generic_onoff_toggle(btn); return LV_RES_OK; } lv_res_t mbox_action(lv_obj_t *btns, const char *txt) { lv_obj_t *mbox = lv_mbox_get_from_btn(btns); lv_obj_t *dark_bg = lv_obj_get_parent(mbox); lv_obj_del(dark_bg); // Deletes children also (mbox). return LV_RES_INV; } void nyx_window_toggle_buttons(lv_obj_t *win, bool disable) { lv_win_ext_t * ext = lv_obj_get_ext_attr(win); lv_obj_t * hbtn; hbtn = lv_obj_get_child_back(ext->header, NULL); hbtn = lv_obj_get_child_back(ext->header, hbtn); // Skip the title. if (disable) { while (hbtn != NULL) { lv_obj_set_opa_scale(hbtn, LV_OPA_40); lv_obj_set_opa_scale_enable(hbtn, true); lv_obj_set_click(hbtn, false); hbtn = lv_obj_get_child_back(ext->header, hbtn); } } else { while (hbtn != NULL) { lv_obj_set_opa_scale(hbtn, LV_OPA_COVER); lv_obj_set_click(hbtn, true); hbtn = lv_obj_get_child_back(ext->header, hbtn); } } } lv_obj_t *nyx_create_standard_window(const char *win_title) { static lv_style_t win_bg_style; lv_style_copy(&win_bg_style, &lv_style_plain); win_bg_style.body.main_color = LV_COLOR_HEX(0x2D2D2D);// TODO: COLOR_HOS_BG win_bg_style.body.grad_color = win_bg_style.body.main_color; lv_obj_t *win = lv_win_create(lv_scr_act(), NULL); lv_win_set_title(win, win_title); lv_win_set_style(win, LV_WIN_STYLE_BG, &win_bg_style); lv_obj_set_size(win, LV_HOR_RES, LV_VER_RES); lv_win_add_btn(win, NULL, SYMBOL_CLOSE" Close", lv_win_close_action); return win; } static void _launch_hos(u8 autoboot, u8 autoboot_list) { b_cfg->boot_cfg = BOOT_CFG_AUTOBOOT_EN; b_cfg->autoboot = autoboot; b_cfg->autoboot_list = autoboot_list; void (*main_ptr)() = (void *)nyx_str->hekate; sd_unmount(true); reconfig_hw_workaround(false, 0); (*main_ptr)(); } static lv_res_t _slider_brightness_action(lv_obj_t * slider) { display_backlight_brightness(lv_slider_get_value(slider) - 20, 0); h_cfg.backlight = lv_slider_get_value(slider); return LV_RES_OK; } struct autoboot_obj { lv_obj_t *obj0; lv_obj_t *obj1; lv_obj_t *obj2; }; static struct autoboot_obj autoboot_obj; static bool autoboot_first_time = true; static lv_res_t _win_autoboot_close_action(lv_obj_t * btn) { if (!h_cfg.autoboot) { lv_obj_set_opa_scale_enable(autoboot_obj.obj1, true); lv_obj_set_click(autoboot_obj.obj2, false); lv_btn_set_state(autoboot_obj.obj0, LV_BTN_STATE_REL); } else { lv_obj_set_opa_scale_enable(autoboot_obj.obj1, false); lv_obj_set_click(autoboot_obj.obj2, true); lv_btn_set_state(autoboot_obj.obj0, LV_BTN_STATE_TGL_REL); } nyx_generic_onoff_toggle(autoboot_obj.obj0); lv_obj_t * win = lv_win_get_from_btn(btn); lv_obj_del(win); return LV_RES_INV; } lv_obj_t *create_window_autoboot(const char *win_title) { static lv_style_t win_bg_style; lv_style_copy(&win_bg_style, &lv_style_plain); win_bg_style.body.main_color = LV_COLOR_HEX(0x2D2D2D);// TODO: COLOR_HOS_BG win_bg_style.body.grad_color = win_bg_style.body.main_color; lv_obj_t *win = lv_win_create(lv_scr_act(), NULL); lv_win_set_title(win, win_title); lv_win_set_style(win, LV_WIN_STYLE_BG, &win_bg_style); lv_obj_set_size(win, LV_HOR_RES, LV_VER_RES); lv_win_add_btn(win, NULL, SYMBOL_CLOSE" Close", _win_autoboot_close_action); return win; } // TODO: instant update of button for these. static lv_res_t _autoboot_disable_action(lv_obj_t *btn) { h_cfg.autoboot = 0; h_cfg.autoboot_list = 0; lv_obj_set_opa_scale_enable(autoboot_obj.obj1, true); lv_obj_set_click(autoboot_obj.obj2, false); lv_btn_set_state(autoboot_obj.obj0, LV_BTN_STATE_REL); nyx_generic_onoff_toggle(autoboot_obj.obj0); lv_obj_t * win = lv_win_get_from_btn(btn); lv_obj_del(win); return LV_RES_OK; } lv_obj_t *auto_main_list; lv_obj_t *auto_more_list; static lv_res_t _autoboot_enable_main_action(lv_obj_t *btn) { h_cfg.autoboot = lv_list_get_btn_index(auto_main_list, btn) + 1; h_cfg.autoboot_list = 0; lv_obj_set_opa_scale_enable(autoboot_obj.obj1, false); lv_obj_set_click(autoboot_obj.obj2, true); lv_btn_set_state(autoboot_obj.obj0, LV_BTN_STATE_TGL_REL); nyx_generic_onoff_toggle(autoboot_obj.obj0); lv_obj_t *obj = lv_obj_get_parent(btn); for (int i = 0; i < 5; i++) obj = lv_obj_get_parent(obj); lv_obj_del(obj); return LV_RES_INV; } static lv_res_t _autoboot_enable_more_action(lv_obj_t *btn) { h_cfg.autoboot = lv_list_get_btn_index(auto_more_list, btn) + 1; h_cfg.autoboot_list = 1; lv_obj_set_opa_scale_enable(autoboot_obj.obj1, false); lv_obj_set_click(autoboot_obj.obj2, true); lv_btn_set_state(autoboot_obj.obj0, LV_BTN_STATE_TGL_REL); nyx_generic_onoff_toggle(autoboot_obj.obj0); lv_obj_t *obj = lv_obj_get_parent(btn); for (int i = 0; i < 5; i++) obj = lv_obj_get_parent(obj); lv_obj_del(obj); return LV_RES_INV; } static void _create_autoboot_window() { lv_obj_t *win = create_window_autoboot(SYMBOL_GPS" Auto Boot"); lv_win_add_btn(win, NULL, SYMBOL_POWER" Disable", _autoboot_disable_action); static lv_style_t h_style; lv_style_copy(&h_style, &lv_style_transp); h_style.body.padding.inner = 0; h_style.body.padding.hor = LV_DPI - (LV_DPI / 4); h_style.body.padding.ver = LV_DPI / 6; // Main configurations container. lv_obj_t *h1 = lv_cont_create(win, NULL); lv_cont_set_style(h1, &h_style); lv_cont_set_fit(h1, false, true); lv_obj_set_width(h1, (LV_HOR_RES / 9) * 4); lv_obj_set_click(h1, false); lv_cont_set_layout(h1, LV_LAYOUT_OFF); lv_obj_t *label_sep = lv_label_create(h1, NULL); lv_label_set_static_text(label_sep, ""); lv_obj_t *label_txt = lv_label_create(h1, NULL); lv_label_set_static_text(label_txt, "Main configurations"); lv_obj_set_style(label_txt, lv_theme_get_current()->label.prim); lv_obj_align(label_txt, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -(LV_DPI / 4)); lv_obj_t *line_sep = lv_line_create(h1, NULL); static const lv_point_t line_pp[] = { {0, 0}, { LV_HOR_RES - (LV_DPI - (LV_DPI / 4)) * 2, 0} }; lv_line_set_points(line_sep, line_pp, 2); lv_line_set_style(line_sep, lv_theme_get_current()->line.decor); lv_obj_align(line_sep, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 8); // Create list and populate it with Main boot entries. lv_obj_t *list_main = lv_list_create(h1, NULL); auto_main_list = list_main; lv_obj_align(list_main, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4); lv_obj_set_size(list_main, LV_HOR_RES * 4 / 10, LV_VER_RES * 4 / 7); lv_list_set_single_mode(list_main, true); sd_mount(); LIST_INIT(ini_sections); if (ini_parse(&ini_sections, "bootloader/hekate_ipl.ini", false)) { LIST_FOREACH_ENTRY(ini_sec_t, ini_sec, &ini_sections, link) { if (!strcmp(ini_sec->name, "config") || (ini_sec->type != INI_CHOICE)) continue; lv_list_add(list_main, NULL, ini_sec->name, _autoboot_enable_main_action); } } // More configuration container. lv_obj_t *h2 = lv_cont_create(win, NULL); lv_cont_set_style(h2, &h_style); lv_cont_set_fit(h2, false, true); lv_obj_set_width(h2, (LV_HOR_RES / 9) * 4); lv_obj_set_click(h2, false); lv_cont_set_layout(h2, LV_LAYOUT_OFF); lv_obj_align(h2, h1, LV_ALIGN_OUT_RIGHT_TOP, LV_DPI * 17 / 29, 0); label_sep = lv_label_create(h2, NULL); lv_label_set_static_text(label_sep, ""); lv_obj_t *label_txt3 = lv_label_create(h2, NULL); lv_label_set_static_text(label_txt3, "Ini folder configurations"); lv_obj_set_style(label_txt3, lv_theme_get_current()->label.prim); lv_obj_align(label_txt3, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -LV_DPI / 11); line_sep = lv_line_create(h2, line_sep); lv_obj_align(line_sep, label_txt3, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 2), LV_DPI / 8); lv_line_set_style(line_sep, lv_theme_get_current()->line.decor); // Create list and populate it with more cfg boot entries. lv_obj_t *list_more_cfg = lv_list_create(h2, NULL); auto_more_list = list_more_cfg; lv_obj_align(list_more_cfg, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 2, LV_DPI / 4); lv_obj_set_size(list_more_cfg, LV_HOR_RES * 4 / 10, LV_VER_RES * 4 / 7); lv_list_set_single_mode(list_more_cfg, true); LIST_INIT(ini_list_sections); if (ini_parse(&ini_list_sections, "bootloader/ini", true)) { LIST_FOREACH_ENTRY(ini_sec_t, ini_sec, &ini_sections, link) { if (!strcmp(ini_sec->name, "config") || (ini_sec->type != INI_CHOICE)) continue; lv_list_add(list_more_cfg, NULL, ini_sec->name, _autoboot_enable_more_action); } } sd_unmount(false); } static lv_res_t _autoboot_hide_delay_action(lv_obj_t *btn) { lv_obj_set_opa_scale(autoboot_obj.obj1, LV_OPA_40); if (!autoboot_first_time) _create_autoboot_window(); if (!h_cfg.autoboot && autoboot_first_time) { lv_obj_set_opa_scale_enable(autoboot_obj.obj1, true); lv_obj_set_click(autoboot_obj.obj2, false); lv_btn_set_state(btn, LV_BTN_STATE_REL); } else { lv_obj_set_opa_scale_enable(autoboot_obj.obj1, false); lv_obj_set_click(autoboot_obj.obj2, true); lv_btn_set_state(btn, LV_BTN_STATE_TGL_REL); } autoboot_first_time = false; nyx_generic_onoff_toggle(btn); return LV_RES_INV; } static lv_res_t _autoboot_delay_action(lv_obj_t *ddlist) { h_cfg.bootwait = lv_ddlist_get_selected(ddlist); return LV_RES_OK; } static lv_res_t _data_verification_action(lv_obj_t *ddlist) { h_cfg.verification = lv_ddlist_get_selected(ddlist); return LV_RES_OK; } static void _reload_nyx() { b_cfg->boot_cfg = BOOT_CFG_AUTOBOOT_EN; b_cfg->autoboot = 0; b_cfg->autoboot_list = 0; b_cfg->extra_cfg = 0; void (*main_ptr)() = (void *)nyx_str->hekate; sd_unmount(true); reconfig_hw_workaround(false, 0); (*main_ptr)(); } static lv_res_t reload_action(lv_obj_t *btns, const char *txt) { if (!lv_btnm_get_pressed(btns)) _reload_nyx(); return mbox_action(btns, txt); } static bool do_reload = false; static void _check_sd_card_removed(void *params) { // The following checks if sdmmc_1 is initialized. // If yes and card was removed, shows a message box, // that will reload Nyx, when the card is inserted again. if (!do_reload && get_sd_card_removed()) { lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL); lv_obj_set_style(dark_bg, &mbox_darken); lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES); lv_obj_t *mbox = lv_mbox_create(dark_bg, NULL); lv_mbox_set_recolor_text(mbox, true); lv_obj_set_width(mbox, LV_HOR_RES * 4 / 9); lv_mbox_set_text(mbox, "\n#FF8000 SD card was removed!#\n\n#96FF00 Nyx will reload after inserting it.#\n"); lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0); lv_obj_set_top(mbox, true); do_reload = true; } // If in reload state and card was inserted, reload nyx. if (do_reload && !get_sd_card_removed()) _reload_nyx(); } static lv_res_t _reboot_action(lv_obj_t *btns, const char *txt) { u32 btnidx = lv_btnm_get_pressed(btns); switch (btnidx) { case 0: reboot_normal(); break; case 1: reboot_rcm(); break; } return mbox_action(btns, txt); } static lv_res_t _poweroff_action(lv_obj_t *btns, const char *txt) { if (!lv_btnm_get_pressed(btns)) power_off(); return mbox_action(btns, txt); } static lv_res_t _create_mbox_reload(lv_obj_t *btn) { lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL); lv_obj_set_style(dark_bg, &mbox_darken); lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES); static const char * mbox_btn_map[] = { "\221Reload", "\221Cancel", "" }; lv_obj_t *mbox = lv_mbox_create(dark_bg, NULL); lv_mbox_set_recolor_text(mbox, true); lv_obj_set_width(mbox, LV_HOR_RES * 4 / 10); lv_mbox_set_text(mbox, "#FF8000 Do you really want#\n#FF8000 to reload hekate?#"); lv_mbox_add_btns(mbox, mbox_btn_map, reload_action); lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0); lv_obj_set_top(mbox, true); return LV_RES_OK; } static lv_res_t _create_mbox_reboot(lv_obj_t *btn) { lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL); lv_obj_set_style(dark_bg, &mbox_darken); lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES); static const char * mbox_btn_map[] = { "\221OFW", "\221RCM", "\221Cancel", "" }; lv_obj_t *mbox = lv_mbox_create(dark_bg, NULL); lv_mbox_set_recolor_text(mbox, true); lv_obj_set_width(mbox, LV_HOR_RES / 2); lv_mbox_set_text(mbox, "#FF8000 Choose where to reboot:#"); lv_mbox_add_btns(mbox, mbox_btn_map, _reboot_action); lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0); lv_obj_set_top(mbox, true); return LV_RES_OK; } static lv_res_t _create_mbox_poweroff(lv_obj_t *btn) { lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL); lv_obj_set_style(dark_bg, &mbox_darken); lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES); static const char * mbox_btn_map[] = { "\221Power Off", "\221Cancel", "" }; lv_obj_t *mbox = lv_mbox_create(dark_bg, NULL); lv_mbox_set_recolor_text(mbox, true); lv_obj_set_width(mbox, LV_HOR_RES * 4 / 10); lv_mbox_set_text(mbox, "#FF8000 Do you really want#\n#FF8000 to power off?#"); lv_mbox_add_btns(mbox, mbox_btn_map, _poweroff_action); lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0); lv_obj_set_top(mbox, true); return LV_RES_OK; } void nyx_create_onoff_button(lv_theme_t *th, lv_obj_t *parent, lv_obj_t *btn, const char *btn_name, lv_action_t action, bool transparent) { // Create buttons that are flat and text, plus On/Off switch. static lv_style_t btn_onoff_rel_hos_style, btn_onoff_pr_hos_style; lv_style_copy(&btn_onoff_rel_hos_style, th->btn.rel); btn_onoff_rel_hos_style.body.shadow.width = 0; btn_onoff_rel_hos_style.body.border.width = 0; btn_onoff_rel_hos_style.body.padding.hor = 0; btn_onoff_rel_hos_style.body.radius = 0; btn_onoff_rel_hos_style.body.empty = 1; lv_style_copy(&btn_onoff_pr_hos_style, &btn_onoff_rel_hos_style); if (transparent) { btn_onoff_pr_hos_style.body.main_color = LV_COLOR_HEX(0xFFFFFF); btn_onoff_pr_hos_style.body.opa = 35; } else btn_onoff_pr_hos_style.body.main_color = LV_COLOR_HEX(0x3D3D3D); btn_onoff_pr_hos_style.body.grad_color = btn_onoff_pr_hos_style.body.main_color; btn_onoff_pr_hos_style.text.color = th->btn.pr->text.color; btn_onoff_pr_hos_style.body.empty = 0; lv_obj_t *label_btn = lv_label_create(btn, NULL); lv_obj_t *label_btnsw = NULL; lv_label_set_recolor(label_btn, true); label_btnsw = lv_label_create(btn, NULL); lv_label_set_recolor(label_btnsw, true); lv_btn_set_layout(btn, LV_LAYOUT_OFF); lv_btn_set_style(btn, LV_BTN_STYLE_REL, &btn_onoff_rel_hos_style); lv_btn_set_style(btn, LV_BTN_STYLE_PR, &btn_onoff_pr_hos_style); lv_btn_set_style(btn, LV_BTN_STYLE_TGL_REL, &btn_onoff_rel_hos_style); lv_btn_set_style(btn, LV_BTN_STYLE_TGL_PR, &btn_onoff_pr_hos_style); lv_btn_set_fit(btn, false, true); lv_obj_set_width(btn, lv_obj_get_width(parent)); lv_btn_set_toggle(btn, true); lv_label_set_text(label_btn, btn_name); lv_label_set_static_text(label_btnsw, "#00ffc9 OFF#"); lv_obj_align(label_btn, btn, LV_ALIGN_IN_LEFT_MID, LV_DPI / 4, 0); lv_obj_align(label_btnsw, btn, LV_ALIGN_IN_RIGHT_MID, -LV_DPI / 4, -LV_DPI / 10); if (action) lv_btn_set_action(btn, LV_BTN_ACTION_CLICK, action); } static void _create_text_button(lv_theme_t *th, lv_obj_t *parent, lv_obj_t *btn, const char *btn_name, lv_action_t action) { // Create buttons that are flat and only have a text label. static lv_style_t btn_onoff_rel_hos_style, btn_onoff_pr_hos_style; lv_style_copy(&btn_onoff_rel_hos_style, th->btn.rel); btn_onoff_rel_hos_style.body.shadow.width = 0; btn_onoff_rel_hos_style.body.border.width = 0; btn_onoff_rel_hos_style.body.radius = 0; btn_onoff_rel_hos_style.body.padding.hor = LV_DPI / 4; btn_onoff_rel_hos_style.body.empty = 1; lv_style_copy(&btn_onoff_pr_hos_style, &btn_onoff_rel_hos_style); if (hekate_bg) { btn_onoff_pr_hos_style.body.main_color = LV_COLOR_HEX(0xFFFFFF); btn_onoff_pr_hos_style.body.opa = 35; } else btn_onoff_pr_hos_style.body.main_color = LV_COLOR_HEX(0x3D3D3D); btn_onoff_pr_hos_style.body.grad_color = btn_onoff_pr_hos_style.body.main_color; btn_onoff_pr_hos_style.text.color = th->btn.pr->text.color; btn_onoff_pr_hos_style.body.empty = 0; lv_obj_t *label_btn = lv_label_create(btn, NULL); lv_label_set_recolor(label_btn, true); lv_btn_set_style(btn, LV_BTN_STYLE_REL, &btn_onoff_rel_hos_style); lv_btn_set_style(btn, LV_BTN_STYLE_PR, &btn_onoff_pr_hos_style); lv_btn_set_style(btn, LV_BTN_STYLE_TGL_REL, &btn_onoff_rel_hos_style); lv_btn_set_style(btn, LV_BTN_STYLE_TGL_PR, &btn_onoff_pr_hos_style); lv_btn_set_fit(btn, true, true); lv_label_set_text(label_btn, btn_name); if (action) lv_btn_set_action(btn, LV_BTN_ACTION_CLICK, action); } static void _create_tab_options(lv_theme_t *th, lv_obj_t *parent) { lv_page_set_scrl_layout(parent, LV_LAYOUT_PRETTY); static lv_style_t h_style; lv_style_copy(&h_style, &lv_style_transp); h_style.body.padding.inner = 0; h_style.body.padding.hor = LV_DPI - (LV_DPI / 4); h_style.body.padding.ver = LV_DPI / 6; // Create containers to keep content inside. lv_obj_t * sw_h2 = lv_cont_create(parent, NULL); lv_cont_set_style(sw_h2, &h_style); lv_cont_set_fit(sw_h2, false, true); lv_obj_set_width(sw_h2, (LV_HOR_RES / 9) * 4); lv_obj_set_click(sw_h2, false); lv_cont_set_layout(sw_h2, LV_LAYOUT_OFF); lv_obj_t * sw_h3 = lv_cont_create(parent, NULL); lv_cont_set_style(sw_h3, &h_style); lv_cont_set_fit(sw_h3, false, true); lv_obj_set_width(sw_h3, (LV_HOR_RES / 9) * 4); lv_obj_set_click(sw_h3, false); lv_cont_set_layout(sw_h3, LV_LAYOUT_OFF); lv_obj_t * l_cont = lv_cont_create(sw_h2, NULL); lv_cont_set_style(l_cont, &lv_style_transp_tight); lv_cont_set_fit(l_cont, true, true); lv_obj_set_click(l_cont, false); lv_cont_set_layout(l_cont, LV_LAYOUT_OFF); lv_obj_set_opa_scale(l_cont, LV_OPA_40); autoboot_obj.obj1 = l_cont; lv_obj_t *label_sep = lv_label_create(sw_h2, NULL); lv_label_set_static_text(label_sep, ""); // Create Auto Boot button. lv_obj_t *btn = lv_btn_create(sw_h2, NULL); if (hekate_bg) { lv_btn_set_style(btn, LV_BTN_STYLE_REL, &btn_transp_rel); lv_btn_set_style(btn, LV_BTN_STYLE_PR, &btn_transp_pr); lv_btn_set_style(btn, LV_BTN_STYLE_TGL_REL, &btn_transp_tgl_rel); lv_btn_set_style(btn, LV_BTN_STYLE_TGL_PR, &btn_transp_tgl_pr); } lv_btn_set_layout(btn, LV_LAYOUT_OFF); lv_obj_t *label_btn = lv_label_create(btn, NULL); lv_label_set_recolor(label_btn, true); lv_btn_set_fit(btn, true, true); lv_btn_set_toggle(btn, true); lv_label_set_static_text(label_btn, SYMBOL_GPS" Auto Boot #00ffc9 ON #"); lv_btn_set_action(btn, LV_BTN_ACTION_CLICK, _autoboot_hide_delay_action); lv_obj_align(btn, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, -LV_DPI / 18 + 6); lv_btn_set_fit(btn, false, false); autoboot_obj.obj0 = btn; lv_obj_t *label_txt2 = lv_label_create(sw_h2, NULL); lv_label_set_static_text(label_txt2, "Choose which boot entry or payload to automatically boot\nwhen injecting."); lv_obj_set_style(label_txt2, &hint_small_style); lv_obj_align(label_txt2, btn, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 3 - 4); lv_obj_t *line_sep = lv_line_create(sw_h2, NULL); static const lv_point_t line_pp[] = { {0, 0}, { LV_HOR_RES - (LV_DPI - (LV_DPI / 4)) * 2, 0} }; lv_line_set_points(line_sep, line_pp, 2); lv_line_set_style(line_sep, th->line.decor); lv_obj_align(line_sep, label_txt2, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 4); // Create Boot time delay list. lv_obj_t *label_txt = lv_label_create(l_cont, NULL); lv_label_set_static_text(label_txt, SYMBOL_CLOCK" Boot Time Delay "); lv_obj_set_style(label_txt, th->label.prim); lv_obj_align(label_txt, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4); lv_obj_t *ddlist = lv_ddlist_create(l_cont, NULL); lv_obj_set_top(ddlist, true); lv_ddlist_set_draw_arrow(ddlist, true); lv_ddlist_set_options(ddlist, "No bootlogo \n" "1 second\n" "2 seconds\n" "3 seconds\n" "4 seconds\n" "5 seconds\n" "6 seconds"); lv_ddlist_set_selected(ddlist, 3); //lv_ddlist_set_align(ddlist, LV_LABEL_ALIGN_RIGHT); lv_obj_align(ddlist, label_txt, LV_ALIGN_OUT_RIGHT_MID, LV_DPI / 4, 0); lv_ddlist_set_action(ddlist, _autoboot_delay_action); lv_ddlist_set_selected(ddlist, h_cfg.bootwait); autoboot_obj.obj2 = ddlist; if (hekate_bg) { lv_ddlist_set_style(ddlist, LV_DDLIST_STYLE_BG, &ddlist_transp_bg); lv_ddlist_set_style(ddlist, LV_DDLIST_STYLE_BGO, &ddlist_transp_bg); lv_ddlist_set_style(ddlist, LV_DDLIST_STYLE_PR, &ddlist_transp_sel); lv_ddlist_set_style(ddlist, LV_DDLIST_STYLE_SEL, &ddlist_transp_sel); } label_txt2 = lv_label_create(l_cont, NULL); lv_label_set_recolor(label_txt2, true); lv_label_set_static_text(label_txt2, "Set how long to show bootlogo when autoboot is enabled.\n" "#C7EA46 You can press# #FF8000 VOL-# #C7EA46 during that time to enter hekate's menu.#\n"); lv_obj_set_style(label_txt2, &hint_small_style); lv_obj_align(label_txt2, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 4); line_sep = lv_line_create(sw_h2, line_sep); lv_obj_align(line_sep, label_txt2, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 4); // Create Auto NoGC button. lv_obj_t *btn2 = lv_btn_create(sw_h2, NULL); nyx_create_onoff_button(th, sw_h2, btn2, SYMBOL_SHRK" Auto NoGC", auto_nogc_toggle, true); lv_obj_align(btn2, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 10); label_txt2 = lv_label_create(sw_h2, NULL); lv_label_set_recolor(label_txt2, true); lv_label_set_static_text(label_txt2, "It checks fuses and applies the patch automatically\n" "if higher firmware. It is now a global config and\n" "set on auto by default. (ON: Auto)\n\n\n"); lv_obj_set_style(label_txt2, &hint_small_style); lv_obj_align(label_txt2, btn2, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 12); label_sep = lv_label_create(sw_h3, NULL); lv_label_set_static_text(label_sep, ""); // Create Auto HOS Power Off button. lv_obj_t *btn3 = lv_btn_create(sw_h3, NULL); nyx_create_onoff_button(th, sw_h3, btn3, SYMBOL_POWER" Auto HOS Power Off", auto_hos_poweroff_toggle, true); lv_obj_align(btn3, label_sep, LV_ALIGN_OUT_BOTTOM_LEFT, 0, 0); label_txt2 = lv_label_create(sw_h3, NULL); lv_label_set_static_text(label_txt2, "When Shutdown is used from HOS, the device wakes up after\n" "15s. Enable this to automatically power off on the next\npayload injection."); lv_obj_set_style(label_txt2, &hint_small_style); lv_obj_align(label_txt2, btn3, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 12); line_sep = lv_line_create(sw_h3, line_sep); lv_obj_align(line_sep, label_txt2, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 4); // Create Backlight slider. label_txt = lv_label_create(sw_h3, NULL); lv_label_set_static_text(label_txt, SYMBOL_BRIGHTNESS" Backlight"); lv_obj_set_style(label_txt, th->label.prim); lv_obj_align(label_txt, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4); lv_obj_t * slider = lv_slider_create(sw_h3, NULL); lv_obj_set_width(slider, LV_DPI * 80 / 34); //lv_obj_set_height(slider, LV_DPI * 4 / 10); lv_bar_set_range(slider, 30, 220); lv_bar_set_value(slider, h_cfg.backlight); lv_slider_set_action(slider, _slider_brightness_action); lv_obj_align(slider, label_txt, LV_ALIGN_OUT_RIGHT_MID, LV_DPI * 20 / 15, 0); label_txt2 = lv_label_create(sw_h3, NULL); lv_label_set_static_text(label_txt2, "Set backlight brightness.\n\n"); lv_obj_set_style(label_txt2, &hint_small_style); lv_obj_align(label_txt2, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 4); line_sep = lv_line_create(sw_h3, line_sep); lv_obj_align(line_sep, label_txt2, LV_ALIGN_OUT_BOTTOM_LEFT, -(LV_DPI / 4), LV_DPI / 4); // Create Backup/Restore Verification list. label_txt = lv_label_create(sw_h3, NULL); lv_label_set_static_text(label_txt, SYMBOL_MODULES_ALT" Data Verification"); lv_obj_set_style(label_txt, th->label.prim); lv_obj_align(label_txt, line_sep, LV_ALIGN_OUT_BOTTOM_LEFT, LV_DPI / 4, LV_DPI / 4); lv_obj_t *ddlist2 = lv_ddlist_create(sw_h3, NULL); lv_obj_set_top(ddlist2, true); lv_ddlist_set_draw_arrow(ddlist2, true); lv_ddlist_set_options(ddlist2, "Off (Fastest)\n" "Sparse (Fast) \n" "Full (Slow)\n" "Full (Hashes)"); lv_ddlist_set_selected(ddlist2, h_cfg.verification); lv_obj_align(ddlist2, label_txt, LV_ALIGN_OUT_RIGHT_MID, LV_DPI * 3 / 8, 0); lv_ddlist_set_action(ddlist2, _data_verification_action); if (hekate_bg) { lv_ddlist_set_style(ddlist2, LV_DDLIST_STYLE_BG, &ddlist_transp_bg); lv_ddlist_set_style(ddlist2, LV_DDLIST_STYLE_BGO, &ddlist_transp_bg); lv_ddlist_set_style(ddlist2, LV_DDLIST_STYLE_PR, &ddlist_transp_sel); lv_ddlist_set_style(ddlist2, LV_DDLIST_STYLE_SEL, &ddlist_transp_sel); } label_txt2 = lv_label_create(sw_h3, NULL); lv_label_set_static_text(label_txt2, "Set the type of data verification done for backup and restore.\n" "Can be canceled without losing the backup/restore.\n\n\n\n"); lv_obj_set_style(label_txt2, &hint_small_style); lv_obj_align(label_txt2, label_txt, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI / 4); // Set default loaded states. if (h_cfg.autohosoff) lv_btn_set_state(btn3, LV_BTN_STATE_TGL_REL); if (h_cfg.autonogc) lv_btn_set_state(btn2, LV_BTN_STATE_TGL_REL); nyx_generic_onoff_toggle(btn2); nyx_generic_onoff_toggle(btn3); _autoboot_hide_delay_action(btn); lv_obj_set_top(l_cont, true); // Set the ddlist container at top. lv_obj_set_parent(ddlist, l_cont); // Reorder ddlist. lv_obj_set_top(ddlist, true); lv_obj_set_top(ddlist2, true); } static void _create_tab_about(lv_theme_t * th, lv_obj_t * parent) { lv_obj_t * lbl_credits = lv_label_create(parent, NULL); lv_obj_align(lbl_credits, NULL, LV_ALIGN_IN_TOP_LEFT, LV_DPI / 2, LV_DPI / 2); lv_ta_set_style(lbl_credits, LV_TA_STYLE_BG, &monospace_text); lv_label_set_recolor(lbl_credits, true); lv_label_set_static_text(lbl_credits, "hekate (c) 2018 naehrwert, st4rk\n\n" "CTCaer mod (c) 2018 CTCaer\n" "\n" "Thanks to: #00CCFF derrek, nedwill, plutoo, #\n" " #00CCFF shuffle2, smea, thexyz, yellows8 #\n" "\n" "Greetings to: fincs, hexkyz, SciresM,\n" " Shiny Quagsire, WinterMute\n" "\n" "Open source and free packages used:\n\n" " - FatFs R0.13c,\n" " Copyright (c) 2018, ChaN\n\n" " - bcl-1.2.0,\n" " Copyright (c) 2003-2006, Marcus Geelnard\n\n" " - Atmosphere (SE sha256, process id patches),\n" " Copyright (c) 2018, Atmosphere-NX\n\n" " - elfload,\n" " Copyright (c) 2014, Owen Shepherd\n" " Copyright (c) 2018, M4xw\n\n" " - Littlev Graphics Library,\n" " Copyright (c) 2016 Gabor Kiss-Vamosi" ); lv_obj_t * lbl_octopus = lv_label_create(parent, NULL); lv_obj_align(lbl_octopus, lbl_credits, LV_ALIGN_OUT_RIGHT_TOP, LV_DPI, 0); lv_ta_set_style(lbl_octopus, LV_TA_STYLE_BG, &monospace_text); lv_label_set_recolor(lbl_octopus, true); lv_label_set_static_text(lbl_octopus, "\n#00CCFF ___#\n" "#00CCFF .-' `'.#\n" "#00CCFF / \\#\n" "#00CCFF | ;#\n" "#00CCFF | | ___.--,#\n" "#00CCFF _.._ |0) = (0) | _.---'`__.-( (_.#\n" "#00CCFF __.--'`_.. '.__.\\ '--. \\_.-' ,.--'` `\"\"`#\n" "#00CCFF ( ,.--'` ',__ /./; ;, '.__.'` __#\n" "#00CCFF _`) ) .---.__.' / | |\\ \\__..--\"\" \"\"\"--.,_#\n" "#00CCFF `---' .'.''-._.-'`_./ /\\ '. \\ _.--''````'''--._`-.__.'#\n" "#00CCFF | | .' _.-' | | \\ \\ '. `----`#\n" "#00CCFF \\ \\/ .' \\ \\ '. '-._)#\n" "#00CCFF \\/ / \\ \\ `=.__`'-.#\n" "#00CCFF / /\\ `) ) / / `\"\".`\\#\n" "#00CCFF , _.-'.'\\ \\ / / ( ( / /#\n" "#00CCFF `--'` ) ) .-'.' '.'. | (#\n" "#00CCFF (/` ( (` ) ) '-; ##00FFCC [switchbrew]#\n" "#00CCFF ` '-; (-'#" ); lv_obj_t *hekate_img = lv_img_create(parent, NULL); lv_img_set_src(hekate_img, &hekate_logo); lv_obj_align(hekate_img, lbl_octopus, LV_ALIGN_OUT_BOTTOM_LEFT, 0, LV_DPI * 2 / 3); lv_obj_t *ctcaer_img = lv_img_create(parent, NULL); lv_img_set_src(ctcaer_img, &ctcaer_logo); lv_obj_align(ctcaer_img, lbl_octopus, LV_ALIGN_OUT_BOTTOM_RIGHT, 0, LV_DPI * 2 / 3); char version[32]; s_printf(version, "Nyx v%d.%d.%d", BL_VER_MJ, BL_VER_MN, BL_VER_HF); lv_obj_t * lbl_ver = lv_label_create(parent, NULL); lv_obj_align(lbl_ver, ctcaer_img, LV_ALIGN_OUT_BOTTOM_RIGHT, -LV_DPI / 20, LV_DPI / 4); lv_ta_set_style(lbl_ver, LV_TA_STYLE_BG, &monospace_text); lv_label_set_text(lbl_ver, version); } #pragma GCC pop_options static void _update_status_bar(void *params) { char *label = (char *)malloc(64); u16 soc_temp; u32 batt_percent; int charge_status; int batt_volt; int batt_curr; rtc_time_t time; // Get sensor data. max77620_rtc_get_time(&time); soc_temp = tmp451_get_soc_temp(false); bq24193_get_property(BQ24193_ChargeStatus, &charge_status); max17050_get_property(MAX17050_RepSOC, (int *)&batt_percent); max17050_get_property(MAX17050_VCELL, &batt_volt); max17050_get_property(MAX17050_Current, &batt_curr); //! TODO: Parse time and use offset. // Set time and SoC temperature. s_printf(label, "%02d:%02d "SYMBOL_DOT" "SYMBOL_TEMPERATURE" %02d.%d", time.hour, time.min, soc_temp >> 8, (soc_temp & 0xFF) / 10); lv_label_set_array_text(status_bar.time_temp, label, 64); lv_obj_realign(status_bar.temp_symbol); lv_obj_realign(status_bar.temp_degrees); // Set battery percent and charging symbol. s_printf(label, " "SYMBOL_DOT" %d.%d%% ", (batt_percent >> 8) & 0xFF, (batt_percent & 0xFF) / 26); u8 batt_level = (batt_percent >> 8) & 0xFF; if (batt_level > 80) s_printf(label + strlen(label), SYMBOL_BATTERY_FULL); else if (batt_level > 60) s_printf(label + strlen(label), SYMBOL_BATTERY_3); else if (batt_level > 40) s_printf(label + strlen(label), SYMBOL_BATTERY_2); else if (batt_level > 15) s_printf(label + strlen(label), SYMBOL_BATTERY_1); else s_printf(label + strlen(label), "#FF3C28 "SYMBOL_BATTERY_EMPTY"#"); if (charge_status) s_printf(label + strlen(label), " #FFDD00 "SYMBOL_CHARGE"#"); lv_label_set_array_text(status_bar.battery, label, 64); lv_obj_realign(status_bar.battery); // Set battery current draw and voltage. if (batt_curr >= 0) s_printf(label, "#96FF00 +%d", batt_curr / 1000); else s_printf(label, "#FF3C28 -%d", (~batt_curr + 1) / 1000); s_printf(label + strlen(label), " mA# (%d mV)", batt_volt); lv_label_set_array_text(status_bar.battery_more, label, 64); lv_obj_realign(status_bar.battery_more); free(label); } static lv_res_t _create_mbox_payloads(lv_obj_t *btn) { lv_obj_t *dark_bg = lv_obj_create(lv_scr_act(), NULL); lv_obj_set_style(dark_bg, &mbox_darken); lv_obj_set_size(dark_bg, LV_HOR_RES, LV_VER_RES); static const char * mbox_btn_map[] = { "\211", "\222Cancel", "\211", "" }; lv_obj_t *mbox = lv_mbox_create(dark_bg, NULL); lv_mbox_set_recolor_text(mbox, true); lv_obj_set_width(mbox, LV_HOR_RES * 5 / 9); lv_mbox_set_text(mbox, "Select a payload to launch:"); // Create a list with all found payloads. //! TODO: SHould that be tabs with buttons? + Icon support? lv_obj_t *list = lv_list_create(mbox, NULL); payload_list = list; lv_obj_set_size(list, LV_HOR_RES * 3 / 7, LV_VER_RES * 3 / 7); lv_list_set_single_mode(list, true); if (!sd_mount()) { lv_mbox_set_text(mbox, "#FFDD00 Failed to init SD!#"); goto out_end; } char *dir = (char *)malloc(256); strcpy(dir, "bootloader/payloads"); char *filelist = dirlist(dir, NULL, false, false); sd_unmount(false); u32 i = 0; if (filelist) { while (true) { if (!filelist[i * 256]) break; lv_list_add(list, NULL, &filelist[i * 256], launch_payload); i++; } } out_end: lv_mbox_add_btns(mbox, mbox_btn_map, mbox_action); lv_obj_align(mbox, NULL, LV_ALIGN_CENTER, 0, 0); lv_obj_set_top(mbox, true); return LV_RES_OK; } static lv_obj_t *launch_ctxt[16]; static lv_res_t _launch_more_cfg_action(lv_obj_t *btn) { lv_btn_ext_t *ext = lv_obj_get_ext_attr(btn); _launch_hos(ext->idx, 1); return LV_RES_OK; } static lv_res_t _win_launch_close_action(lv_obj_t * btn) { // Cleanup icons. lv_btn_ext_t *ext; lv_obj_t *btn_tmp; for (u32 i = 0; i < 16; i += 2) { btn_tmp = launch_ctxt[i]; ext = lv_obj_get_ext_attr(btn_tmp); if (ext->idx) { btn_tmp = lv_obj_get_child(btn_tmp, NULL); lv_img_dsc_t *tmp = (lv_img_dsc_t *)lv_img_get_src(btn_tmp); // Avoid freeing base icons. if ((tmp != icon_switch) && (tmp != icon_payload) && (tmp != icon_lakka)) free(tmp); } } lv_obj_t * win = lv_win_get_from_btn(btn); lv_obj_del(win); return LV_RES_INV; } lv_obj_t *create_window_launch(const char *win_title) { static lv_style_t win_bg_style; lv_style_copy(&win_bg_style, &lv_style_plain); win_bg_style.body.main_color = LV_COLOR_HEX(0x2D2D2D);// TODO: COLOR_HOS_BG win_bg_style.body.grad_color = win_bg_style.body.main_color; lv_obj_t *win = lv_win_create(lv_scr_act(), NULL); lv_win_set_title(win, win_title); lv_win_set_style(win, LV_WIN_STYLE_BG, &win_bg_style); lv_obj_set_size(win, LV_HOR_RES, LV_VER_RES); lv_win_add_btn(win, NULL, SYMBOL_CLOSE" Close", _win_launch_close_action); return win; } static lv_res_t _launch_action(lv_obj_t *btn) { lv_btn_ext_t *ext = lv_obj_get_ext_attr(btn); _launch_hos(ext->idx, 0); return LV_RES_OK; } static lv_res_t _create_window_home_launch(lv_obj_t *btn) { char *icon_path; static lv_style_t btn_home_transp_rel; lv_style_copy(&btn_home_transp_rel, lv_theme_get_current()->btn.rel); btn_home_transp_rel.body.opa = LV_OPA_0; static lv_style_t btn_home_transp_pr; lv_style_copy(&btn_home_transp_pr, lv_theme_get_current()->btn.pr); btn_home_transp_pr.body.main_color = LV_COLOR_HEX(0xFFFFFF); btn_home_transp_pr.body.grad_color = btn_home_transp_pr.body.main_color; btn_home_transp_pr.body.opa = LV_OPA_30; lv_obj_t *win; bool more_cfg = false; if (strcmp(lv_label_get_text(lv_obj_get_child(btn, NULL)),"#00EDBA Launch#")) more_cfg = true; if (!more_cfg) win = create_window_launch(SYMBOL_GPS" Launch"); else win = create_window_launch(SYMBOL_GPS" More Configurations"); lv_cont_set_fit(lv_page_get_scrl(lv_win_get_content(win)), false, false); lv_page_set_scrl_height(lv_win_get_content(win), 572); lv_obj_t *btn_boot_entry; lv_obj_t *boot_entry_lbl_cont; lv_obj_t *boot_entry_label; bool no_boot_entries = false; u32 max_entries = 8; lv_btn_ext_t * ext; //! TODO: Use an array/loop for the initialization. // Create CFW buttons. btn_boot_entry = lv_btn_create(win, NULL); launch_ctxt[0] = btn_boot_entry; // Buttons are 200 x 200 with 4 pixel borders. // Icons must be <= 192 x 192. // Button 0. lv_obj_set_size(btn_boot_entry, 200, 200); lv_obj_set_pos(btn_boot_entry, 19, 36); lv_obj_set_opa_scale(btn_boot_entry, LV_OPA_0); lv_obj_set_opa_scale_enable(btn_boot_entry, true); lv_btn_set_layout(btn_boot_entry, LV_LAYOUT_OFF); boot_entry_lbl_cont = lv_cont_create(win, NULL); boot_entry_label = lv_label_create(boot_entry_lbl_cont, NULL); lv_obj_set_style(boot_entry_label, &hint_small_style_white); lv_label_set_static_text(boot_entry_label, ""); launch_ctxt[1] = boot_entry_label; lv_cont_set_fit(boot_entry_lbl_cont, false, false); lv_cont_set_layout(boot_entry_lbl_cont, LV_LAYOUT_CENTER); lv_obj_set_size(boot_entry_lbl_cont, 238, 20); lv_obj_set_pos(boot_entry_lbl_cont, 0, 245); // Button 1. btn_boot_entry = lv_btn_create(win, btn_boot_entry); launch_ctxt[2] = btn_boot_entry; lv_obj_set_pos(btn_boot_entry, 340, 36); boot_entry_lbl_cont = lv_cont_create(win, boot_entry_lbl_cont); boot_entry_label = lv_label_create(boot_entry_lbl_cont, boot_entry_label); lv_obj_set_pos(boot_entry_lbl_cont, 321, 245); launch_ctxt[3] = boot_entry_label; // Button 2. btn_boot_entry = lv_btn_create(win, btn_boot_entry); launch_ctxt[4] = btn_boot_entry; lv_obj_set_pos(btn_boot_entry, 661, 36); boot_entry_lbl_cont = lv_cont_create(win, boot_entry_lbl_cont); boot_entry_label = lv_label_create(boot_entry_lbl_cont, boot_entry_label); lv_obj_set_pos(boot_entry_lbl_cont, 642, 245); launch_ctxt[5] = boot_entry_label; // Button 3. btn_boot_entry = lv_btn_create(win, btn_boot_entry); launch_ctxt[6] = btn_boot_entry; lv_obj_set_pos(btn_boot_entry, 982, 36); boot_entry_lbl_cont = lv_cont_create(win, boot_entry_lbl_cont); boot_entry_label = lv_label_create(boot_entry_lbl_cont, boot_entry_label); lv_obj_set_pos(boot_entry_lbl_cont, 963, 245); launch_ctxt[7] = boot_entry_label; // Button 4. btn_boot_entry = lv_btn_create(win, btn_boot_entry); launch_ctxt[8] = btn_boot_entry; lv_obj_set_pos(btn_boot_entry, 19, 313); boot_entry_lbl_cont = lv_cont_create(win, boot_entry_lbl_cont); boot_entry_label = lv_label_create(boot_entry_lbl_cont, boot_entry_label); lv_obj_set_pos(boot_entry_lbl_cont, 0, 522); launch_ctxt[9] = boot_entry_label; // Button 5. btn_boot_entry = lv_btn_create(win, btn_boot_entry); launch_ctxt[10] = btn_boot_entry; lv_obj_set_pos(btn_boot_entry, 340, 313); boot_entry_lbl_cont = lv_cont_create(win, boot_entry_lbl_cont); boot_entry_label = lv_label_create(boot_entry_lbl_cont, boot_entry_label); lv_obj_set_pos(boot_entry_lbl_cont, 321, 522); launch_ctxt[11] = boot_entry_label; // Button 6. btn_boot_entry = lv_btn_create(win, btn_boot_entry); launch_ctxt[12] = btn_boot_entry; lv_obj_set_pos(btn_boot_entry, 661, 313); boot_entry_lbl_cont = lv_cont_create(win, boot_entry_lbl_cont); boot_entry_label = lv_label_create(boot_entry_lbl_cont, boot_entry_label); lv_obj_set_pos(boot_entry_lbl_cont, 642, 522); launch_ctxt[13] = boot_entry_label; // Button 7. btn_boot_entry = lv_btn_create(win, btn_boot_entry); launch_ctxt[14] = btn_boot_entry; lv_obj_set_pos(btn_boot_entry, 982, 313); boot_entry_lbl_cont = lv_cont_create(win, boot_entry_lbl_cont); boot_entry_label = lv_label_create(boot_entry_lbl_cont, boot_entry_label); lv_obj_set_pos(boot_entry_lbl_cont, 963, 522); launch_ctxt[15] = boot_entry_label; // Parse ini boot entries and set buttons/icons. char *tmp_path = calloc(0x80, 1); LIST_INIT(ini_sections); if (sd_mount()) { // Choose what to parse. bool ini_parse_success = false; if (!more_cfg) ini_parse_success = ini_parse(&ini_sections, "bootloader/hekate_ipl.ini", false); else ini_parse_success = ini_parse(&ini_sections, "bootloader/ini", true); if (ini_parse_success) { // Iterate to all boot entries and load icons. u32 i = 1, x = 0; LIST_FOREACH_ENTRY(ini_sec_t, ini_sec, &ini_sections, link) { if (!strcmp(ini_sec->name, "config") || (ini_sec->type != INI_CHOICE)) continue; icon_path = NULL; bool payload = false; lv_img_dsc_t *bmp = NULL; // Check for icons. LIST_FOREACH_ENTRY(ini_kv_t, kv, &ini_sec->kvs, link) { if (!strcmp("icon", kv->key)) { icon_path = kv->val; break; } else if (!strcmp("payload", kv->key)) { payload = true; break; } } // If icon not found, check res folder for section_name.bmp. // If not, use defaults. if (!icon_path) { s_printf(tmp_path, "bootloader/res/%s.bmp", ini_sec->name); bmp = bmp_to_lvimg_obj(tmp_path); if (!bmp) { if (!strcmp(ini_sec->name, "Lakka")) bmp = icon_lakka; else if (payload) bmp = icon_payload; } } else bmp = bmp_to_lvimg_obj(icon_path); // Enable button. ext = lv_obj_get_ext_attr(launch_ctxt[x]); ext->idx = i; lv_obj_set_opa_scale(launch_ctxt[x], LV_OPA_COVER); // Default to switch logo if no icon found at all. if (!bmp) bmp = icon_switch; //Set icon and border/radius mask. if (bmp) { lv_obj_t *img = lv_img_create(launch_ctxt[x], NULL); lv_img_set_src(img, bmp); lv_obj_t *btn = lv_btn_create(launch_ctxt[x], NULL); lv_obj_set_size(btn, 200, 200); lv_btn_set_style(btn, LV_BTN_STYLE_REL, &btn_home_transp_rel); lv_btn_set_style(btn, LV_BTN_STYLE_PR, &btn_home_transp_pr); lv_obj_align(img, NULL, LV_ALIGN_CENTER, 0, 0); ext = lv_obj_get_ext_attr(btn); ext->idx = i; if (!more_cfg) lv_btn_set_action(btn, LV_BTN_ACTION_CLICK, _launch_action); else lv_btn_set_action(btn, LV_BTN_ACTION_CLICK, _launch_more_cfg_action); } // Set button's label text. lv_label_set_array_text(launch_ctxt[x + 1], ini_sec->name, strlen(ini_sec->name)); lv_obj_set_opa_scale(launch_ctxt[x + 1], LV_OPA_COVER); //! TODO: Check strlen and decide on rolling text //lv_label_set_long_mode(boot_entry_label, LV_LABEL_LONG_ROLL); i++; x += 2; if (i > max_entries) break; } if (i < 2) no_boot_entries = true; } else no_boot_entries = true; } else no_boot_entries = true; sd_unmount(false); free(tmp_path); // No boot entries found. if (no_boot_entries) { lv_obj_t *label_error = lv_label_create(win, NULL); lv_label_set_recolor(label_error, true); if (!more_cfg) { lv_label_set_static_text(label_error, "#FFDD00 No main boot entries found...\n" "You can use the following entry to boot stock,\n" "or use #C7EA46 More configs# button for more boot entries."); } else { lv_label_set_static_text(label_error, "#FFDD00 No .ini or boot entries found...\n" "Check that a .ini file exists in #96FF00 /bootloader/ini/#,\n" "and that it contains at least one entry."); } lv_obj_set_pos(label_error, 19, 0); } return LV_RES_OK; } #pragma GCC push_options #pragma GCC target ("thumb") static void _create_tab_home(lv_theme_t *th, lv_obj_t *parent) { lv_page_set_scrl_layout(parent, LV_LAYOUT_OFF); lv_page_set_scrl_fit(parent, false, false); lv_page_set_scrl_height(parent, 592); // Set brand label. lv_obj_t *label_brand = lv_label_create(parent, NULL); lv_label_set_recolor(label_brand, true); lv_label_set_static_text(label_brand, "#00EDBA hekate"); lv_obj_set_pos(label_brand, 50, 48); // Set tagline label. lv_obj_t *label_tagline = lv_label_create(parent, NULL); lv_obj_set_style(label_tagline, &hint_small_style_white); lv_label_set_static_text(label_tagline, "THE ALL IN ONE BOOTLOADER FOR ALL YOUR NEEDS"); lv_obj_set_pos(label_tagline, 50, 82); static lv_style_t icons; lv_style_copy(&icons, th->label.prim); icons.text.font = &hekate_symbol_120; // Launch button. lv_obj_t *btn_launch = lv_btn_create(parent, NULL); if (hekate_bg) { lv_btn_set_style(btn_launch, LV_BTN_STYLE_REL, &btn_transp_rel); lv_btn_set_style(btn_launch, LV_BTN_STYLE_PR, &btn_transp_pr); } lv_obj_t *label_btn = lv_label_create(btn_launch, NULL); lv_label_set_recolor(label_btn, true); lv_obj_set_style(label_btn, &icons); lv_label_set_static_text(label_btn, "#00EDBA "SYMBOL_DOT"#"); lv_btn_set_action(btn_launch, LV_BTN_ACTION_CLICK, _create_window_home_launch); lv_obj_set_size(btn_launch, 256, 256); lv_obj_set_pos(btn_launch, 50, 160); lv_btn_set_layout(btn_launch, LV_LAYOUT_OFF); lv_obj_align(label_btn, NULL, LV_ALIGN_CENTER, 0, -28); lv_obj_t *label_btn2 = lv_label_create(btn_launch, NULL); lv_label_set_recolor(label_btn2, true); lv_label_set_static_text(label_btn2, "#00EDBA Launch#"); lv_obj_align(label_btn2, NULL, LV_ALIGN_IN_TOP_MID, 0, 174); // More Configs button. lv_obj_t *btn_more_cfg = lv_btn_create(parent, btn_launch); label_btn = lv_label_create(btn_more_cfg, label_btn); lv_label_set_static_text(label_btn, "#00EDBA "SYMBOL_CLOCK"#"); lv_btn_set_action(btn_more_cfg, LV_BTN_ACTION_CLICK, _create_window_home_launch); lv_btn_set_layout(btn_more_cfg, LV_LAYOUT_OFF); lv_obj_align(label_btn, NULL, LV_ALIGN_CENTER, 0, -28); label_btn2 = lv_label_create(btn_more_cfg, label_btn2); lv_label_set_static_text(label_btn2, "#00EDBA More Configs#"); lv_obj_set_pos(btn_more_cfg, 341, 160); lv_obj_align(label_btn2, NULL, LV_ALIGN_IN_TOP_MID, 0, 174); // Quick Launch button. // lv_obj_t *btn_quick_launch = lv_btn_create(parent, NULL); // label_btn = lv_label_create(btn_quick_launch, label_btn); // lv_label_set_static_text(label_btn, SYMBOL_EDIT" Quick Launch"); // lv_obj_set_width(btn_quick_launch, 256); // lv_obj_set_pos(btn_quick_launch, 343, 448); // lv_btn_set_action(btn_quick_launch, LV_BTN_ACTION_CLICK, NULL); // Payloads button. lv_obj_t *btn_payloads = lv_btn_create(parent, btn_launch); label_btn = lv_label_create(btn_payloads, label_btn); lv_label_set_static_text(label_btn, "#00EDBA "SYMBOL_OK"#"); lv_btn_set_action(btn_payloads, LV_BTN_ACTION_CLICK, _create_mbox_payloads); lv_btn_set_layout(btn_payloads, LV_LAYOUT_OFF); lv_obj_align(label_btn, NULL, LV_ALIGN_CENTER, 0, -28); label_btn2 = lv_label_create(btn_payloads, label_btn2); lv_label_set_static_text(label_btn2, "#00EDBA Payloads#"); lv_obj_set_pos(btn_payloads, 632, 160); lv_obj_align(label_btn2, NULL, LV_ALIGN_IN_TOP_MID, 0, 174); lv_obj_t *line_sep = lv_line_create(parent, NULL); static const lv_point_t line_pp[] = { {0, 0}, {0, LV_DPI * 3} }; lv_line_set_points(line_sep, line_pp, 2); lv_line_set_style(line_sep, th->line.decor); lv_obj_align(line_sep, btn_payloads, LV_ALIGN_OUT_RIGHT_MID, 35, 0); // emuMMC manage button. lv_obj_t *btn_emummc = lv_btn_create(parent, btn_launch); label_btn = lv_label_create(btn_emummc, label_btn); lv_label_set_static_text(label_btn, "#00EDBA "SYMBOL_LIST"#"); lv_btn_set_action(btn_emummc, LV_BTN_ACTION_CLICK,create_win_emummc_tools); lv_btn_set_layout(btn_emummc, LV_LAYOUT_OFF); lv_obj_align(label_btn, NULL, LV_ALIGN_CENTER, 0, -28); lv_obj_set_pos(btn_emummc, 959, 160); label_btn2 = lv_label_create(btn_emummc, label_btn2); lv_label_set_static_text(label_btn2, "#00EDBA emuMMC#"); lv_obj_align(label_btn2, NULL, LV_ALIGN_IN_TOP_MID, 0, 174); // Create bottom right power buttons. lv_obj_t *btn_reboot = lv_btn_create(parent, NULL); lv_obj_t *btn_power_off = lv_btn_create(parent, NULL); lv_obj_t *btn_reload = lv_btn_create(parent, NULL); _create_text_button(th, NULL, btn_power_off, SYMBOL_POWER" Power Off", _create_mbox_poweroff); lv_obj_align(btn_power_off, NULL, LV_ALIGN_IN_BOTTOM_RIGHT, -LV_DPI / 4, -LV_DPI / 12); _create_text_button(th, NULL, btn_reboot, SYMBOL_REBOOT" Reboot", _create_mbox_reboot); lv_obj_align(btn_reboot, btn_power_off, LV_ALIGN_OUT_LEFT_MID, 0, 0); _create_text_button(th, NULL, btn_reload, SYMBOL_REFRESH" Reload", _create_mbox_reload); lv_obj_align(btn_reload, btn_reboot, LV_ALIGN_OUT_LEFT_MID, 0, 0); } static lv_res_t _save_options_action(lv_obj_t *btn) { static const char * mbox_btn_map[] = {"\211", "\222OK!", "\211", ""}; lv_obj_t * mbox = lv_mbox_create(lv_scr_act(), NULL); lv_mbox_set_recolor_text(mbox, true); int res = !create_config_entry(); if (res) lv_mbox_set_text(mbox, "#FF8000 hekate Configuration#\n\n#96FF00 The configuration was saved to sd card!#"); else lv_mbox_set_text(mbox, "#FF8000 hekate Configuration#\n\n#FFDD00 Failed to save the configuration#\n#FFDD00 to sd card!#"); lv_mbox_add_btns(mbox, mbox_btn_map, NULL); lv_obj_set_top(mbox, true); return LV_RES_OK; } static void _create_status_bar(lv_theme_t * th) { static lv_obj_t *status_bar_bg; status_bar_bg = lv_cont_create(lv_layer_top(), NULL); static lv_style_t status_bar_style; lv_style_copy(&status_bar_style, &lv_style_plain_color); status_bar_style.body.opa = LV_OPA_0; status_bar_style.body.shadow.width = 0; lv_obj_set_style(status_bar_bg, &status_bar_style); lv_obj_set_size(status_bar_bg, LV_HOR_RES, LV_DPI * 9 / 14); lv_obj_align(status_bar_bg, NULL, LV_ALIGN_IN_BOTTOM_LEFT, 0, 0); // Battery percentages. lv_obj_t *lbl_battery = lv_label_create(status_bar_bg, NULL); lv_label_set_recolor(lbl_battery, true); lv_label_set_static_text(lbl_battery, " "SYMBOL_DOT" 00.0% "SYMBOL_BATTERY_1" #FFDD00 "SYMBOL_CHARGE"#"); lv_obj_align(lbl_battery, NULL, LV_ALIGN_IN_RIGHT_MID, -LV_DPI * 6 / 11, 0); status_bar.battery = lbl_battery; // Amperages, voltages. lbl_battery = lv_label_create(status_bar_bg, lbl_battery); lv_obj_set_style(lbl_battery, &hint_small_style_white); lv_label_set_static_text(lbl_battery, "#96FF00 +0 mA# (0 mV)"); lv_obj_align(lbl_battery, status_bar.battery, LV_ALIGN_OUT_LEFT_MID, -LV_DPI / 25, -1); status_bar.battery_more = lbl_battery; lv_obj_t *lbl_left = lv_label_create(status_bar_bg, NULL); lv_label_set_static_text(lbl_left, SYMBOL_CLOCK" "); lv_obj_align(lbl_left, NULL, LV_ALIGN_IN_LEFT_MID, LV_DPI * 6 / 11, 0); // Time, temperature. lv_obj_t *lbl_time_temp = lv_label_create(status_bar_bg, NULL); lv_label_set_static_text(lbl_time_temp, "00:00 "SYMBOL_DOT" "SYMBOL_TEMPERATURE" 00.0"); lv_obj_align(lbl_time_temp, lbl_left, LV_ALIGN_OUT_RIGHT_MID, 0, 0); status_bar.time_temp = lbl_time_temp; lbl_left = lv_label_create(status_bar_bg, NULL); lv_label_set_static_text(lbl_left, " "SYMBOL_DOT); lv_obj_align(lbl_left, lbl_time_temp, LV_ALIGN_OUT_RIGHT_MID, 0, -LV_DPI / 14); status_bar.temp_symbol = lbl_left; lv_obj_t *lbl_degrees = lv_label_create(status_bar_bg, NULL); lv_label_set_static_text(lbl_degrees, "C"); lv_obj_align(lbl_degrees, lbl_left, LV_ALIGN_OUT_RIGHT_MID, LV_DPI / 50, LV_DPI / 14); status_bar.temp_degrees = lbl_degrees; // Middle button. //! TODO: Utilize it for more. lv_obj_t *btn_mid = lv_btn_create(status_bar_bg, NULL); lv_obj_t *lbl_mid = lv_label_create(btn_mid, NULL); lv_label_set_static_text(lbl_mid, "Save Options"); lv_obj_set_size(btn_mid, LV_DPI * 5 / 2, LV_DPI / 2); lv_obj_align(btn_mid, NULL, LV_ALIGN_CENTER, 0, 0); status_bar.mid = btn_mid; lv_obj_set_opa_scale(status_bar.mid, LV_OPA_0); lv_obj_set_opa_scale_enable(status_bar.mid, true); lv_obj_set_click(status_bar.mid, false); lv_btn_set_action(btn_mid, LV_BTN_ACTION_CLICK, _save_options_action); } static lv_res_t _show_hide_save_button(lv_obj_t *tv, uint16_t tab_idx) { if (tab_idx == 4) // Options. { lv_obj_set_opa_scale(status_bar.mid, LV_OPA_COVER); lv_obj_set_click(status_bar.mid, true); } else { lv_obj_set_opa_scale(status_bar.mid, LV_OPA_0); lv_obj_set_click(status_bar.mid, false); } return LV_RES_OK; } static void _nyx_set_default_styles(lv_theme_t * th) { lv_style_copy(&mbox_darken, &lv_style_plain); mbox_darken.body.main_color = LV_COLOR_BLACK; mbox_darken.body.grad_color = mbox_darken.body.main_color; mbox_darken.body.opa = LV_OPA_30; lv_style_copy(&hint_small_style, th->label.hint); hint_small_style.text.letter_space = 1; hint_small_style.text.font = &interui_20; lv_style_copy(&hint_small_style_white, th->label.prim); hint_small_style_white.text.letter_space = 1; hint_small_style_white.text.font = &interui_20; lv_style_copy(&monospace_text, &lv_style_plain); monospace_text.body.main_color = LV_COLOR_HEX(0x1b1b1b); monospace_text.body.grad_color = LV_COLOR_HEX(0x1b1b1b); monospace_text.body.border.color = LV_COLOR_HEX(0x1b1b1b); monospace_text.body.border.width = 0; monospace_text.body.opa = LV_OPA_TRANSP; monospace_text.text.color = LV_COLOR_HEX(0xD8D8D8); monospace_text.text.font = &ubuntu_mono; monospace_text.text.letter_space = 0; monospace_text.text.line_space = 0; lv_style_copy(&btn_transp_rel, th->btn.rel); btn_transp_rel.body.main_color = LV_COLOR_HEX(0x444444); btn_transp_rel.body.grad_color = btn_transp_rel.body.main_color; btn_transp_rel.body.opa = LV_OPA_50; lv_style_copy(&btn_transp_pr, th->btn.pr); btn_transp_pr.body.main_color = LV_COLOR_HEX(0x888888); btn_transp_pr.body.grad_color = btn_transp_pr.body.main_color; btn_transp_pr.body.opa = LV_OPA_50; lv_style_copy(&btn_transp_tgl_rel, th->btn.tgl_rel); btn_transp_tgl_rel.body.main_color = LV_COLOR_HEX(0x444444); btn_transp_tgl_rel.body.grad_color = btn_transp_tgl_rel.body.main_color; btn_transp_tgl_rel.body.opa = LV_OPA_50; lv_style_copy(&btn_transp_tgl_pr, th->btn.tgl_pr); btn_transp_tgl_pr.body.main_color = LV_COLOR_HEX(0x888888); btn_transp_tgl_pr.body.grad_color = btn_transp_tgl_pr.body.main_color; btn_transp_tgl_pr.body.opa = LV_OPA_50; lv_style_copy(&ddlist_transp_bg, th->ddlist.bg); ddlist_transp_bg.body.main_color = LV_COLOR_HEX(0x2D2D2D); ddlist_transp_bg.body.grad_color = ddlist_transp_bg.body.main_color; ddlist_transp_bg.body.opa = 180; lv_style_copy(&ddlist_transp_sel, th->ddlist.sel); ddlist_transp_sel.body.main_color = LV_COLOR_HEX(0x4D4D4D); ddlist_transp_sel.body.grad_color = ddlist_transp_sel.body.main_color; ddlist_transp_sel.body.opa = 180; lv_style_copy(&tabview_btn_pr, th->tabview.btn.pr); tabview_btn_pr.body.main_color = LV_COLOR_HEX(0xFFFFFF); tabview_btn_pr.body.grad_color = tabview_btn_pr.body.main_color; tabview_btn_pr.body.opa = 35; lv_style_copy(&tabview_btn_tgl_pr, th->tabview.btn.tgl_pr); tabview_btn_tgl_pr.body.main_color = LV_COLOR_HEX(0xFFFFFF); tabview_btn_tgl_pr.body.grad_color = tabview_btn_tgl_pr.body.main_color; tabview_btn_tgl_pr.body.opa = 35; } static void _nyx_main_menu(lv_theme_t * th) { static lv_style_t no_padding; lv_style_copy(&no_padding, &lv_style_transp); no_padding.body.padding.hor = 0; // Initialize global styles. _nyx_set_default_styles(th); // Create screen container. lv_obj_t *scr = lv_cont_create(NULL, NULL); lv_scr_load(scr); lv_cont_set_style(scr, th->bg); // Create base background and add a custom one if exists. lv_obj_t *cnr = lv_cont_create(scr, NULL); static lv_style_t base_bg_style; lv_style_copy(&base_bg_style, &lv_style_plain_color); base_bg_style.body.main_color = LV_COLOR_HEX(0x2D2D2D); base_bg_style.body.grad_color = LV_COLOR_HEX(0x2D2D2D); lv_cont_set_style(cnr, &base_bg_style); lv_obj_set_size(cnr, LV_HOR_RES, LV_VER_RES); if (hekate_bg) { lv_obj_t *img = lv_img_create(cnr, NULL); lv_img_set_src(img, hekate_bg); } // Add tabview page to screen. lv_obj_t *tv = lv_tabview_create(scr, NULL); if (hekate_bg) { lv_tabview_set_style(tv, LV_TABVIEW_STYLE_BTN_PR, &tabview_btn_pr); lv_tabview_set_style(tv, LV_TABVIEW_STYLE_BTN_TGL_PR, &tabview_btn_tgl_pr); } lv_tabview_set_sliding(tv, false); lv_obj_set_size(tv, LV_HOR_RES, LV_VER_RES); // Add all tabs content. char version[32]; s_printf(version, "hekate v%d.%d.%d", nyx_str->version & 0xFF, (nyx_str->version >> 8) & 0xFF, (nyx_str->version >> 16) & 0xFF); lv_obj_t *tab_about = lv_tabview_add_tab(tv, version); lv_obj_t *tab_home = lv_tabview_add_tab(tv, SYMBOL_HOME" Home"); lv_obj_t *tab_tools = lv_tabview_add_tab(tv, SYMBOL_TOOLS" Tools"); lv_page_set_style(tab_tools, LV_PAGE_STYLE_BG, &no_padding); lv_page_set_style(tab_tools, LV_PAGE_STYLE_SCRL, &no_padding); lv_obj_t *tab_info = lv_tabview_add_tab(tv, SYMBOL_INFO" Console Info"); lv_page_set_style(tab_info, LV_PAGE_STYLE_BG, &no_padding); lv_page_set_style(tab_info, LV_PAGE_STYLE_SCRL, &no_padding); lv_obj_t *tab_options = lv_tabview_add_tab(tv, SYMBOL_SETTINGS" Options"); _create_tab_about(th, tab_about); _create_tab_home(th, tab_home); create_tab_tools(th, tab_tools); create_tab_info(th, tab_info); _create_tab_options(th, tab_options); lv_tabview_set_tab_act(tv, 1, false); // Create status bar. _create_status_bar(th); // Create tasks. system_tasks.task.dram_periodic_comp = lv_task_create(minerva_periodic_training, 500, LV_TASK_PRIO_HIGHEST, NULL); lv_task_ready(system_tasks.task.status_bar); system_tasks.task.status_bar = lv_task_create(_update_status_bar, 5000, LV_TASK_PRIO_LOW, NULL); lv_task_ready(system_tasks.task.status_bar); lv_task_create(_check_sd_card_removed, 2000, LV_TASK_PRIO_LOWEST, NULL); // Create top level global line separators. lv_obj_t *line = lv_cont_create(lv_layer_top(), NULL); static lv_style_t line_style; lv_style_copy(&line_style, &lv_style_plain_color); line_style.body.main_color = LV_COLOR_HEX3(0xDDD); // 0x505050 line_style.body.grad_color = line_style.body.main_color; line_style.body.shadow.width = 0; lv_cont_set_style(line, &line_style); lv_obj_set_size(line, LV_HOR_RES - LV_DPI * 3 / 5, 1); lv_obj_set_pos(line, LV_DPI * 3 / 10, 63); lv_obj_set_top(line, true); line = lv_cont_create(lv_layer_top(), line); lv_obj_set_pos(line, LV_DPI * 3 / 10, 656); lv_obj_set_top(line, true); // Option save button. lv_tabview_set_tab_load_action(tv, _show_hide_save_button); // If we rebooted to run sept for dumping, lunch dump immediately. if (nyx_str->cfg & NYX_CFG_DUMP) { nyx_str->cfg &= ~(NYX_CFG_DUMP); lv_task_t *task_run_dump = lv_task_create(sept_run_dump, 0, LV_TASK_PRIO_MID, NULL); lv_task_once(task_run_dump); } } #pragma GCC pop_options void nyx_load_and_run() { memset(&system_tasks, 0, sizeof(system_maintenance_tasks_t)); lv_init(); gfx_con.fillbg = 1; // Initialize framebuffer drawing functions. lv_disp_drv_t disp_drv; lv_disp_drv_init(&disp_drv); disp_drv.disp_flush = _disp_fb_flush; lv_disp_drv_register(&disp_drv); // Initialize touch. touch_power_on(); lv_indev_drv_t indev_drv; lv_indev_drv_init(&indev_drv); indev_drv.type = LV_INDEV_TYPE_POINTER; indev_drv.read = _fts_touch_read; lv_indev_drv_register(&indev_drv); touchpad.touch = false; //Set the theme. //! TODO: Finish theme support. lv_theme_t *th = lv_theme_hekate_init(167, NULL); lv_theme_set_current(th); _nyx_main_menu(th); while (true) lv_task_handler(); }