init_deinit.c 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. #include "init_deinit.h"
  2. #include "input_event.h"
  3. #include "diskop.h"
  4. #define AUDIO_MODES_COUNT 2
  5. TrackerView* tracker_view_alloc(FlizzerTrackerApp* tracker) {
  6. TrackerView* tracker_view = malloc(sizeof(TrackerView));
  7. tracker_view->view = view_alloc();
  8. tracker_view->context = tracker;
  9. view_set_context(tracker_view->view, tracker_view);
  10. view_allocate_model(tracker_view->view, ViewModelTypeLocking, sizeof(TrackerViewModel));
  11. view_set_draw_callback(tracker_view->view, draw_callback);
  12. view_set_input_callback(tracker_view->view, input_callback);
  13. return tracker_view;
  14. }
  15. void tracker_view_free(TrackerView* tracker_view) {
  16. furi_assert(tracker_view);
  17. view_free(tracker_view->view);
  18. free(tracker_view);
  19. }
  20. uint8_t my_value_index_bool(
  21. const bool value,
  22. const bool values[],
  23. uint8_t
  24. values_count) // why the fuck it gives unresolved symbol if I include it from toolbox???!!!
  25. {
  26. uint8_t index = 0;
  27. for(uint8_t i = 0; i < values_count; i++) {
  28. if(value == values[i]) {
  29. index = i;
  30. break;
  31. }
  32. }
  33. return index;
  34. }
  35. FlizzerTrackerApp* init_tracker(
  36. uint32_t sample_rate,
  37. uint8_t rate,
  38. bool external_audio_output,
  39. uint32_t audio_buffer_size) {
  40. FlizzerTrackerApp* tracker = malloc(sizeof(FlizzerTrackerApp));
  41. memset(tracker, 0, sizeof(FlizzerTrackerApp));
  42. tracker->external_audio = external_audio_output;
  43. sound_engine_init(
  44. &tracker->sound_engine, sample_rate, external_audio_output, audio_buffer_size);
  45. tracker_engine_init(&tracker->tracker_engine, rate, &tracker->sound_engine);
  46. tracker->tracker_engine.song = &tracker->song;
  47. tracker->current_note = MIDDLE_C;
  48. // Очередь событий на 8 элементов размера FlizzerTrackerEvent
  49. tracker->event_queue = furi_message_queue_alloc(8, sizeof(FlizzerTrackerEvent));
  50. tracker->gui = furi_record_open(RECORD_GUI);
  51. tracker->view_dispatcher = view_dispatcher_alloc();
  52. tracker->tracker_view = tracker_view_alloc(tracker);
  53. view_dispatcher_add_view(tracker->view_dispatcher, VIEW_TRACKER, tracker->tracker_view->view);
  54. view_dispatcher_attach_to_gui(
  55. tracker->view_dispatcher, tracker->gui, ViewDispatcherTypeFullscreen);
  56. with_view_model(
  57. tracker->tracker_view->view, TrackerViewModel * model, { model->tracker = tracker; }, true);
  58. tracker->storage = furi_record_open(RECORD_STORAGE);
  59. tracker->stream = file_stream_alloc(tracker->storage);
  60. tracker->text_input = text_input_alloc();
  61. view_dispatcher_add_view(
  62. tracker->view_dispatcher, VIEW_KEYBOARD, text_input_get_view(tracker->text_input));
  63. tracker->pattern_submenu = submenu_alloc();
  64. tracker->instrument_submenu = submenu_alloc();
  65. view_set_previous_callback(submenu_get_view(tracker->pattern_submenu), submenu_exit_callback);
  66. view_set_previous_callback(
  67. submenu_get_view(tracker->instrument_submenu), submenu_exit_callback);
  68. submenu_add_item(
  69. tracker->pattern_submenu,
  70. "Load song",
  71. SUBMENU_PATTERN_LOAD_SONG,
  72. submenu_callback,
  73. tracker);
  74. submenu_add_item(
  75. tracker->pattern_submenu,
  76. "Save song",
  77. SUBMENU_PATTERN_SAVE_SONG,
  78. submenu_callback,
  79. tracker);
  80. submenu_add_item(
  81. tracker->pattern_submenu, "Settings", SUBMENU_PATTERN_SETTINGS, submenu_callback, tracker);
  82. submenu_add_item(
  83. tracker->pattern_submenu, "Exit", SUBMENU_PATTERN_EXIT, submenu_callback, tracker);
  84. submenu_add_item(
  85. tracker->instrument_submenu, "Exit", SUBMENU_INSTRUMENT_EXIT, submenu_callback, tracker);
  86. view_dispatcher_add_view(
  87. tracker->view_dispatcher,
  88. VIEW_SUBMENU_PATTERN,
  89. submenu_get_view(tracker->pattern_submenu));
  90. view_dispatcher_add_view(
  91. tracker->view_dispatcher,
  92. VIEW_SUBMENU_INSTRUMENT,
  93. submenu_get_view(tracker->instrument_submenu));
  94. load_config(tracker);
  95. tracker->settings_list = variable_item_list_alloc();
  96. View* view = variable_item_list_get_view(tracker->settings_list);
  97. view_set_previous_callback(view, submenu_settings_exit_callback);
  98. VariableItem* item;
  99. uint8_t value_index;
  100. item = variable_item_list_add(
  101. tracker->settings_list,
  102. "Audio output",
  103. AUDIO_MODES_COUNT,
  104. audio_output_changed_callback,
  105. tracker);
  106. value_index =
  107. my_value_index_bool(tracker->external_audio, audio_modes_values, AUDIO_MODES_COUNT);
  108. variable_item_set_current_value_index(item, value_index);
  109. variable_item_set_current_value_text(item, audio_modes_text[value_index]);
  110. view_dispatcher_add_view(tracker->view_dispatcher, VIEW_SETTINGS, view);
  111. tracker->overwrite_file_widget = widget_alloc();
  112. widget_add_button_element(
  113. tracker->overwrite_file_widget,
  114. GuiButtonTypeLeft,
  115. "No",
  116. (ButtonCallback)overwrite_file_widget_no_input_callback,
  117. tracker);
  118. widget_add_button_element(
  119. tracker->overwrite_file_widget,
  120. GuiButtonTypeRight,
  121. "Yes",
  122. (ButtonCallback)overwrite_file_widget_yes_input_callback,
  123. tracker);
  124. widget_add_text_scroll_element(
  125. tracker->overwrite_file_widget,
  126. 0,
  127. 0,
  128. 128,
  129. 64,
  130. "This song file already exists,\n do you want to overwrite it?");
  131. view_dispatcher_add_view(
  132. tracker->view_dispatcher,
  133. VIEW_FILE_OVERWRITE,
  134. widget_get_view(tracker->overwrite_file_widget));
  135. tracker->notification = furi_record_open(RECORD_NOTIFICATION);
  136. notification_message(tracker->notification, &sequence_display_backlight_enforce_on);
  137. set_default_song(tracker);
  138. return tracker;
  139. }
  140. void deinit_tracker(FlizzerTrackerApp* tracker) {
  141. notification_message(tracker->notification, &sequence_display_backlight_enforce_auto);
  142. furi_record_close(RECORD_NOTIFICATION);
  143. // Специальная очистка памяти, занимаемой очередью
  144. furi_message_queue_free(tracker->event_queue);
  145. view_dispatcher_remove_view(tracker->view_dispatcher, VIEW_SETTINGS);
  146. view_dispatcher_remove_view(tracker->view_dispatcher, VIEW_FILE_OVERWRITE);
  147. view_dispatcher_remove_view(tracker->view_dispatcher, VIEW_SUBMENU_INSTRUMENT);
  148. view_dispatcher_remove_view(tracker->view_dispatcher, VIEW_SUBMENU_PATTERN);
  149. view_dispatcher_remove_view(tracker->view_dispatcher, VIEW_KEYBOARD);
  150. view_dispatcher_remove_view(tracker->view_dispatcher, VIEW_TRACKER);
  151. text_input_free(tracker->text_input);
  152. variable_item_list_free(tracker->settings_list);
  153. submenu_free(tracker->pattern_submenu);
  154. submenu_free(tracker->instrument_submenu);
  155. widget_free(tracker->overwrite_file_widget);
  156. view_dispatcher_free(tracker->view_dispatcher);
  157. tracker_view_free(tracker->tracker_view);
  158. furi_record_close(RECORD_GUI);
  159. stream_free(tracker->stream);
  160. furi_record_close(RECORD_STORAGE);
  161. sound_engine_deinit(&tracker->sound_engine);
  162. if(tracker->tracker_engine.song == NULL) {
  163. tracker_engine_set_song(&tracker->tracker_engine, &tracker->song);
  164. }
  165. tracker_engine_deinit(&tracker->tracker_engine, false);
  166. FURI_CRITICAL_ENTER();
  167. LL_TIM_DeInit(TRACKER_ENGINE_TIMER);
  168. LL_TIM_DeInit(SAMPLE_RATE_TIMER);
  169. LL_TIM_DeInit(SPEAKER_PWM_TIMER);
  170. FURI_CRITICAL_EXIT();
  171. free(tracker);
  172. }