instrument_editor.c 13 KB

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