util.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  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) return false; //if we hop through several patterns (e.g. editing upper digit)
  80. if(!(is_pattern_empty(song, pattern - 1))) //don't let the user flood the song with empty patterns
  81. {
  82. song->pattern[pattern].step = malloc(sizeof(TrackerSongPatternStep) * song->pattern_length);
  83. set_empty_pattern(song, pattern);
  84. song->num_patterns++;
  85. return true;
  86. }
  87. else
  88. {
  89. return false;
  90. }
  91. }
  92. }