input_event.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  1. #include "input_event.h"
  2. #include "diskop.h"
  3. #define AUDIO_MODES_COUNT 2
  4. void return_from_keyboard_callback(void *ctx)
  5. {
  6. FlizzerTrackerApp *tracker = (FlizzerTrackerApp *)ctx;
  7. if (!tracker->is_loading && !tracker->is_saving)
  8. {
  9. uint8_t string_length = 0;
  10. char *string = NULL;
  11. if (tracker->focus == EDIT_SONGINFO && tracker->mode == PATTERN_VIEW)
  12. {
  13. switch (tracker->selected_param)
  14. {
  15. case SI_SONGNAME:
  16. {
  17. string_length = MUS_SONG_NAME_LEN;
  18. string = (char *)&tracker->song.song_name;
  19. break;
  20. }
  21. case SI_INSTRUMENTNAME:
  22. {
  23. string_length = MUS_INST_NAME_LEN;
  24. string = (char *)&tracker->song.instrument[tracker->current_instrument]->name;
  25. break;
  26. }
  27. }
  28. }
  29. if (tracker->focus == EDIT_INSTRUMENT && tracker->mode == INST_EDITOR_VIEW)
  30. {
  31. switch (tracker->selected_param)
  32. {
  33. case INST_INSTRUMENTNAME:
  34. {
  35. string_length = MUS_INST_NAME_LEN;
  36. string = (char *)&tracker->song.instrument[tracker->current_instrument]->name;
  37. break;
  38. }
  39. }
  40. }
  41. if (string == NULL || string_length == 0)
  42. return;
  43. for (uint8_t i = 0; i < string_length; i++) // I tinyfied the font by deleting lowercase chars, and I don't like the lowercase chars of any 3x5 pixels font
  44. {
  45. string[i] = toupper(string[i]);
  46. }
  47. }
  48. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_TRACKER);
  49. if (tracker->is_saving)
  50. {
  51. stop_song(tracker);
  52. tracker->filepath = furi_string_alloc();
  53. furi_string_cat_printf(tracker->filepath, "%s/%s%s", FLIZZER_TRACKER_FOLDER, tracker->filename, SONG_FILE_EXT);
  54. if (storage_file_exists(tracker->storage, furi_string_get_cstr(tracker->filepath)))
  55. {
  56. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_FILE_OVERWRITE);
  57. return;
  58. }
  59. else
  60. {
  61. FlizzerTrackerEvent event = {.type = EventTypeSaveSong, .input = {0}, .period = 0};
  62. furi_message_queue_put(tracker->event_queue, &event, FuriWaitForever);
  63. }
  64. }
  65. }
  66. void overwrite_file_widget_yes_input_callback(GuiButtonType result, InputType type, void *ctx)
  67. {
  68. UNUSED(result);
  69. FlizzerTrackerApp *tracker = (FlizzerTrackerApp *)ctx;
  70. if (type == InputTypeShort)
  71. {
  72. tracker->is_saving = true;
  73. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_TRACKER);
  74. // save_song(tracker, tracker->filepath);
  75. static FlizzerTrackerEvent event = {.type = EventTypeSaveSong, .input = {0}, .period = 0};
  76. furi_message_queue_put(tracker->event_queue, &event, FuriWaitForever);
  77. }
  78. }
  79. void overwrite_file_widget_no_input_callback(GuiButtonType result, InputType type, void *ctx)
  80. {
  81. UNUSED(result);
  82. FlizzerTrackerApp *tracker = (FlizzerTrackerApp *)ctx;
  83. if (type == InputTypeShort)
  84. {
  85. tracker->is_saving = false;
  86. furi_string_free(tracker->filepath);
  87. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_TRACKER);
  88. }
  89. }
  90. uint32_t submenu_settings_exit_callback(void *context)
  91. {
  92. UNUSED(context);
  93. return VIEW_SUBMENU_PATTERN;
  94. }
  95. uint32_t submenu_exit_callback(void *context)
  96. {
  97. UNUSED(context);
  98. return VIEW_TRACKER;
  99. }
  100. void submenu_callback(void *context, uint32_t index)
  101. {
  102. FlizzerTrackerApp *tracker = (FlizzerTrackerApp *)context;
  103. switch (tracker->mode)
  104. {
  105. case PATTERN_VIEW:
  106. {
  107. switch (index)
  108. {
  109. case SUBMENU_PATTERN_EXIT:
  110. {
  111. tracker->quit = true;
  112. static InputEvent inevent = {.sequence = 0, .key = InputKeyLeft, .type = InputTypeMAX};
  113. FlizzerTrackerEvent event = {.type = EventTypeInput, .input = inevent, .period = 0}; // making an event so tracker does not wait for next keypress and exits immediately
  114. furi_message_queue_put(tracker->event_queue, &event, FuriWaitForever);
  115. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_TRACKER);
  116. break;
  117. }
  118. case SUBMENU_PATTERN_SAVE_SONG:
  119. {
  120. text_input_set_header_text(tracker->text_input, "Song filename:");
  121. memset(&tracker->filename, 0, FILE_NAME_LEN);
  122. text_input_set_result_callback(tracker->text_input, return_from_keyboard_callback, tracker, (char *)&tracker->filename, FILE_NAME_LEN, true);
  123. tracker->is_saving = true;
  124. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_KEYBOARD);
  125. break;
  126. }
  127. case SUBMENU_PATTERN_LOAD_SONG:
  128. {
  129. FlizzerTrackerEvent event = {.type = EventTypeLoadSong, .input = {0}, .period = 0};
  130. furi_message_queue_put(tracker->event_queue, &event, FuriWaitForever);
  131. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_TRACKER);
  132. break;
  133. }
  134. case SUBMENU_PATTERN_SETTINGS:
  135. {
  136. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_SETTINGS);
  137. break;
  138. }
  139. default:
  140. break;
  141. }
  142. break;
  143. }
  144. case INST_EDITOR_VIEW:
  145. {
  146. switch (index)
  147. {
  148. case SUBMENU_INSTRUMENT_EXIT:
  149. {
  150. tracker->quit = true;
  151. static InputEvent inevent = {.sequence = 0, .key = InputKeyLeft, .type = InputTypeMAX};
  152. FlizzerTrackerEvent event = {.type = EventTypeInput, .input = inevent, .period = 0}; // making an event so tracker does not wait for next keypress and exits immediately
  153. furi_message_queue_put(tracker->event_queue, &event, FuriWaitForever);
  154. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_TRACKER);
  155. break;
  156. }
  157. default:
  158. break;
  159. }
  160. break;
  161. }
  162. default:
  163. break;
  164. }
  165. }
  166. void audio_output_changed_callback(VariableItem *item)
  167. {
  168. FlizzerTrackerApp *tracker = (FlizzerTrackerApp *)variable_item_get_context(item);
  169. uint8_t index = variable_item_get_current_value_index(item);
  170. variable_item_set_current_value_text(item, audio_modes_text[(index > 1 ? 1 : index)]);
  171. if (tracker)
  172. {
  173. tracker->external_audio = (bool)index;
  174. tracker->external_audio = audio_modes_values[(index > 1 ? 1 : index)];
  175. // sound_engine_init(&tracker->sound_engine, tracker->sound_engine.sample_rate, tracker->external_audio, tracker->sound_engine.audio_buffer_size);
  176. // sound_engine_init_hardware(tracker->sound_engine.sample_rate, tracker->external_audio, tracker->sound_engine.audio_buffer, tracker->sound_engine.audio_buffer_size);
  177. FlizzerTrackerEvent event = {.type = EventTypeSetAudioMode, .input = {0}, .period = 0};
  178. furi_message_queue_put(tracker->event_queue, &event, FuriWaitForever);
  179. UNUSED(event);
  180. }
  181. }
  182. void cycle_focus(FlizzerTrackerApp *tracker)
  183. {
  184. switch (tracker->mode)
  185. {
  186. case PATTERN_VIEW:
  187. {
  188. tracker->focus++;
  189. if (tracker->focus > EDIT_SONGINFO)
  190. {
  191. tracker->focus = EDIT_PATTERN;
  192. }
  193. break;
  194. }
  195. case INST_EDITOR_VIEW:
  196. {
  197. tracker->focus++;
  198. if (tracker->focus > EDIT_PROGRAM)
  199. {
  200. tracker->focus = EDIT_INSTRUMENT;
  201. if (tracker->current_digit > 1)
  202. {
  203. tracker->current_digit = 1;
  204. }
  205. }
  206. break;
  207. }
  208. default:
  209. break;
  210. }
  211. }
  212. void cycle_view(FlizzerTrackerApp *tracker)
  213. {
  214. if (tracker->mode == PATTERN_VIEW)
  215. {
  216. tracker->mode = INST_EDITOR_VIEW;
  217. tracker->focus = EDIT_INSTRUMENT;
  218. tracker->selected_param = 0;
  219. tracker->current_digit = 0;
  220. return;
  221. }
  222. if (tracker->mode == INST_EDITOR_VIEW)
  223. {
  224. tracker->mode = PATTERN_VIEW;
  225. tracker->focus = EDIT_PATTERN;
  226. if (tracker->tracker_engine.song == NULL)
  227. {
  228. stop_song(tracker);
  229. tracker_engine_set_song(&tracker->tracker_engine, &tracker->song);
  230. }
  231. tracker->selected_param = 0;
  232. tracker->current_digit = 0;
  233. return;
  234. }
  235. }
  236. void process_input_event(FlizzerTrackerApp *tracker, FlizzerTrackerEvent *event)
  237. {
  238. if (event->input.key == InputKeyBack && event->input.type == InputTypeShort && event->period > 0 && event->period < 300 && !(tracker->editing))
  239. {
  240. cycle_view(tracker);
  241. return;
  242. }
  243. else if (event->input.key == InputKeyBack && event->input.type == InputTypeShort && !(tracker->editing))
  244. {
  245. cycle_focus(tracker);
  246. return;
  247. }
  248. // Если нажата кнопка "назад", то выходим из цикла, а следовательно и из приложения
  249. if (event->input.key == InputKeyBack && event->input.type == InputTypeLong)
  250. {
  251. switch (tracker->mode)
  252. {
  253. case PATTERN_VIEW:
  254. {
  255. submenu_set_selected_item(tracker->pattern_submenu, SUBMENU_PATTERN_LOAD_SONG);
  256. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_SUBMENU_PATTERN);
  257. break;
  258. }
  259. case INST_EDITOR_VIEW:
  260. {
  261. submenu_set_selected_item(tracker->instrument_submenu, SUBMENU_INSTRUMENT_EXIT);
  262. view_dispatcher_switch_to_view(tracker->view_dispatcher, VIEW_SUBMENU_INSTRUMENT);
  263. break;
  264. }
  265. default:
  266. break;
  267. }
  268. return;
  269. }
  270. switch (tracker->focus)
  271. {
  272. case EDIT_PATTERN:
  273. {
  274. pattern_edit_event(tracker, event);
  275. break;
  276. }
  277. case EDIT_SEQUENCE:
  278. {
  279. sequence_edit_event(tracker, event);
  280. break;
  281. }
  282. case EDIT_SONGINFO:
  283. {
  284. songinfo_edit_event(tracker, event);
  285. break;
  286. }
  287. case EDIT_INSTRUMENT:
  288. {
  289. instrument_edit_event(tracker, event);
  290. break;
  291. }
  292. case EDIT_PROGRAM:
  293. {
  294. instrument_program_edit_event(tracker, event);
  295. break;
  296. }
  297. default:
  298. break;
  299. }
  300. }