instrument_editor.c 14 KB

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