flizzer_tracker.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. #include "flizzer_tracker.h"
  2. #include "init_deinit.h"
  3. #include "event.h"
  4. #include "view/pattern_editor.h"
  5. #define FLIZZER_TRACKER_FOLDER "/ext/flizzer_tracker"
  6. #include <flizzer_tracker_icons.h>
  7. /*
  8. Fontname: -Raccoon-Fixed4x6-Medium-R-Normal--6-60-75-75-P-40-ISO10646-1
  9. Copyright:
  10. Glyphs: 95/203
  11. BBX Build Mode: 0
  12. */
  13. const uint8_t u8g2_font_tom_thumb_4x6_tr[725] U8G2_FONT_SECTION("u8g2_font_tom_thumb_4x6_tr") =
  14. "_\0\2\2\2\3\3\4\4\3\6\0\377\5\377\5\0\0\352\1\330\2\270 \5\340\315\0!\6\265\310"
  15. "\254\0\42\6\213\313$\25#\10\227\310\244\241\206\12$\10\227\310\215\70b\2%\10\227\310d\324F\1"
  16. "&\10\227\310(\65R\22'\5\251\313\10(\6\266\310\251\62)\10\226\310\304\224\24\0*\6\217\312\244"
  17. "\16+\7\217\311\245\225\0,\6\212\310)\0-\5\207\312\14.\5\245\310\4/\7\227\310Ve\4\60"
  18. "\7\227\310-k\1\61\6\226\310\255\6\62\10\227\310h\220\312\1\63\11\227\310h\220\62X\0\64\10\227"
  19. "\310$\65b\1\65\10\227\310\214\250\301\2\66\10\227\310\315\221F\0\67\10\227\310\314TF\0\70\10\227"
  20. "\310\214\64\324\10\71\10\227\310\214\64\342\2:\6\255\311\244\0;\7\222\310e\240\0<\10\227\310\246\32"
  21. "d\20=\6\217\311l\60>\11\227\310d\220A*\1\77\10\227\310\314\224a\2@\10\227\310UC\3"
  22. "\1A\10\227\310UC\251\0B\10\227\310\250\264\322\2C\7\227\310\315\32\10D\10\227\310\250d-\0"
  23. "E\10\227\310\214\70\342\0F\10\227\310\214\70b\4G\10\227\310\315\221\222\0H\10\227\310$\65\224\12"
  24. "I\7\227\310\254X\15J\7\227\310\226\252\2K\10\227\310$\265\222\12L\7\227\310\304\346\0M\10\227"
  25. "\310\244\61\224\12N\10\227\310\244q\250\0O\7\227\310UV\5P\10\227\310\250\264b\4Q\10\227\310"
  26. "Uj$\1R\10\227\310\250\64V\1S\10\227\310m\220\301\2T\7\227\310\254\330\2U\7\227\310$"
  27. "W\22V\10\227\310$\253L\0W\10\227\310$\65\206\12X\10\227\310$\325R\1Y\10\227\310$U"
  28. "V\0Z\7\227\310\314T\16[\7\227\310\214X\16\134\10\217\311d\220A\0]\7\227\310\314r\4^"
  29. "\5\213\313\65_\5\207\310\14`\6\212\313\304\0a\7\223\310\310\65\2b\10\227\310D\225\324\2c\7"
  30. "\223\310\315\14\4d\10\227\310\246\245\222\0e\6\223\310\235\2f\10\227\310\246\264b\2g\10\227\307\35"
  31. "\61%\0h\10\227\310D\225\254\0i\6\265\310\244\1j\10\233\307f\30U\5k\10\227\310\304\264T"
  32. "\1l\7\227\310\310\326\0m\7\223\310<R\0n\7\223\310\250d\5o\7\223\310U\252\2p\10\227"
  33. "\307\250\244V\4q\10\227\307-\225d\0r\6\223\310\315\22s\10\223\310\215\70\22\0t\10\227\310\245"
  34. "\25\243\0u\7\223\310$+\11v\10\223\310$\65R\2w\7\223\310\244q\4x\7\223\310\244\62\25"
  35. "y\11\227\307$\225dJ\0z\7\223\310\254\221\6{\10\227\310\251\32D\1|\6\265\310(\1}\11"
  36. "\227\310\310\14RR\0~\6\213\313\215\4\0\0\0\4\377\377\0";
  37. typedef enum
  38. {
  39. EventTypeInput,
  40. } EventType;
  41. typedef struct
  42. {
  43. EventType type;
  44. InputEvent input;
  45. } FlizzerTrackerEvent;
  46. typedef enum
  47. {
  48. PARAM_FREQUENCY,
  49. PARAM_WAVEFORM,
  50. PARAM_PW,
  51. PARAM_ENABLE_FILTER,
  52. PARAM_FILTER_CUTOFF,
  53. PARAM_FILTER_RESONANCE,
  54. PARAM_FILTER_TYPE,
  55. } SelectedParam;
  56. const char* wave_names[] =
  57. {
  58. "NONE",
  59. "NOISE",
  60. "PULSE",
  61. "TRIANGLE",
  62. "SAWTOOTH",
  63. "METAL NOISE",
  64. "SINE",
  65. };
  66. const char* filter_names[] =
  67. {
  68. "NONE",
  69. "LOWPASS",
  70. "HIGHPASS",
  71. "BANDPASS",
  72. };
  73. #define NUM_PARAMS 10
  74. static void draw_callback(Canvas* canvas, void* ctx)
  75. {
  76. FlizzerTrackerApp* tracker = (FlizzerTrackerApp*)ctx;
  77. canvas_clear(canvas);
  78. canvas_set_color(canvas, ColorXOR);
  79. canvas_set_custom_font(canvas, u8g2_font_tom_thumb_4x6_tr);
  80. draw_pattern_view(canvas, tracker);
  81. /*char buffer[30] = {0};
  82. snprintf(buffer, 20, "FREQUENCY:%ld Hz", tracker->frequency);
  83. canvas_draw_str(canvas, 0, 6, buffer);
  84. snprintf(buffer, 20, "WAVEFORM:%s", wave_names[tracker->current_waveform_index]);
  85. canvas_draw_str(canvas, 0, 12, buffer);
  86. snprintf(buffer, 20, "PULSE WIDTH:$%03X", tracker->pw);
  87. canvas_draw_str(canvas, 0, 18, buffer);
  88. snprintf(buffer, 20, "FILTER:%s", (tracker->flags & SE_ENABLE_FILTER) ? "ENABLED" : "DISABLED");
  89. canvas_draw_str(canvas, 0, 24, buffer);
  90. snprintf(buffer, 20, "CUTOFF:%d", tracker->cutoff);
  91. canvas_draw_str(canvas, 0, 30, buffer);
  92. snprintf(buffer, 20, "RESONANCE:%d", tracker->resonance);
  93. canvas_draw_str(canvas, 0, 36, buffer);
  94. snprintf(buffer, 20, "TYPE:%s", filter_names[tracker->filter_type]);
  95. canvas_draw_str(canvas, 0, 42, buffer);
  96. snprintf(buffer, 20, "TR.ENG.CALLS: %d", tracker->sound_engine.channel[0].adsr.volume);
  97. canvas_draw_str(canvas, 0, 48, buffer);
  98. canvas_draw_str(canvas, 70, tracker->selected_param * 6 + 6, "<");
  99. uint32_t bytes = memmgr_get_free_heap();
  100. snprintf(buffer, 20, "BYTES FREE:%ld", bytes);
  101. canvas_draw_str(canvas, 0, 64, buffer);*/
  102. //canvas_draw_icon(canvas, 0, 0, &I_test);
  103. }
  104. static void input_callback(InputEvent* input_event, void* ctx)
  105. {
  106. // Проверяем, что контекст не нулевой
  107. furi_assert(ctx);
  108. FuriMessageQueue* event_queue = ctx;
  109. FlizzerTrackerEvent event = {.type = EventTypeInput, .input = *input_event};
  110. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  111. }
  112. const uint8_t waveforms[] =
  113. {
  114. SE_WAVEFORM_NONE,
  115. SE_WAVEFORM_NOISE,
  116. SE_WAVEFORM_PULSE,
  117. SE_WAVEFORM_TRIANGLE,
  118. SE_WAVEFORM_SAW,
  119. SE_WAVEFORM_NOISE_METAL,
  120. SE_WAVEFORM_SINE,
  121. };
  122. int32_t flizzer_tracker_app(void* p)
  123. {
  124. UNUSED(p);
  125. // Текущее событие типа кастомного типа FlizzerTrackerEvent
  126. FlizzerTrackerEvent event;
  127. // Очередь событий на 8 элементов размера FlizzerTrackerEvent
  128. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(FlizzerTrackerEvent));
  129. FlizzerTrackerApp* tracker = init_tracker(44100, 50, true, 1024);
  130. // Создаем новый view port
  131. ViewPort* view_port = view_port_alloc();
  132. // Создаем callback отрисовки, без контекста
  133. view_port_draw_callback_set(view_port, draw_callback, tracker);
  134. // Создаем callback нажатий на клавиши, в качестве контекста передаем
  135. // нашу очередь сообщений, чтоб запихивать в неё эти события
  136. view_port_input_callback_set(view_port, input_callback, event_queue);
  137. // Создаем GUI приложения
  138. Gui* gui = furi_record_open(RECORD_GUI);
  139. // Подключаем view port к GUI в полноэкранном режиме
  140. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  141. tracker->notification = furi_record_open(RECORD_NOTIFICATION);
  142. notification_message(tracker->notification, &sequence_display_backlight_enforce_on);
  143. //tracker->sound_engine.channel[0].waveform = SE_WAVEFORM_NOISE;
  144. tracker->frequency = 440;
  145. tracker->current_waveform_index = 1;
  146. /*sound_engine_set_channel_frequency(&tracker->sound_engine, &tracker->sound_engine.channel[0], ((12 * 4) << 8));
  147. tracker->sound_engine.channel[0].adsr.a = 0x10;
  148. tracker->sound_engine.channel[0].adsr.d = 0x10;
  149. tracker->sound_engine.channel[0].adsr.s = 0xff;
  150. tracker->sound_engine.channel[0].adsr.volume = 0x80;
  151. tracker->sound_engine.channel[0].adsr.envelope_state = ATTACK;
  152. SoundEngine* eng = &(tracker->sound_engine);
  153. tracker->sound_engine.channel[0].adsr.envelope_speed = envspd(eng, tracker->sound_engine.channel[0].adsr.a);*/
  154. tracker->song.speed = 5;
  155. tracker->song.num_instruments = 2;
  156. tracker->song.num_patterns = 2;
  157. tracker->song.num_sequence_steps = 1;
  158. tracker->song.pattern_length = 64;
  159. tracker->song.sequence.sequence_step[0].pattern_indices[0] = 0;
  160. tracker->song.sequence.sequence_step[0].pattern_indices[1] = 1;
  161. tracker->song.sequence.sequence_step[0].pattern_indices[2] = 2;
  162. tracker->song.sequence.sequence_step[0].pattern_indices[3] = 2;
  163. tracker->song.pattern[0].step = malloc(64 * sizeof(TrackerSongPatternStep));
  164. tracker->song.pattern[1].step = malloc(64 * sizeof(TrackerSongPatternStep));
  165. tracker->song.pattern[2].step = malloc(64 * sizeof(TrackerSongPatternStep));
  166. memset(tracker->song.pattern[0].step, 0, 64 * sizeof(TrackerSongPatternStep));
  167. memset(tracker->song.pattern[1].step, 0, 64 * sizeof(TrackerSongPatternStep));
  168. memset(tracker->song.pattern[2].step, 0, 64 * sizeof(TrackerSongPatternStep));
  169. tracker->song.instrument[0] = malloc(sizeof(Instrument));
  170. tracker->song.instrument[1] = malloc(sizeof(Instrument));
  171. for(int i = 0; i < 64; ++i)
  172. {
  173. set_note(&tracker->song.pattern[0].step[i], MUS_NOTE_NONE);
  174. set_note(&tracker->song.pattern[1].step[i], MUS_NOTE_NONE);
  175. set_note(&tracker->song.pattern[2].step[i], MUS_NOTE_NONE);
  176. set_instrument(&tracker->song.pattern[0].step[i], MUS_NOTE_INSTRUMENT_NONE);
  177. set_instrument(&tracker->song.pattern[1].step[i], MUS_NOTE_INSTRUMENT_NONE);
  178. set_instrument(&tracker->song.pattern[2].step[i], MUS_NOTE_INSTRUMENT_NONE);
  179. set_volume(&tracker->song.pattern[0].step[i], MUS_NOTE_VOLUME_NONE);
  180. set_volume(&tracker->song.pattern[1].step[i], MUS_NOTE_VOLUME_NONE);
  181. set_volume(&tracker->song.pattern[2].step[i], MUS_NOTE_VOLUME_NONE);
  182. }
  183. for(int i = 0; i < 64; i += 8)
  184. {
  185. set_note(&tracker->song.pattern[0].step[0 + i], 12 * 5);
  186. set_note(&tracker->song.pattern[0].step[2 + i], 12 * 5 + 2);
  187. set_note(&tracker->song.pattern[0].step[4 + i], 12 * 5 - 2);
  188. set_note(&tracker->song.pattern[0].step[6 + i], 12 * 5 + 4);
  189. set_instrument(&tracker->song.pattern[0].step[0 + i], 0);
  190. set_instrument(&tracker->song.pattern[0].step[2 + i], 0);
  191. set_instrument(&tracker->song.pattern[0].step[4 + i], 0);
  192. set_instrument(&tracker->song.pattern[0].step[6 + i], 0);
  193. }
  194. for(int i = 0; i < 64; i++)
  195. {
  196. set_note(&tracker->song.pattern[1].step[i], 12 * 7 + 11);
  197. set_instrument(&tracker->song.pattern[1].step[i], 1);
  198. }
  199. tracker->song.instrument[0]->adsr.a = 0x2;
  200. tracker->song.instrument[0]->adsr.d = 0x9;
  201. tracker->song.instrument[0]->adsr.volume = 0x80;
  202. tracker->song.instrument[0]->waveform = SE_WAVEFORM_TRIANGLE;
  203. tracker->song.instrument[1]->adsr.a = 0x0;
  204. tracker->song.instrument[1]->adsr.d = 0x3;
  205. tracker->song.instrument[1]->adsr.volume = 0x18;
  206. tracker->song.instrument[1]->waveform = SE_WAVEFORM_NOISE;
  207. tracker->tracker_engine.playing = false;
  208. play();
  209. // Бесконечный цикл обработки очереди событий
  210. while(1)
  211. {
  212. // Выбираем событие из очереди в переменную event (ждём бесконечно долго, если очередь пуста)
  213. // и проверяем, что у нас получилось это сделать
  214. furi_check(furi_message_queue_get(event_queue, &event, FuriWaitForever) == FuriStatusOk);
  215. // Наше событие — это нажатие кнопки
  216. if(event.type == EventTypeInput)
  217. {
  218. // Если нажата кнопка "назад", то выходим из цикла, а следовательно и из приложения
  219. if(event.input.key == InputKeyBack && event.input.type == InputTypeShort)
  220. {
  221. break;
  222. }
  223. if(event.input.key == InputKeyOk && event.input.type == InputTypeShort)
  224. {
  225. tracker->tracker_engine.playing = !(tracker->tracker_engine.playing);
  226. }
  227. if(event.input.key == InputKeyUp && event.input.type == InputTypeShort)
  228. {
  229. if(tracker->selected_param > 0)
  230. {
  231. tracker->selected_param--;
  232. }
  233. }
  234. if(event.input.key == InputKeyDown && event.input.type == InputTypeShort)
  235. {
  236. if(tracker->selected_param < NUM_PARAMS - 1)
  237. {
  238. tracker->selected_param++;
  239. }
  240. }
  241. if(event.input.key == InputKeyRight && event.input.type == InputTypeShort)
  242. {
  243. switch(tracker->selected_param)
  244. {
  245. case PARAM_FREQUENCY:
  246. {
  247. tracker->frequency += 25;
  248. sound_engine_set_channel_frequency(&tracker->sound_engine, &tracker->sound_engine.channel[0], tracker->frequency * 1024);
  249. break;
  250. }
  251. case PARAM_WAVEFORM:
  252. {
  253. if(tracker->current_waveform_index < 6)
  254. {
  255. tracker->current_waveform_index++;
  256. }
  257. tracker->sound_engine.channel[0].waveform = waveforms[tracker->current_waveform_index];
  258. break;
  259. }
  260. case PARAM_PW:
  261. {
  262. if(tracker->pw + 0x80 < 0xFFF)
  263. {
  264. tracker->pw += 0x80;
  265. }
  266. else
  267. {
  268. tracker->pw = 0x80;
  269. }
  270. tracker->sound_engine.channel[0].pw = tracker->pw;
  271. break;
  272. }
  273. case PARAM_ENABLE_FILTER:
  274. {
  275. tracker->flags ^= SE_ENABLE_FILTER;
  276. tracker->sound_engine.channel[0].flags = tracker->flags;
  277. break;
  278. }
  279. case PARAM_FILTER_CUTOFF:
  280. {
  281. tracker->cutoff += 10;
  282. if(tracker->cutoff > 0xFFF)
  283. {
  284. tracker->cutoff = 0xFFF;
  285. }
  286. tracker->sound_engine.channel[0].filter_cutoff = tracker->cutoff;
  287. sound_engine_filter_set_coeff(&tracker->sound_engine.channel[0].filter, tracker->cutoff, tracker->resonance * 50);
  288. break;
  289. }
  290. case PARAM_FILTER_RESONANCE:
  291. {
  292. tracker->resonance++;
  293. sound_engine_filter_set_coeff(&tracker->sound_engine.channel[0].filter, tracker->cutoff, tracker->resonance * 50);
  294. break;
  295. }
  296. case PARAM_FILTER_TYPE:
  297. {
  298. if(tracker->filter_type < 3)
  299. {
  300. tracker->filter_type++;
  301. }
  302. tracker->sound_engine.channel[0].filter_mode = tracker->filter_type;
  303. break;
  304. }
  305. }
  306. }
  307. if(event.input.key == InputKeyLeft && event.input.type == InputTypeShort)
  308. {
  309. switch(tracker->selected_param)
  310. {
  311. case PARAM_FREQUENCY:
  312. {
  313. if(tracker->frequency > 25)
  314. {
  315. tracker->frequency -= 25;
  316. }
  317. sound_engine_set_channel_frequency(&tracker->sound_engine, &tracker->sound_engine.channel[0], tracker->frequency * 1024);
  318. break;
  319. }
  320. case PARAM_WAVEFORM:
  321. {
  322. if(tracker->current_waveform_index > 0)
  323. {
  324. tracker->current_waveform_index--;
  325. }
  326. tracker->sound_engine.channel[0].waveform = waveforms[tracker->current_waveform_index];
  327. break;
  328. }
  329. case PARAM_PW:
  330. {
  331. if(tracker->pw - 0x80 > 0)
  332. {
  333. tracker->pw -= 0x80;
  334. }
  335. else
  336. {
  337. tracker->pw = 0xF80;
  338. }
  339. tracker->sound_engine.channel[0].pw = tracker->pw;
  340. break;
  341. }
  342. case PARAM_ENABLE_FILTER:
  343. {
  344. tracker->flags ^= SE_ENABLE_FILTER;
  345. tracker->sound_engine.channel[0].flags = tracker->flags;
  346. break;
  347. }
  348. case PARAM_FILTER_CUTOFF:
  349. {
  350. if(tracker->cutoff > 10)
  351. {
  352. tracker->cutoff -= 10;
  353. }
  354. tracker->sound_engine.channel[0].filter_cutoff = tracker->cutoff;
  355. sound_engine_filter_set_coeff(&tracker->sound_engine.channel[0].filter, tracker->cutoff, tracker->resonance * 50);
  356. break;
  357. }
  358. case PARAM_FILTER_RESONANCE:
  359. {
  360. if(tracker->resonance > 0)
  361. {
  362. tracker->resonance--;
  363. sound_engine_filter_set_coeff(&tracker->sound_engine.channel[0].filter, tracker->cutoff, tracker->resonance * 50);
  364. }
  365. break;
  366. }
  367. case PARAM_FILTER_TYPE:
  368. {
  369. if(tracker->filter_type > 0)
  370. {
  371. tracker->filter_type--;
  372. }
  373. tracker->sound_engine.channel[0].filter_mode = tracker->filter_type;
  374. break;
  375. }
  376. }
  377. }
  378. }
  379. }
  380. stop();
  381. notification_message(tracker->notification, &sequence_display_backlight_enforce_auto);
  382. furi_record_close(RECORD_NOTIFICATION);
  383. // Специальная очистка памяти, занимаемой очередью
  384. furi_message_queue_free(event_queue);
  385. // Чистим созданные объекты, связанные с интерфейсом
  386. gui_remove_view_port(gui, view_port);
  387. view_port_free(view_port);
  388. furi_record_close(RECORD_GUI);
  389. deinit_tracker(tracker);
  390. return 0;
  391. }