archive_scene_browser.c 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. #include "../archive_i.h"
  2. #include "../helpers/archive_files.h"
  3. #include "../helpers/archive_apps.h"
  4. #include "../helpers/archive_favorites.h"
  5. #include "../helpers/archive_browser.h"
  6. #include "../views/archive_browser_view.h"
  7. #include "archive/scenes/archive_scene.h"
  8. #define TAG "ArchiveSceneBrowser"
  9. #define SCENE_STATE_DEFAULT (0)
  10. #define SCENE_STATE_NEED_REFRESH (1)
  11. static const char* flipper_app_name[] = {
  12. [ArchiveFileTypeIButton] = "iButton",
  13. [ArchiveFileTypeNFC] = "NFC",
  14. [ArchiveFileTypeSubGhz] = "Sub-GHz",
  15. [ArchiveFileTypeLFRFID] = "125 kHz RFID",
  16. [ArchiveFileTypeInfrared] = "Infrared",
  17. [ArchiveFileTypeBadUsb] = "Bad USB",
  18. [ArchiveFileTypeU2f] = "U2F",
  19. [ArchiveFileTypeUpdateManifest] = "UpdaterApp",
  20. };
  21. static void archive_loader_callback(const void* message, void* context) {
  22. furi_assert(message);
  23. furi_assert(context);
  24. const LoaderEvent* event = message;
  25. ArchiveApp* archive = (ArchiveApp*)context;
  26. if(event->type == LoaderEventTypeApplicationStopped) {
  27. view_dispatcher_send_custom_event(
  28. archive->view_dispatcher, ArchiveBrowserEventListRefresh);
  29. }
  30. }
  31. static void archive_run_in_app(ArchiveBrowserView* browser, ArchiveFile_t* selected) {
  32. UNUSED(browser);
  33. Loader* loader = furi_record_open(RECORD_LOADER);
  34. LoaderStatus status;
  35. if(selected->is_app) {
  36. char* param = strrchr(string_get_cstr(selected->path), '/');
  37. if(param != NULL) {
  38. param++;
  39. }
  40. status = loader_start(loader, flipper_app_name[selected->type], param);
  41. } else {
  42. status = loader_start(
  43. loader, flipper_app_name[selected->type], string_get_cstr(selected->path));
  44. }
  45. if(status != LoaderStatusOk) {
  46. FURI_LOG_E(TAG, "loader_start failed: %d", status);
  47. }
  48. furi_record_close(RECORD_LOADER);
  49. }
  50. void archive_scene_browser_callback(ArchiveBrowserEvent event, void* context) {
  51. ArchiveApp* archive = (ArchiveApp*)context;
  52. view_dispatcher_send_custom_event(archive->view_dispatcher, event);
  53. }
  54. void archive_scene_browser_on_enter(void* context) {
  55. ArchiveApp* archive = (ArchiveApp*)context;
  56. ArchiveBrowserView* browser = archive->browser;
  57. browser->is_root = true;
  58. archive_browser_set_callback(browser, archive_scene_browser_callback, archive);
  59. archive_update_focus(browser, archive->text_store);
  60. view_dispatcher_switch_to_view(archive->view_dispatcher, ArchiveViewBrowser);
  61. Loader* loader = furi_record_open(RECORD_LOADER);
  62. archive->loader_stop_subscription =
  63. furi_pubsub_subscribe(loader_get_pubsub(loader), archive_loader_callback, archive);
  64. furi_record_close(RECORD_LOADER);
  65. uint32_t state = scene_manager_get_scene_state(archive->scene_manager, ArchiveAppSceneBrowser);
  66. if(state == SCENE_STATE_NEED_REFRESH) {
  67. view_dispatcher_send_custom_event(
  68. archive->view_dispatcher, ArchiveBrowserEventListRefresh);
  69. }
  70. scene_manager_set_scene_state(
  71. archive->scene_manager, ArchiveAppSceneBrowser, SCENE_STATE_DEFAULT);
  72. }
  73. bool archive_scene_browser_on_event(void* context, SceneManagerEvent event) {
  74. ArchiveApp* archive = (ArchiveApp*)context;
  75. ArchiveBrowserView* browser = archive->browser;
  76. ArchiveFile_t* selected = archive_get_current_file(browser);
  77. bool favorites = archive_get_tab(browser) == ArchiveTabFavorites;
  78. bool consumed = false;
  79. if(event.type == SceneManagerEventTypeCustom) {
  80. switch(event.event) {
  81. case ArchiveBrowserEventFileMenuOpen:
  82. archive_show_file_menu(browser, true);
  83. consumed = true;
  84. break;
  85. case ArchiveBrowserEventFileMenuClose:
  86. archive_show_file_menu(browser, false);
  87. consumed = true;
  88. break;
  89. case ArchiveBrowserEventFileMenuRun:
  90. if(archive_is_known_app(selected->type)) {
  91. archive_run_in_app(browser, selected);
  92. archive_show_file_menu(browser, false);
  93. }
  94. consumed = true;
  95. break;
  96. case ArchiveBrowserEventFileMenuPin: {
  97. const char* name = archive_get_name(browser);
  98. if(favorites) {
  99. archive_favorites_delete(name);
  100. archive_file_array_rm_selected(browser);
  101. archive_show_file_menu(browser, false);
  102. } else if(archive_is_known_app(selected->type)) {
  103. if(archive_is_favorite("%s", name)) {
  104. archive_favorites_delete("%s", name);
  105. } else {
  106. archive_file_append(ARCHIVE_FAV_PATH, "%s\n", name);
  107. }
  108. archive_show_file_menu(browser, false);
  109. }
  110. consumed = true;
  111. } break;
  112. case ArchiveBrowserEventFileMenuRename:
  113. if(favorites) {
  114. browser->callback(ArchiveBrowserEventEnterFavMove, browser->context);
  115. } else if((archive_is_known_app(selected->type)) && (selected->is_app == false)) {
  116. archive_show_file_menu(browser, false);
  117. scene_manager_set_scene_state(
  118. archive->scene_manager, ArchiveAppSceneBrowser, SCENE_STATE_NEED_REFRESH);
  119. scene_manager_next_scene(archive->scene_manager, ArchiveAppSceneRename);
  120. }
  121. consumed = true;
  122. break;
  123. case ArchiveBrowserEventFileMenuDelete:
  124. if(archive_get_tab(browser) != ArchiveTabFavorites) {
  125. scene_manager_next_scene(archive->scene_manager, ArchiveAppSceneDelete);
  126. }
  127. consumed = true;
  128. break;
  129. case ArchiveBrowserEventEnterDir:
  130. archive_enter_dir(browser, selected->path);
  131. consumed = true;
  132. break;
  133. case ArchiveBrowserEventFavMoveUp:
  134. archive_file_array_swap(browser, 1);
  135. consumed = true;
  136. break;
  137. case ArchiveBrowserEventFavMoveDown:
  138. archive_file_array_swap(browser, -1);
  139. consumed = true;
  140. break;
  141. case ArchiveBrowserEventEnterFavMove:
  142. string_set(archive->fav_move_str, selected->path);
  143. archive_show_file_menu(browser, false);
  144. archive_favorites_move_mode(archive->browser, true);
  145. consumed = true;
  146. break;
  147. case ArchiveBrowserEventExitFavMove:
  148. archive_update_focus(browser, string_get_cstr(archive->fav_move_str));
  149. archive_favorites_move_mode(archive->browser, false);
  150. consumed = true;
  151. break;
  152. case ArchiveBrowserEventSaveFavMove:
  153. archive_favorites_move_mode(archive->browser, false);
  154. archive_favorites_save(archive->browser);
  155. consumed = true;
  156. break;
  157. case ArchiveBrowserEventLoadPrevItems:
  158. archive_file_array_load(archive->browser, -1);
  159. consumed = true;
  160. break;
  161. case ArchiveBrowserEventLoadNextItems:
  162. archive_file_array_load(archive->browser, 1);
  163. consumed = true;
  164. break;
  165. case ArchiveBrowserEventListRefresh:
  166. if(!favorites) {
  167. archive_refresh_dir(browser);
  168. } else {
  169. archive_favorites_read(browser);
  170. }
  171. consumed = true;
  172. break;
  173. case ArchiveBrowserEventExit:
  174. if(!archive_is_home(browser)) {
  175. archive_leave_dir(browser);
  176. } else {
  177. Loader* loader = furi_record_open(RECORD_LOADER);
  178. furi_pubsub_unsubscribe(
  179. loader_get_pubsub(loader), archive->loader_stop_subscription);
  180. furi_record_close(RECORD_LOADER);
  181. view_dispatcher_stop(archive->view_dispatcher);
  182. }
  183. consumed = true;
  184. break;
  185. default:
  186. break;
  187. }
  188. }
  189. return consumed;
  190. }
  191. void archive_scene_browser_on_exit(void* context) {
  192. ArchiveApp* archive = (ArchiveApp*)context;
  193. Loader* loader = furi_record_open(RECORD_LOADER);
  194. furi_pubsub_unsubscribe(loader_get_pubsub(loader), archive->loader_stop_subscription);
  195. furi_record_close(RECORD_LOADER);
  196. }