util.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. #include "util.h"
  2. void set_note(TrackerSongPatternStep *step, uint8_t note)
  3. {
  4. step->note &= 0x80;
  5. step->note |= (note & 0x7f);
  6. }
  7. void set_instrument(TrackerSongPatternStep *step, uint8_t inst)
  8. {
  9. step->note &= 0x7f;
  10. step->inst_vol &= 0x0f;
  11. step->note |= ((inst & 0x10) << 3);
  12. step->inst_vol |= ((inst & 0xf) << 4);
  13. }
  14. void set_volume(TrackerSongPatternStep *step, uint8_t vol)
  15. {
  16. step->command &= 0x7fff;
  17. step->inst_vol &= 0xf0;
  18. step->command |= ((vol & 0x10) << 11);
  19. step->inst_vol |= (vol & 0xf);
  20. }
  21. void set_command(TrackerSongPatternStep *step, uint16_t command)
  22. {
  23. step->command &= 0x8000;
  24. step->command |= command & (0x7fff);
  25. }
  26. void play_song(FlizzerTrackerApp *tracker, bool from_cursor)
  27. {
  28. tracker->tracker_engine.playing = true;
  29. tracker->was_editing = tracker->editing;
  30. tracker->editing = false;
  31. if (!(from_cursor))
  32. {
  33. tracker->tracker_engine.pattern_position = 0;
  34. }
  35. play();
  36. }
  37. void stop_song(FlizzerTrackerApp *tracker)
  38. {
  39. tracker->tracker_engine.playing = false;
  40. tracker->editing = tracker->was_editing;
  41. stop();
  42. }
  43. bool is_pattern_empty(TrackerSong *song, uint8_t pattern)
  44. {
  45. TrackerSongPattern song_pattern = song->pattern[pattern];
  46. for (int i = 0; i < song->pattern_length; i++)
  47. {
  48. TrackerSongPatternStep *step = &song_pattern.step[i];
  49. if (tracker_engine_get_note(step) != MUS_NOTE_NONE ||
  50. tracker_engine_get_instrument(step) != MUS_NOTE_INSTRUMENT_NONE ||
  51. tracker_engine_get_volume(step) != MUS_NOTE_VOLUME_NONE ||
  52. tracker_engine_get_command(step) != 0)
  53. {
  54. return false;
  55. }
  56. }
  57. return true;
  58. }
  59. void set_empty_pattern(TrackerSong *song, uint8_t pattern)
  60. {
  61. TrackerSongPattern song_pattern = song->pattern[pattern];
  62. for (int i = 0; i < song->pattern_length; i++)
  63. {
  64. TrackerSongPatternStep *step = &song_pattern.step[i];
  65. set_note(step, MUS_NOTE_NONE);
  66. set_instrument(step, MUS_NOTE_INSTRUMENT_NONE);
  67. set_volume(step, MUS_NOTE_VOLUME_NONE);
  68. set_command(step, 0);
  69. }
  70. }
  71. bool check_and_allocate_pattern(TrackerSong *song, uint8_t pattern)
  72. {
  73. if (pattern < song->num_patterns) // we can set this pattern since it already exists
  74. {
  75. return true;
  76. }
  77. else
  78. {
  79. if (song->pattern[pattern - 1].step == NULL)
  80. return false; // if we hop through several patterns (e.g. editing upper digit)
  81. if (!(is_pattern_empty(song, pattern - 1))) // don't let the user flood the song with empty patterns
  82. {
  83. song->pattern[pattern].step = malloc(sizeof(TrackerSongPatternStep) * song->pattern_length);
  84. set_empty_pattern(song, pattern);
  85. song->num_patterns++;
  86. return true;
  87. }
  88. else
  89. {
  90. return false;
  91. }
  92. }
  93. }