#include #include #include #include #include #include "../desktop_i.h" #include "desktop_view_debug.h" void desktop_debug_set_callback( DesktopDebugView* debug_view, DesktopDebugViewCallback callback, void* context) { furi_assert(debug_view); furi_assert(callback); debug_view->callback = callback; debug_view->context = context; } void desktop_debug_render(Canvas* canvas, void* model) { canvas_clear(canvas); DesktopDebugViewModel* m = model; const Version* ver; char buffer[64]; static const char* headers[] = {"Device Info:", "Dolphin Info:"}; canvas_set_color(canvas, ColorBlack); canvas_set_font(canvas, FontPrimary); canvas_draw_str_aligned( canvas, 64, 1 + STATUS_BAR_Y_SHIFT, AlignCenter, AlignTop, headers[m->screen]); canvas_set_font(canvas, FontSecondary); if(m->screen != DesktopViewStatsMeta) { // Hardware version const char* my_name = furi_hal_version_get_name_ptr(); snprintf( buffer, sizeof(buffer), "%d.F%dB%dC%d %s:%s %s", furi_hal_version_get_hw_version(), furi_hal_version_get_hw_target(), furi_hal_version_get_hw_body(), furi_hal_version_get_hw_connect(), furi_hal_version_get_hw_region_name(), furi_hal_region_get_name(), my_name ? my_name : "Unknown"); canvas_draw_str(canvas, 0, 19 + STATUS_BAR_Y_SHIFT, buffer); ver = furi_hal_version_get_firmware_version(); const BleGlueC2Info* c2_ver = NULL; #ifdef SRV_BT c2_ver = ble_glue_get_c2_info(); #endif if(!ver) { //-V1051 canvas_draw_str(canvas, 0, 30 + STATUS_BAR_Y_SHIFT, "No info"); return; } snprintf( buffer, sizeof(buffer), "%s [%s]", version_get_version(ver), version_get_builddate(ver)); canvas_draw_str(canvas, 0, 30 + STATUS_BAR_Y_SHIFT, buffer); uint16_t api_major, api_minor; furi_hal_info_get_api_version(&api_major, &api_minor); snprintf( buffer, sizeof(buffer), "%s%s [%d.%d] %s", version_get_dirty_flag(ver) ? "[!] " : "", version_get_githash(ver), api_major, api_minor, c2_ver ? c2_ver->StackTypeString : ""); canvas_draw_str(canvas, 0, 40 + STATUS_BAR_Y_SHIFT, buffer); snprintf( buffer, sizeof(buffer), "[%d] %s", version_get_target(ver), version_get_gitbranch(ver)); canvas_draw_str(canvas, 0, 50 + STATUS_BAR_Y_SHIFT, buffer); } else { Dolphin* dolphin = furi_record_open(RECORD_DOLPHIN); DolphinStats stats = dolphin_stats(dolphin); furi_record_close(RECORD_DOLPHIN); uint32_t current_lvl = stats.level; uint32_t remaining = dolphin_state_xp_to_levelup(m->icounter); canvas_set_font(canvas, FontSecondary); snprintf(buffer, sizeof(buffer), "Icounter: %lu Butthurt %lu", m->icounter, m->butthurt); canvas_draw_str(canvas, 5, 19 + STATUS_BAR_Y_SHIFT, buffer); snprintf( buffer, sizeof(buffer), "Level: %lu To level up: %lu", current_lvl, (remaining == (uint32_t)(-1) ? remaining : 0)); canvas_draw_str(canvas, 5, 29 + STATUS_BAR_Y_SHIFT, buffer); // even if timestamp is uint64_t, it's safe to cast it to uint32_t, because furi_hal_rtc_datetime_to_timestamp only returns uint32_t snprintf(buffer, sizeof(buffer), "%lu", (uint32_t)m->timestamp); canvas_draw_str(canvas, 5, 39 + STATUS_BAR_Y_SHIFT, buffer); canvas_draw_str(canvas, 0, 49 + STATUS_BAR_Y_SHIFT, "[< >] icounter value [ok] save"); } } View* desktop_debug_get_view(DesktopDebugView* debug_view) { furi_assert(debug_view); return debug_view->view; } bool desktop_debug_input(InputEvent* event, void* context) { furi_assert(event); furi_assert(context); DesktopDebugView* debug_view = context; if(event->type != InputTypeShort && event->type != InputTypeRepeat) { return false; } DesktopViewStatsScreens current = 0; with_view_model( debug_view->view, DesktopDebugViewModel * model, { #ifdef SRV_DOLPHIN_STATE_DEBUG if((event->key == InputKeyDown) || (event->key == InputKeyUp)) { model->screen = !model->screen; } #endif current = model->screen; }, true); size_t count = (event->type == InputTypeRepeat) ? 10 : 1; if(current == DesktopViewStatsMeta) { if(event->key == InputKeyLeft) { while(count-- > 0) { debug_view->callback(DesktopDebugEventWrongDeed, debug_view->context); } } else if(event->key == InputKeyRight) { while(count-- > 0) { debug_view->callback(DesktopDebugEventDeed, debug_view->context); } } else if(event->key == InputKeyOk) { debug_view->callback(DesktopDebugEventSaveState, debug_view->context); } else { return false; } } if(event->key == InputKeyBack) { debug_view->callback(DesktopDebugEventExit, debug_view->context); } return true; } DesktopDebugView* desktop_debug_alloc() { DesktopDebugView* debug_view = malloc(sizeof(DesktopDebugView)); debug_view->view = view_alloc(); view_allocate_model(debug_view->view, ViewModelTypeLocking, sizeof(DesktopDebugViewModel)); view_set_context(debug_view->view, debug_view); view_set_draw_callback(debug_view->view, (ViewDrawCallback)desktop_debug_render); view_set_input_callback(debug_view->view, desktop_debug_input); return debug_view; } void desktop_debug_free(DesktopDebugView* debug_view) { furi_assert(debug_view); view_free(debug_view->view); free(debug_view); } void desktop_debug_get_dolphin_data(DesktopDebugView* debug_view) { Dolphin* dolphin = furi_record_open(RECORD_DOLPHIN); DolphinStats stats = dolphin_stats(dolphin); with_view_model( debug_view->view, DesktopDebugViewModel * model, { model->icounter = stats.icounter; model->butthurt = stats.butthurt; model->timestamp = stats.timestamp; }, true); furi_record_close(RECORD_DOLPHIN); } void desktop_debug_reset_screen_idx(DesktopDebugView* debug_view) { with_view_model( debug_view->view, DesktopDebugViewModel * model, { model->screen = 0; }, true); }