| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- #include <furi.h>
- #include <gui/gui.h>
- #include <assets_icons.h>
- #include <gui/view_dispatcher.h>
- #include <storage/storage.h>
- #include <gui/modules/loading.h>
- #include <dialogs/dialogs.h>
- #include <flipper_application/flipper_application.h>
- #include "elf_cpp/elf_hashtable.h"
- #include "fap_loader_app.h"
- #define TAG "fap_loader_app"
- struct FapLoader {
- FlipperApplication* app;
- Storage* storage;
- DialogsApp* dialogs;
- Gui* gui;
- FuriString* fap_path;
- ViewDispatcher* view_dispatcher;
- Loading* loading;
- };
- bool fap_loader_load_name_and_icon(
- FuriString* path,
- Storage* storage,
- uint8_t** icon_ptr,
- FuriString* item_name) {
- FlipperApplication* app = flipper_application_alloc(storage, &hashtable_api_interface);
- FlipperApplicationPreloadStatus preload_res =
- flipper_application_preload_manifest(app, furi_string_get_cstr(path));
- bool load_success = false;
- if(preload_res == FlipperApplicationPreloadStatusSuccess) {
- const FlipperApplicationManifest* manifest = flipper_application_get_manifest(app);
- if(manifest->has_icon) {
- memcpy(*icon_ptr, manifest->icon, FAP_MANIFEST_MAX_ICON_SIZE);
- }
- furi_string_set(item_name, manifest->name);
- load_success = true;
- } else {
- FURI_LOG_E(TAG, "FAP Loader failed to preload %s", furi_string_get_cstr(path));
- load_success = false;
- }
- flipper_application_free(app);
- return load_success;
- }
- static bool fap_loader_item_callback(
- FuriString* path,
- void* context,
- uint8_t** icon_ptr,
- FuriString* item_name) {
- FapLoader* fap_loader = context;
- furi_assert(fap_loader);
- return fap_loader_load_name_and_icon(path, fap_loader->storage, icon_ptr, item_name);
- }
- static bool fap_loader_run_selected_app(FapLoader* loader) {
- furi_assert(loader);
- FuriString* error_message;
- error_message = furi_string_alloc_set("unknown error");
- bool file_selected = false;
- bool show_error = true;
- do {
- file_selected = true;
- loader->app = flipper_application_alloc(loader->storage, &hashtable_api_interface);
- size_t start = furi_get_tick();
- FURI_LOG_I(TAG, "FAP Loader is loading %s", furi_string_get_cstr(loader->fap_path));
- FlipperApplicationPreloadStatus preload_res =
- flipper_application_preload(loader->app, furi_string_get_cstr(loader->fap_path));
- if(preload_res != FlipperApplicationPreloadStatusSuccess) {
- const char* err_msg = flipper_application_preload_status_to_string(preload_res);
- furi_string_printf(error_message, "Preload failed: %s", err_msg);
- FURI_LOG_E(
- TAG,
- "FAP Loader failed to preload %s: %s",
- furi_string_get_cstr(loader->fap_path),
- err_msg);
- break;
- }
- FURI_LOG_I(TAG, "FAP Loader is mapping");
- FlipperApplicationLoadStatus load_status = flipper_application_map_to_memory(loader->app);
- if(load_status != FlipperApplicationLoadStatusSuccess) {
- const char* err_msg = flipper_application_load_status_to_string(load_status);
- furi_string_printf(error_message, "Load failed: %s", err_msg);
- FURI_LOG_E(
- TAG,
- "FAP Loader failed to map to memory %s: %s",
- furi_string_get_cstr(loader->fap_path),
- err_msg);
- break;
- }
- FURI_LOG_I(TAG, "Loaded in %ums", (size_t)(furi_get_tick() - start));
- FURI_LOG_I(TAG, "FAP Loader is starting app");
- FuriThread* thread = flipper_application_spawn(loader->app, NULL);
- furi_thread_start(thread);
- furi_thread_join(thread);
- show_error = false;
- int ret = furi_thread_get_return_code(thread);
- FURI_LOG_I(TAG, "FAP app returned: %i", ret);
- } while(0);
- if(show_error) {
- DialogMessage* message = dialog_message_alloc();
- dialog_message_set_header(message, "Error", 64, 0, AlignCenter, AlignTop);
- dialog_message_set_buttons(message, NULL, NULL, NULL);
- FuriString* buffer;
- buffer = furi_string_alloc();
- furi_string_printf(buffer, "%s", furi_string_get_cstr(error_message));
- furi_string_replace(buffer, ":", "\n");
- dialog_message_set_text(
- message, furi_string_get_cstr(buffer), 64, 32, AlignCenter, AlignCenter);
- dialog_message_show(loader->dialogs, message);
- dialog_message_free(message);
- furi_string_free(buffer);
- }
- furi_string_free(error_message);
- if(file_selected) {
- flipper_application_free(loader->app);
- }
- return file_selected;
- }
- static bool fap_loader_select_app(FapLoader* loader) {
- const DialogsFileBrowserOptions browser_options = {
- .extension = ".fap",
- .skip_assets = true,
- .icon = &I_unknown_10px,
- .hide_ext = true,
- .item_loader_callback = fap_loader_item_callback,
- .item_loader_context = loader,
- };
- return dialog_file_browser_show(
- loader->dialogs, loader->fap_path, loader->fap_path, &browser_options);
- }
- static FapLoader* fap_loader_alloc(const char* path) {
- FapLoader* loader = malloc(sizeof(FapLoader)); //-V773
- loader->fap_path = furi_string_alloc_set(path);
- loader->storage = furi_record_open(RECORD_STORAGE);
- loader->dialogs = furi_record_open(RECORD_DIALOGS);
- loader->gui = furi_record_open(RECORD_GUI);
- loader->view_dispatcher = view_dispatcher_alloc();
- loader->loading = loading_alloc();
- view_dispatcher_attach_to_gui(
- loader->view_dispatcher, loader->gui, ViewDispatcherTypeFullscreen);
- view_dispatcher_add_view(loader->view_dispatcher, 0, loading_get_view(loader->loading));
- return loader;
- }
- static void fap_loader_free(FapLoader* loader) {
- view_dispatcher_remove_view(loader->view_dispatcher, 0);
- loading_free(loader->loading);
- view_dispatcher_free(loader->view_dispatcher);
- furi_string_free(loader->fap_path);
- furi_record_close(RECORD_GUI);
- furi_record_close(RECORD_DIALOGS);
- furi_record_close(RECORD_STORAGE);
- free(loader);
- }
- int32_t fap_loader_app(void* p) {
- FapLoader* loader;
- if(p) {
- loader = fap_loader_alloc((const char*)p);
- view_dispatcher_switch_to_view(loader->view_dispatcher, 0);
- fap_loader_run_selected_app(loader);
- } else {
- loader = fap_loader_alloc(EXT_PATH("apps"));
- while(fap_loader_select_app(loader)) {
- view_dispatcher_switch_to_view(loader->view_dispatcher, 0);
- fap_loader_run_selected_app(loader);
- };
- }
- fap_loader_free(loader);
- return 0;
- }
|