scene_playlist_edit.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. #include "../subghz_playlist_creator.h"
  2. #include "scene_playlist_edit.h"
  3. #include "scene_menu.h"
  4. #include "scene_file_browser.h"
  5. #include "scene_popup.h"
  6. #include "scene_text_input.h"
  7. #include <furi.h>
  8. #include <gui/modules/file_browser.h>
  9. #include <gui/modules/widget.h>
  10. #include <storage/storage.h>
  11. #include <string.h>
  12. #include <gui/modules/dialog_ex.h>
  13. #include <gui/view.h>
  14. #include "scene_dialog.h"
  15. #include <gui/modules/submenu.h>
  16. #include <furi_hal.h>
  17. #define MAX_PLAYLIST_LINES 128
  18. #define MAX_FILENAME_LENGTH 128
  19. #define SUBGHZ_DIRECTORY "/ext/subghz"
  20. #define TAG "PlaylistEditScene"
  21. // Dialog type for PlaylistEdit
  22. typedef enum {
  23. PlaylistEditDialog_None = 0,
  24. PlaylistEditDialog_Discard,
  25. PlaylistEditDialog_Save,
  26. PlaylistEditDialog_Delete,
  27. } PlaylistEditDialogType;
  28. static PlaylistEditDialogType playlist_edit_dialog_type = PlaylistEditDialog_None;
  29. static uint32_t playlist_edit_selected_index = 0;
  30. // Add a static flag to indicate we should show menu after popup
  31. static bool playlist_edit_show_menu_after_popup = false;
  32. // Helper to read a line from file (since storage_file_read_line is not available)
  33. // static bool file_read_line(File* file, char* buffer, size_t max_len) { ... }
  34. // Dialog callback
  35. static void edit_dialog_callback(DialogExResult result, void* context) {
  36. SubGhzPlaylistCreator* app = context;
  37. FURI_LOG_D(TAG, "edit_dialog_callback: dialog_type=%d, result=%d", (int)playlist_edit_dialog_type, (int)result);
  38. if(playlist_edit_dialog_type == PlaylistEditDialog_Discard) {
  39. if(result == DialogExResultLeft) {
  40. FURI_LOG_D(TAG, "Discard: DialogExResultLeft -> menu");
  41. app->playlist_modified = false;
  42. scene_menu_show(app);
  43. } else if(result == DialogExResultRight) {
  44. FURI_LOG_D(TAG, "Discard: DialogExResultRight -> stay in edit");
  45. scene_playlist_edit_show(app);
  46. } else {
  47. FURI_LOG_D(TAG, "Discard: Unknown result -> stay in edit");
  48. scene_playlist_edit_show(app);
  49. }
  50. } else if(playlist_edit_dialog_type == PlaylistEditDialog_Save) {
  51. if(result == DialogExResultRight) {
  52. FURI_LOG_D(TAG, "Save: DialogExResultRight -> save and menu");
  53. File* file = storage_file_alloc(app->storage);
  54. if(file && storage_file_open(file, furi_string_get_cstr(app->playlist_path), FSAM_WRITE, FSOM_CREATE_ALWAYS)) {
  55. for(size_t i = 0; i < app->playlist_entry_count; ++i) {
  56. storage_file_write(file, "sub: ", 5);
  57. storage_file_write(file, app->playlist_entries[i], strlen(app->playlist_entries[i]));
  58. storage_file_write(file, "\n", 1);
  59. }
  60. storage_file_close(file);
  61. }
  62. if(file) storage_file_free(file);
  63. app->playlist_modified = false;
  64. for(size_t i = 0; i < app->playlist_entry_count; ++i) {
  65. free(app->playlist_entries[i]);
  66. }
  67. free(app->playlist_entries);
  68. app->playlist_entries = NULL;
  69. app->playlist_entry_count = 0;
  70. app->playlist_entry_capacity = 0;
  71. playlist_edit_show_menu_after_popup = true;
  72. scene_popup_show(app, "Success", "Playlist saved!");
  73. return;
  74. } else if(result == DialogExResultLeft) {
  75. FURI_LOG_D(TAG, "Save: DialogExResultLeft -> stay in edit");
  76. scene_playlist_edit_show(app);
  77. } else {
  78. FURI_LOG_D(TAG, "Save: Unknown result -> stay in edit");
  79. scene_playlist_edit_show(app);
  80. }
  81. } else if(playlist_edit_dialog_type == PlaylistEditDialog_Delete) {
  82. if(result == DialogExResultRight) {
  83. FURI_LOG_D(TAG, "Delete: DialogExResultRight -> remove entry");
  84. if(playlist_edit_selected_index < app->playlist_entry_count) {
  85. free(app->playlist_entries[playlist_edit_selected_index]);
  86. for(size_t i = playlist_edit_selected_index; i + 1 < app->playlist_entry_count; ++i) {
  87. app->playlist_entries[i] = app->playlist_entries[i + 1];
  88. }
  89. app->playlist_entry_count--;
  90. app->playlist_modified = true;
  91. if(playlist_edit_selected_index >= app->playlist_entry_count) playlist_edit_selected_index = app->playlist_entry_count ? app->playlist_entry_count - 1 : 0;
  92. }
  93. scene_playlist_edit_show(app);
  94. } else {
  95. FURI_LOG_D(TAG, "Delete: Not right -> stay in edit");
  96. scene_playlist_edit_show(app);
  97. }
  98. }
  99. playlist_edit_dialog_type = PlaylistEditDialog_None;
  100. }
  101. // Move the full definition of on_add_file_selected here:
  102. static void on_add_file_selected(SubGhzPlaylistCreator* app, const char* path) {
  103. if(path && strlen(path)) {
  104. if(app->playlist_entry_count == app->playlist_entry_capacity) {
  105. app->playlist_entry_capacity = app->playlist_entry_capacity ? app->playlist_entry_capacity * 2 : 8;
  106. app->playlist_entries = realloc(app->playlist_entries, app->playlist_entry_capacity * sizeof(char*));
  107. }
  108. app->playlist_entries[app->playlist_entry_count++] = strdup(path);
  109. app->playlist_modified = true;
  110. }
  111. scene_playlist_edit_show(app);
  112. }
  113. // Move the full definition of playlist_edit_submenu_callback here:
  114. static void playlist_edit_submenu_callback(void* context, uint32_t index) {
  115. SubGhzPlaylistCreator* app = context;
  116. FURI_LOG_D(TAG, "playlist_edit_submenu_callback: index=%u, entry_count=%u", (unsigned int)index, (unsigned int)app->playlist_entry_count);
  117. if(index == app->playlist_entry_count) {
  118. FURI_LOG_D(TAG, "[+] Add file selected");
  119. app->return_scene = ReturnScene_PlaylistEdit;
  120. scene_file_browser_select(app, SUBGHZ_DIRECTORY, ".sub", on_add_file_selected);
  121. } else if(index == app->playlist_entry_count + 1) {
  122. FURI_LOG_D(TAG, "Save playlist selected, showing dialog");
  123. scene_dialog_show_custom(
  124. app,
  125. "Save playlist?",
  126. "Are you sure you want to save playlist?",
  127. "Cancel",
  128. "Save",
  129. edit_dialog_callback,
  130. app
  131. );
  132. playlist_edit_dialog_type = PlaylistEditDialog_Save;
  133. } else {
  134. FURI_LOG_D(TAG, "Entry selected for delete, showing dialog");
  135. scene_dialog_show_custom(
  136. app,
  137. "Delete entry?",
  138. "Remove file from playlist?",
  139. "Cancel",
  140. "Delete",
  141. edit_dialog_callback,
  142. app
  143. );
  144. playlist_edit_dialog_type = PlaylistEditDialog_Delete;
  145. }
  146. }
  147. // Show PlaylistEdit scene using SubMenu
  148. void scene_playlist_edit_show(SubGhzPlaylistCreator* app) {
  149. if(playlist_edit_show_menu_after_popup) {
  150. playlist_edit_show_menu_after_popup = false;
  151. scene_menu_show(app);
  152. return;
  153. }
  154. app->current_view = SubGhzPlaylistCreatorViewPlaylistEdit;
  155. submenu_reset(app->playlist_edit_submenu);
  156. // Set header to playlist name with extension
  157. submenu_set_header(app->playlist_edit_submenu, furi_string_get_cstr(app->playlist_name));
  158. // Add playlist entries
  159. for(size_t i = 0; i < app->playlist_entry_count; ++i) {
  160. const char* fname = strrchr(app->playlist_entries[i], '/');
  161. fname = fname ? fname + 1 : app->playlist_entries[i];
  162. submenu_add_item(app->playlist_edit_submenu, fname, i, playlist_edit_submenu_callback, app);
  163. }
  164. // Add '[+] Add file' as next-to-last item
  165. submenu_add_item(app->playlist_edit_submenu, "[+] Add file", app->playlist_entry_count, playlist_edit_submenu_callback, app);
  166. // Add 'Save playlist' as last item
  167. submenu_add_item(app->playlist_edit_submenu, "[s] Save playlist", app->playlist_entry_count + 1, playlist_edit_submenu_callback, app);
  168. view_dispatcher_switch_to_view(app->view_dispatcher, SubGhzPlaylistCreatorViewPlaylistEdit);
  169. }
  170. void scene_playlist_edit_init_view(SubGhzPlaylistCreator* app) {
  171. // No need to allocate VariableItemList, use SubMenu
  172. // SubMenu is already allocated in app->playlist_edit_submenu
  173. // Just ensure view is added for PlaylistEdit (done in app alloc)
  174. UNUSED(app);
  175. }
  176. // Custom back event handler for PlaylistEdit
  177. bool scene_playlist_edit_back_event_callback(void* context) {
  178. SubGhzPlaylistCreator* app = context;
  179. // Show discard dialog instead of going to main menu
  180. scene_dialog_show_custom(
  181. app,
  182. "Discard changes?",
  183. "Are you sure you want to discard playlist?",
  184. "Discard",
  185. "Keep",
  186. edit_dialog_callback,
  187. app
  188. );
  189. playlist_edit_dialog_type = PlaylistEditDialog_Discard;
  190. return false; // Prevent view dispatcher from popping the view
  191. }