instrument_editor.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376
  1. #include "instrument_editor.h"
  2. #include "pattern_editor.h"
  3. #include "../macros.h"
  4. #include "opcode_description.h"
  5. #include <flizzer_tracker_icons.h>
  6. void draw_inst_flag(FlizzerTrackerApp *tracker, Canvas *canvas, uint8_t focus, uint8_t param, const char *text, uint8_t x, uint8_t y, uint16_t flags, uint16_t mask)
  7. {
  8. canvas_draw_icon(canvas, x, y - 5, ((flags & mask) ? &I_checkbox_checked : &I_checkbox_empty));
  9. canvas_draw_str(canvas, x + 6, y, text);
  10. if (tracker->focus == focus && tracker->selected_param == param && tracker->editing)
  11. {
  12. if (text[strlen(text) - 1] == ':')
  13. {
  14. canvas_draw_box(canvas, x + 5, y - 6, strlen(text) * 4 - 1, 7);
  15. }
  16. else
  17. {
  18. canvas_draw_box(canvas, x + 5, y - 6, strlen(text) * 4 + 1, 7);
  19. }
  20. }
  21. }
  22. void draw_inst_text_one_digit(FlizzerTrackerApp *tracker, Canvas *canvas, uint8_t focus, uint8_t param, const char *text, uint8_t x, uint8_t y, uint8_t value) // text MUST end with semicolon
  23. {
  24. canvas_draw_str(canvas, x, y, text);
  25. char buffer[4];
  26. snprintf(buffer, sizeof(buffer), "%01X", (value & 0xF));
  27. canvas_draw_str(canvas, x + strlen(text) * 4 - 2, y, buffer);
  28. if (tracker->focus == focus && tracker->selected_param == param && tracker->editing)
  29. {
  30. canvas_draw_box(canvas, x + strlen(text) * 4 - 3, y - 6, 5, 7);
  31. }
  32. }
  33. void draw_inst_text_two_digits(FlizzerTrackerApp *tracker, Canvas *canvas, uint8_t focus, uint8_t param, const char *text, uint8_t x, uint8_t y, uint8_t value) // text MUST end with semicolon
  34. {
  35. canvas_draw_str(canvas, x, y, text);
  36. char buffer[4];
  37. snprintf(buffer, sizeof(buffer), "%02X", value);
  38. canvas_draw_str(canvas, x + strlen(text) * 4 - 2, y, buffer);
  39. if (tracker->focus == focus && tracker->selected_param == param && tracker->editing)
  40. {
  41. canvas_draw_box(canvas, x + strlen(text) * 4 + 4 * tracker->current_digit - 3, y - 6, 5, 7);
  42. }
  43. }
  44. static const char *filter_types[] =
  45. {
  46. "NONE",
  47. "LOW",
  48. "HIGH",
  49. "BAND",
  50. "LOHI",
  51. "HIBD",
  52. "LOBD",
  53. "ALL",
  54. };
  55. static const char *instrument_editor_params_description[] =
  56. {
  57. "CURRENT INSTRUMENT",
  58. "CURRENT INSTRUMENT NAME",
  59. "INSTRUMENT BASE NOTE",
  60. "INSTRUMENT FINETUNE",
  61. "SLIDE SPEED",
  62. "SET PULSE WIDTH ON KEYDOWN",
  63. "PULSE WIDTH",
  64. "SET FILTER PARAMETERS ON KEYDOWN",
  65. "NOISE WAVEFORM",
  66. "PULSE WAVEFORM",
  67. "TRIANGLE WAVEFORM",
  68. "SAWTOOTH WAVEFORM",
  69. "METALLIC NOISE WAVEFORM",
  70. "SINE WAVEFORM",
  71. "ENVELOPE ATTACK",
  72. "ENVELOPE DECAY",
  73. "ENVELOPE SUSTAIN",
  74. "ENVELOPE RELEASE",
  75. "ENVELOPE VOLUME",
  76. "ENABLE FILTER",
  77. "FILTER CUTOFF FREQUENCY",
  78. "FILTER RESONANCE",
  79. "FILTER TYPE (NONE=OFF)",
  80. "ENABLE RING MODULATION",
  81. "RINGMOD SOURCE CHANNEL (F=SELF)",
  82. "ENABLE HARD SYNC",
  83. "HARDSYNC SOURCE CHANNEL (F=SELF)",
  84. "RETRIGGER INSTRUMENT ON SLIDE",
  85. "SYNC OSCILLATORS ON KEYDOWN",
  86. "ENABLE VIBRATO",
  87. "VIBRATO SPEED",
  88. "VIBRATO DEPTH",
  89. "VIBRATO DELAY (IN TICKS)",
  90. "ENABLE PWM",
  91. "PWM SPEED",
  92. "PWM DEPTH",
  93. "PWM DELAY (IN TICKS)",
  94. "DON'T RESTART PROGRAM ON KEYDOWN",
  95. "PROG.PERIOD (00 = PROGRAM OFF)",
  96. };
  97. void draw_instrument_view(Canvas *canvas, FlizzerTrackerApp *tracker)
  98. {
  99. SoundEngineChannel *se_channel = &tracker->sound_engine.channel[0];
  100. if (!(se_channel->flags & SE_ENABLE_GATE) && tracker->tracker_engine.song == NULL)
  101. {
  102. stop();
  103. tracker->tracker_engine.playing = false;
  104. tracker_engine_set_song(&tracker->tracker_engine, &tracker->song);
  105. }
  106. char buffer[30];
  107. Instrument *inst = tracker->song.instrument[tracker->current_instrument];
  108. uint8_t shift = tracker->inst_editor_shift;
  109. if (shift < 6)
  110. {
  111. snprintf(buffer, sizeof(buffer), "INST:%c", to_char(tracker->current_instrument));
  112. draw_generic_n_digit_field(tracker, canvas, EDIT_INSTRUMENT, INST_CURRENTINSTRUMENT, buffer, 0, 5 - shift, 1);
  113. snprintf(buffer, sizeof(buffer), "%s", tracker->song.instrument[tracker->current_instrument]->name);
  114. draw_generic_n_digit_field(tracker, canvas, EDIT_INSTRUMENT, INST_INSTRUMENTNAME, buffer, 4 * 7 - 1, 5 - shift, 1);
  115. }
  116. if (shift < 12)
  117. {
  118. snprintf(buffer, sizeof(buffer), "NOTE:%s", notename(inst->base_note));
  119. canvas_draw_str(canvas, 0, 11 - shift, buffer);
  120. if (tracker->editing && tracker->focus == EDIT_INSTRUMENT && tracker->selected_param == INST_CURRENT_NOTE)
  121. {
  122. if (tracker->current_digit)
  123. {
  124. canvas_draw_box(canvas, 19 + 2 * 4, 5 - shift, 5, 7);
  125. }
  126. else
  127. {
  128. canvas_draw_box(canvas, 19, 5 - shift, 5 + 4, 7);
  129. }
  130. }
  131. snprintf(buffer, sizeof(buffer), "FINE:%+02d", inst->finetune);
  132. canvas_draw_str(canvas, 37, 11 - shift, buffer);
  133. if (tracker->editing && tracker->focus == EDIT_INSTRUMENT && tracker->selected_param == INST_FINETUNE)
  134. {
  135. if (tracker->current_digit)
  136. {
  137. canvas_draw_box(canvas, 60 + 4, 5 - shift, 5, 7);
  138. }
  139. else
  140. {
  141. canvas_draw_box(canvas, 60, 5 - shift, 5, 7);
  142. }
  143. }
  144. }
  145. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_SLIDESPEED, "SL.SPD:", 0, 17 - shift, inst->slide_speed);
  146. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_SETPW, "PW:", 36, 17 - shift, inst->flags, TE_SET_PW);
  147. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_PW, "", 54, 17 - shift, inst->pw);
  148. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_SETCUTOFF, "CUT", 61, 17 - shift, inst->flags, TE_SET_CUTOFF);
  149. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_WAVE_NOISE, "N", 0, 23 - shift, inst->waveform, SE_WAVEFORM_NOISE);
  150. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_WAVE_PULSE, "P", 10, 23 - shift, inst->waveform, SE_WAVEFORM_PULSE);
  151. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_WAVE_TRIANGLE, "T", 20, 23 - shift, inst->waveform, SE_WAVEFORM_TRIANGLE);
  152. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_WAVE_SAWTOOTH, "S", 30, 23 - shift, inst->waveform, SE_WAVEFORM_SAW);
  153. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_WAVE_NOISE_METAL, "M", 40, 23 - shift, inst->waveform, SE_WAVEFORM_NOISE_METAL);
  154. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_WAVE_SINE, "SINE", 50, 23 - shift, inst->waveform, SE_WAVEFORM_SINE);
  155. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_ATTACK, "A:", 0, 29 - shift, inst->adsr.a);
  156. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_DECAY, "D:", 16, 29 - shift, inst->adsr.d);
  157. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_SUSTAIN, "S:", 32, 29 - shift, inst->adsr.s);
  158. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_RELEASE, "R:", 48, 29 - shift, inst->adsr.r);
  159. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_VOLUME, "V:", 64, 29 - shift, inst->adsr.volume);
  160. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_ENABLEFILTER, "FIL", 0, 35 - shift, inst->sound_engine_flags, SE_ENABLE_FILTER);
  161. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_FILTERCUTOFF, "CUT:", 20, 35 - shift, inst->filter_cutoff);
  162. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_FILTERRESONANCE, "RES:", 44, 35 - shift, inst->filter_resonance);
  163. snprintf(buffer, sizeof(buffer), "TYPE:%s", filter_types[inst->filter_type]);
  164. canvas_draw_str(canvas, 0, 41 - shift, buffer);
  165. if (tracker->editing && tracker->focus == EDIT_INSTRUMENT && tracker->selected_param == INST_FILTERTYPE)
  166. {
  167. canvas_draw_box(canvas, 19, 35 - shift, strlen(filter_types[inst->filter_type]) * 4 + 1, 7);
  168. }
  169. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_ENABLERINGMOD, "R:", 38, 41 - shift, inst->sound_engine_flags, SE_ENABLE_RING_MOD);
  170. draw_inst_text_one_digit(tracker, canvas, EDIT_INSTRUMENT, INST_RINGMODSRC, "", 52, 41 - shift, inst->ring_mod);
  171. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_ENABLEHARDSYNC, "H:", 56, 41 - shift, inst->sound_engine_flags, SE_ENABLE_HARD_SYNC);
  172. draw_inst_text_one_digit(tracker, canvas, EDIT_INSTRUMENT, INST_HARDSYNCSRC, "", 70, 41 - shift, inst->hard_sync);
  173. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_RETRIGGERONSLIDE, "SL.RETRIG", 0, 47 - shift, inst->flags, TE_RETRIGGER_ON_SLIDE);
  174. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_ENABLEKEYSYNC, "KSYNC", 44, 47 - shift, inst->sound_engine_flags, SE_ENABLE_KEYDOWN_SYNC);
  175. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_ENABLEVIBRATO, "VIB", 0, 53 - shift, inst->flags, TE_ENABLE_VIBRATO);
  176. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_VIBRATOSPEED, "S:", 20, 53 - shift, inst->vibrato_speed);
  177. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_VIBRATODEPTH, "D:", 36, 53 - shift, inst->vibrato_depth);
  178. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_VIBRATODELAY, "DEL:", 52, 53 - shift, inst->vibrato_delay);
  179. if (shift >= 6)
  180. {
  181. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_ENABLEPWM, "PWM", 0, 59 - shift, inst->flags, TE_ENABLE_PWM);
  182. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_PWMSPEED, "S:", 20, 59 - shift, inst->pwm_speed);
  183. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_PWMDEPTH, "D:", 36, 59 - shift, inst->pwm_depth);
  184. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_PWMDELAY, "DEL:", 52, 59 - shift, inst->pwm_delay);
  185. }
  186. if (shift >= 12)
  187. {
  188. draw_inst_flag(tracker, canvas, EDIT_INSTRUMENT, INST_PROGRESTART, "NO PROG.RESTART", 0, 65 - shift, inst->flags, TE_PROG_NO_RESTART);
  189. }
  190. draw_inst_text_two_digits(tracker, canvas, EDIT_INSTRUMENT, INST_PROGRAMEPERIOD, "P.PERIOD:", 81, 56, inst->program_period);
  191. canvas_draw_line(canvas, 0, 57, 127, 57);
  192. canvas_draw_line(canvas, 79, 0, 79, 56);
  193. canvas_draw_line(canvas, 80, 49, 127, 49);
  194. if (tracker->focus == EDIT_INSTRUMENT)
  195. {
  196. canvas_draw_str(canvas, 0, 64, instrument_editor_params_description[tracker->selected_param]);
  197. }
  198. }
  199. char command_get_char(uint16_t command)
  200. {
  201. if ((command >> 8) < 36)
  202. {
  203. return to_char_array[(command >> 8)];
  204. }
  205. if (command == TE_PROGRAM_END)
  206. {
  207. return ':';
  208. }
  209. if ((command & 0xff00) == TE_PROGRAM_JUMP)
  210. {
  211. return '^';
  212. }
  213. if ((command & 0xff00) == TE_PROGRAM_LOOP_END)
  214. {
  215. return '>';
  216. }
  217. if ((command & 0xff00) == TE_PROGRAM_LOOP_BEGIN)
  218. {
  219. return '<';
  220. }
  221. return '?';
  222. }
  223. void draw_program_step(Canvas *canvas, uint8_t y, FlizzerTrackerApp *tracker, uint8_t index)
  224. {
  225. char buffer[15];
  226. Instrument *inst = tracker->song.instrument[tracker->current_instrument];
  227. uint16_t opcode = inst->program[index];
  228. if (opcode != TE_PROGRAM_NOP)
  229. {
  230. if ((opcode & 0x7f00) == TE_EFFECT_ARPEGGIO)
  231. {
  232. if ((opcode & 0xff) != 0xf0 && (opcode & 0xff) != 0xf1)
  233. {
  234. snprintf(buffer, sizeof(buffer), "%01X %c%02X %s", index, command_get_char(opcode & 0x7fff), (opcode & 0xff), notename(my_min(12 * 7 + 11, (opcode & 0xff) + tracker->song.instrument[tracker->current_instrument]->base_note)));
  235. }
  236. else
  237. {
  238. snprintf(buffer, sizeof(buffer), "%01X %c%02X %s", index, command_get_char(opcode & 0x7fff), (opcode & 0xff), notename((opcode & 0xff)));
  239. }
  240. }
  241. else if ((opcode & 0x7f00) == TE_EFFECT_ARPEGGIO_ABS)
  242. {
  243. snprintf(buffer, sizeof(buffer), "%01X %c%02X F.%s", index, command_get_char(opcode & 0x7fff), (opcode & 0xff), notename(opcode & 0xff));
  244. }
  245. else
  246. {
  247. snprintf(buffer, sizeof(buffer), "%01X %c%02X %s", index, command_get_char(opcode & 0x7fff), (opcode & 0xff), get_opcode_description(opcode, true) ? get_opcode_description(opcode, true) : "");
  248. }
  249. if (opcode & 0x8000)
  250. {
  251. if (index == 0)
  252. {
  253. canvas_draw_line(canvas, 84 + 4 * 4 + 2, y, 84 + 4 * 4 + 2, y - 3);
  254. canvas_draw_dot(canvas, 84 + 4 * 4 + 1, y - 4);
  255. }
  256. if (index > 0 && !(inst->program[index - 1] & 0x8000))
  257. {
  258. canvas_draw_line(canvas, 84 + 4 * 4 + 2, y, 84 + 4 * 4 + 2, y - 3);
  259. canvas_draw_dot(canvas, 84 + 4 * 4 + 1, y - 4);
  260. }
  261. if (index > 0 && (inst->program[index - 1] & 0x8000))
  262. {
  263. canvas_draw_line(canvas, 84 + 4 * 4 + 2, y, 84 + 4 * 4 + 2, y - 5);
  264. }
  265. }
  266. else
  267. {
  268. if (index > 0 && (inst->program[index - 1] & 0x8000))
  269. {
  270. canvas_draw_line(canvas, 84 + 4 * 4 + 2, y - 3, 84 + 4 * 4 + 2, y - 5);
  271. canvas_draw_dot(canvas, 84 + 4 * 4 + 1, y - 2);
  272. }
  273. }
  274. }
  275. else
  276. {
  277. snprintf(buffer, sizeof(buffer), "%01X ---", index);
  278. }
  279. canvas_draw_str(canvas, 81, y, buffer);
  280. }
  281. void draw_instrument_program_view(Canvas *canvas, FlizzerTrackerApp *tracker)
  282. {
  283. Instrument *inst = tracker->song.instrument[tracker->current_instrument];
  284. for (uint8_t i = tracker->program_position; i < my_min(INST_PROG_LEN, tracker->program_position + 8); i++)
  285. {
  286. draw_program_step(canvas, 6 + 6 * i - tracker->program_position * 6, tracker, i);
  287. if (i == tracker->current_program_step && tracker->focus == EDIT_PROGRAM)
  288. {
  289. if (tracker->editing)
  290. {
  291. canvas_draw_box(canvas, 80 + 8 + tracker->current_digit * 4, 6 * i - tracker->program_position * 6, 5, 7);
  292. }
  293. else
  294. {
  295. canvas_draw_box(canvas, 80, 6 * i - tracker->program_position * 6, 5, 7);
  296. }
  297. }
  298. }
  299. // draw arrow pointing at current program step
  300. for (uint8_t i = 0; i < SONG_MAX_CHANNELS; i++)
  301. {
  302. if (tracker->tracker_engine.channel[i].instrument == inst && (tracker->tracker_engine.channel[i].channel_flags & TEC_PROGRAM_RUNNING) && (tracker->tracker_engine.sound_engine->channel[i].flags & SE_ENABLE_GATE))
  303. {
  304. if (tracker->tracker_engine.channel[i].program_tick >= tracker->program_position && tracker->tracker_engine.channel[i].program_tick < tracker->program_position + 8)
  305. {
  306. canvas_draw_str(canvas, 85, 6 * tracker->tracker_engine.channel[i].program_tick - tracker->program_position * 6 + 6, ">");
  307. break;
  308. }
  309. }
  310. }
  311. if (tracker->focus == EDIT_PROGRAM)
  312. {
  313. uint16_t opcode = (inst->program[tracker->current_program_step] & 0x7fff);
  314. canvas_draw_str(canvas, 0, 64, get_opcode_description(opcode, false) ? get_opcode_description(opcode, false) : "");
  315. }
  316. }