84410c83b5
* Desktop: cleanup headers * Get loader pubsub via record * [FL-2183] Dolphin refactoring 2022.01 * Restruct animations assets structure * Rename assets * Cleanup headers * Update Recording animation * Add BadBattery animation * Provide loader's pubsub via record * Fix load/unload animations * Scripts: add flipper format support, initial dolphin packager rework. Assets: internal and external dolphin. * Sync internal meta.txt and manifest.txt * Reorder, rename dolphin assets * Split essential generated assets * Add ReadMe for dolphin assets * Separate essential blocking animations * Scripts: full dolphin validation before packaging * Assets, Scripts: dolphin external resources packer * Github: update codeowners * Scripts: proper slots handling in dolphin animation meta * Scripts: correct frames enumeration and fix compiled assets. * [FL-2209] Add Dolphin Deeds points and many more * Remove excess frame_rate * Change dolphin assets directory * Scripts: add internal resource support to dolphin compiler * Scripts: add internal assets generation, renaming * Scripts: correct assert, renaming * Code cleanup, documentation, fixes * Update Levelup animations * Rename essential -> blocking * Fix Unlocked hint * Scripts: rewrite Templite compiller, replace regexps with token parser, split block types into code and variable blocks. Update dolphin templates. * Documentation: add key combos description and use information * Scripts: cleanup templit, more debug info and add dev comment Co-authored-by: あく <alleteam@gmail.com>
214 lines
8.2 KiB
C
214 lines
8.2 KiB
C
#include <furi.h>
|
|
#include <furi_hal.h>
|
|
#include <applications.h>
|
|
#include <assets_icons.h>
|
|
#include <loader/loader.h>
|
|
|
|
#include "desktop/desktop_i.h"
|
|
#include "desktop/views/desktop_main.h"
|
|
#include "desktop_scene.h"
|
|
#include "desktop_scene_i.h"
|
|
|
|
#define MAIN_VIEW_DEFAULT (0UL)
|
|
|
|
static void desktop_scene_main_app_started_callback(const void* message, void* context) {
|
|
furi_assert(context);
|
|
Desktop* desktop = context;
|
|
const LoaderEvent* event = message;
|
|
|
|
if(event->type == LoaderEventTypeApplicationStarted) {
|
|
view_dispatcher_send_custom_event(
|
|
desktop->view_dispatcher, DesktopMainEventBeforeAppStarted);
|
|
osSemaphoreAcquire(desktop->unload_animation_semaphore, osWaitForever);
|
|
} else if(event->type == LoaderEventTypeApplicationStopped) {
|
|
view_dispatcher_send_custom_event(
|
|
desktop->view_dispatcher, DesktopMainEventAfterAppFinished);
|
|
}
|
|
}
|
|
|
|
static void desktop_scene_main_new_idle_animation_callback(void* context) {
|
|
furi_assert(context);
|
|
Desktop* desktop = context;
|
|
view_dispatcher_send_custom_event(desktop->view_dispatcher, DesktopMainEventNewIdleAnimation);
|
|
}
|
|
|
|
static void desktop_scene_main_check_animation_callback(void* context) {
|
|
furi_assert(context);
|
|
Desktop* desktop = context;
|
|
view_dispatcher_send_custom_event(desktop->view_dispatcher, DesktopMainEventCheckAnimation);
|
|
}
|
|
|
|
static void desktop_scene_main_interact_animation_callback(void* context) {
|
|
furi_assert(context);
|
|
Desktop* desktop = context;
|
|
view_dispatcher_send_custom_event(desktop->view_dispatcher, DesktopMainEventInteractAnimation);
|
|
}
|
|
|
|
static void desktop_switch_to_app(Desktop* desktop, const FlipperApplication* flipper_app) {
|
|
furi_assert(desktop);
|
|
furi_assert(flipper_app);
|
|
furi_assert(flipper_app->app);
|
|
furi_assert(flipper_app->name);
|
|
|
|
if(furi_thread_get_state(desktop->scene_thread) != FuriThreadStateStopped) {
|
|
FURI_LOG_E("Desktop", "Thread is already running");
|
|
return;
|
|
}
|
|
|
|
furi_thread_set_name(desktop->scene_thread, flipper_app->name);
|
|
furi_thread_set_stack_size(desktop->scene_thread, flipper_app->stack_size);
|
|
furi_thread_set_callback(desktop->scene_thread, flipper_app->app);
|
|
|
|
furi_thread_start(desktop->scene_thread);
|
|
}
|
|
|
|
void desktop_scene_main_callback(DesktopEvent event, void* context) {
|
|
Desktop* desktop = (Desktop*)context;
|
|
view_dispatcher_send_custom_event(desktop->view_dispatcher, event);
|
|
}
|
|
|
|
void desktop_scene_main_on_enter(void* context) {
|
|
Desktop* desktop = (Desktop*)context;
|
|
DesktopMainView* main_view = desktop->main_view;
|
|
|
|
animation_manager_set_context(desktop->animation_manager, desktop);
|
|
animation_manager_set_new_idle_callback(
|
|
desktop->animation_manager, desktop_scene_main_new_idle_animation_callback);
|
|
animation_manager_set_check_callback(
|
|
desktop->animation_manager, desktop_scene_main_check_animation_callback);
|
|
animation_manager_set_interact_callback(
|
|
desktop->animation_manager, desktop_scene_main_interact_animation_callback);
|
|
desktop_locked_set_callback(desktop->locked_view, desktop_scene_main_callback, desktop);
|
|
|
|
furi_assert(osSemaphoreGetCount(desktop->unload_animation_semaphore) == 0);
|
|
Loader* loader = furi_record_open("loader");
|
|
desktop->app_start_stop_subscription = furi_pubsub_subscribe(
|
|
loader_get_pubsub(loader), desktop_scene_main_app_started_callback, desktop);
|
|
furi_record_close("loader");
|
|
|
|
desktop_main_set_callback(main_view, desktop_scene_main_callback, desktop);
|
|
|
|
DesktopMainSceneState state =
|
|
scene_manager_get_scene_state(desktop->scene_manager, DesktopSceneMain);
|
|
if(state == DesktopMainSceneStateLockedNoPin) {
|
|
desktop_locked_lock(desktop->locked_view);
|
|
view_port_enabled_set(desktop->lock_viewport, true);
|
|
} else if(state == DesktopMainSceneStateLockedWithPin) {
|
|
LOAD_DESKTOP_SETTINGS(&desktop->settings);
|
|
furi_assert(desktop->settings.pincode.length > 0);
|
|
desktop_locked_lock_pincode(desktop->locked_view, desktop->settings.pincode);
|
|
view_port_enabled_set(desktop->lock_viewport, true);
|
|
furi_hal_rtc_set_flag(FuriHalRtcFlagLock);
|
|
furi_hal_usb_disable();
|
|
} else {
|
|
furi_assert(state == DesktopMainSceneStateUnlocked);
|
|
view_port_enabled_set(desktop->lock_viewport, false);
|
|
}
|
|
|
|
view_dispatcher_switch_to_view(desktop->view_dispatcher, DesktopViewMain);
|
|
}
|
|
|
|
bool desktop_scene_main_on_event(void* context, SceneManagerEvent event) {
|
|
Desktop* desktop = (Desktop*)context;
|
|
bool consumed = false;
|
|
|
|
if(event.type == SceneManagerEventTypeCustom) {
|
|
switch(event.event) {
|
|
case DesktopMainEventOpenMenu:
|
|
loader_show_menu();
|
|
consumed = true;
|
|
break;
|
|
|
|
case DesktopMainEventOpenLockMenu:
|
|
scene_manager_next_scene(desktop->scene_manager, DesktopSceneLockMenu);
|
|
consumed = true;
|
|
break;
|
|
|
|
case DesktopMainEventOpenDebug:
|
|
scene_manager_next_scene(desktop->scene_manager, DesktopSceneDebug);
|
|
consumed = true;
|
|
break;
|
|
|
|
case DesktopMainEventOpenArchive:
|
|
#ifdef APP_ARCHIVE
|
|
desktop_switch_to_app(desktop, &FLIPPER_ARCHIVE);
|
|
#endif
|
|
consumed = true;
|
|
break;
|
|
|
|
case DesktopMainEventOpenFavorite:
|
|
LOAD_DESKTOP_SETTINGS(&desktop->settings);
|
|
if(desktop->settings.favorite < FLIPPER_APPS_COUNT) {
|
|
Loader* loader = furi_record_open("loader");
|
|
LoaderStatus status =
|
|
loader_start(loader, FLIPPER_APPS[desktop->settings.favorite].name, NULL);
|
|
furi_check(status == LoaderStatusOk);
|
|
furi_record_close("loader");
|
|
} else {
|
|
FURI_LOG_E("DesktopSrv", "Can't find favorite application");
|
|
}
|
|
consumed = true;
|
|
break;
|
|
|
|
case DesktopMainEventCheckAnimation:
|
|
animation_manager_check_blocking_process(desktop->animation_manager);
|
|
consumed = true;
|
|
break;
|
|
case DesktopMainEventNewIdleAnimation:
|
|
animation_manager_new_idle_process(desktop->animation_manager);
|
|
consumed = true;
|
|
break;
|
|
case DesktopMainEventInteractAnimation:
|
|
animation_manager_interact_process(desktop->animation_manager);
|
|
consumed = true;
|
|
break;
|
|
case DesktopMainEventBeforeAppStarted:
|
|
animation_manager_unload_and_stall_animation(desktop->animation_manager);
|
|
osSemaphoreRelease(desktop->unload_animation_semaphore);
|
|
consumed = true;
|
|
break;
|
|
case DesktopMainEventAfterAppFinished:
|
|
animation_manager_load_and_continue_animation(desktop->animation_manager);
|
|
consumed = true;
|
|
break;
|
|
case DesktopMainEventUnlocked:
|
|
consumed = true;
|
|
furi_hal_rtc_reset_flag(FuriHalRtcFlagLock);
|
|
furi_hal_usb_enable();
|
|
view_port_enabled_set(desktop->lock_viewport, false);
|
|
scene_manager_set_scene_state(
|
|
desktop->scene_manager, DesktopSceneMain, DesktopMainSceneStateUnlocked);
|
|
break;
|
|
case DesktopMainEventUpdate:
|
|
desktop_locked_update(desktop->locked_view);
|
|
consumed = true;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return consumed;
|
|
}
|
|
|
|
void desktop_scene_main_on_exit(void* context) {
|
|
Desktop* desktop = (Desktop*)context;
|
|
|
|
/**
|
|
* We're allowed to leave this scene only when any other app & loader
|
|
* is finished, that's why we can be sure there is no task waiting
|
|
* for start/stop semaphore
|
|
*/
|
|
Loader* loader = furi_record_open("loader");
|
|
furi_pubsub_unsubscribe(loader_get_pubsub(loader), desktop->app_start_stop_subscription);
|
|
furi_record_close("loader");
|
|
furi_assert(osSemaphoreGetCount(desktop->unload_animation_semaphore) == 0);
|
|
|
|
animation_manager_set_new_idle_callback(desktop->animation_manager, NULL);
|
|
animation_manager_set_check_callback(desktop->animation_manager, NULL);
|
|
animation_manager_set_interact_callback(desktop->animation_manager, NULL);
|
|
animation_manager_set_context(desktop->animation_manager, desktop);
|
|
scene_manager_set_scene_state(desktop->scene_manager, DesktopSceneMain, MAIN_VIEW_DEFAULT);
|
|
}
|