48class ArpeggiatorSettings {
50 ArpeggiatorSettings();
54 void cloneFrom(ArpeggiatorSettings
const* other);
56 bool readCommonTagsFromFile(
Deserializer& reader,
char const* tagName,
Song* songToConvertSyncLevel);
58 bool readNonAudioTagsFromFile(
Deserializer& reader,
char const* tagName);
60 void writeCommonParamsToFile(
Serializer& writer,
Song* songToConvertSyncLevel);
62 void writeNonAudioParamsToFile(
Serializer& writer);
64 void generateNewNotePattern();
66 void updatePresetFromCurrentSettings();
68 void updateSettingsFromCurrentPreset();
70 uint32_t getPhaseIncrement(int32_t arpRate);
73 ArpPreset preset{ArpPreset::OFF};
74 ArpMode mode{ArpMode::OFF};
76 bool includeInKitArp{
true};
79 ArpOctaveMode octaveMode{ArpOctaveMode::UP};
80 ArpNoteMode noteMode{ArpNoteMode::UP};
83 uint8_t numOctaves{2};
86 uint8_t numStepRepeats{1};
89 uint8_t chordTypeIndex{0};
96 bool randomizerLock{
false};
99 ArpMpeModSource mpeVelocity{ArpMpeModSource::OFF};
102 uint32_t lastLockedNoteProbabilityParameterValue{0};
103 uint32_t lastLockedBassProbabilityParameterValue{0};
104 uint32_t lastLockedSwapProbabilityParameterValue{0};
105 uint32_t lastLockedGlideProbabilityParameterValue{0};
106 uint32_t lastLockedReverseProbabilityParameterValue{0};
107 uint32_t lastLockedChordProbabilityParameterValue{0};
108 uint32_t lastLockedRatchetProbabilityParameterValue{0};
109 uint32_t lastLockedSpreadVelocityParameterValue{0};
110 uint32_t lastLockedSpreadGateParameterValue{0};
111 uint32_t lastLockedSpreadOctaveParameterValue{0};
114 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedNoteProbabilityValues;
115 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedBassProbabilityValues;
116 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedSwapProbabilityValues;
117 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedGlideProbabilityValues;
118 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedReverseProbabilityValues;
119 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedChordProbabilityValues;
120 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedRatchetProbabilityValues;
121 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedSpreadVelocityValues;
122 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedSpreadGateValues;
123 std::array<int8_t, RANDOMIZER_LOCK_MAX_SAVED_VALUES> lockedSpreadOctaveValues;
126 std::array<int8_t, PATTERN_MAX_BUFFER_SIZE> notePattern;
129 bool flagForceArpRestart{
false};
135 uint32_t sequenceLength{0};
136 uint32_t chordPolyphony{0};
137 uint32_t ratchetAmount{0};
138 uint32_t noteProbability{4294967295u};
139 uint32_t bassProbability{0};
140 uint32_t swapProbability{0};
141 uint32_t glideProbability{0};
142 uint32_t reverseProbability{0};
143 uint32_t chordProbability{0};
144 uint32_t ratchetProbability{0};
145 uint32_t spreadVelocity{0};
146 uint32_t spreadGate{0};
147 uint32_t spreadOctave{0};
157 outputMemberChannel.fill(MIDI_CHANNEL_NONE);
158 noteCodeOnPostArp.fill(ARP_NOTE_NONE);
159 noteStatus.fill(ArpNoteStatus::OFF);
162 return std::ranges::any_of(noteStatus, [](ArpNoteStatus status) {
return status == ArpNoteStatus::PENDING; });
164 void resetPostArpArrays() {
165 outputMemberChannel.fill(MIDI_CHANNEL_NONE);
166 noteCodeOnPostArp.fill(ARP_NOTE_NONE);
167 noteStatus.fill(ArpNoteStatus::OFF);
169 int16_t inputCharacteristics[2]{};
171 int16_t mpeValues[kNumExpressionDimensions]{};
173 uint8_t baseVelocity{0};
176 std::array<ArpNoteStatus, ARP_MAX_INSTRUCTION_NOTES> noteStatus{};
177 std::array<uint8_t, ARP_MAX_INSTRUCTION_NOTES> outputMemberChannel{};
178 std::array<int16_t, ARP_MAX_INSTRUCTION_NOTES> noteCodeOnPostArp{};
185class ArpReturnInstruction {
187 ArpReturnInstruction() {
188 sampleSyncLengthOn = 0;
189 invertReversed =
false;
191 outputMIDIChannelOff.fill(MIDI_CHANNEL_NONE);
192 noteCodeOffPostArp.fill(ARP_NOTE_NONE);
193 glideOutputMIDIChannelOff.fill(MIDI_CHANNEL_NONE);
194 glideNoteCodeOffPostArp.fill(ARP_NOTE_NONE);
199 uint32_t sampleSyncLengthOn;
209 std::array<uint8_t, ARP_MAX_INSTRUCTION_NOTES> outputMIDIChannelOff;
210 std::array<int16_t, ARP_MAX_INSTRUCTION_NOTES> noteCodeOffPostArp;
211 std::array<uint8_t, ARP_MAX_INSTRUCTION_NOTES> glideOutputMIDIChannelOff;
212 std::array<int16_t, ARP_MAX_INSTRUCTION_NOTES> glideNoteCodeOffPostArp;
215class ArpeggiatorBase {
217 virtual ~ArpeggiatorBase() =
default;
219 glideNoteCodeCurrentlyOnPostArp.fill(ARP_NOTE_NONE);
220 outputMIDIChannelForGlideNoteCurrentlyOnPostArp.fill(0);
230 uint32_t gateThreshold, uint32_t phaseIncrement);
232 bool currentlyPlayingReversed);
234 virtual bool hasAnyInputNotesActive() = 0;
235 virtual void reset() = 0;
236 virtual ArpType getArpType() = 0;
239 std::array<int16_t, ARP_MAX_INSTRUCTION_NOTES> glideNoteCodeCurrentlyOnPostArp;
240 std::array<uint8_t, ARP_MAX_INSTRUCTION_NOTES> outputMIDIChannelForGlideNoteCurrentlyOnPostArp;
241 uint32_t gatePos = 0;
242 uint8_t lastVelocity = 0;
250 uint32_t maxSequenceLength, uint32_t rhythm,
bool* shouldCarryOnRhythmNote,
251 bool* shouldPlayNote,
bool* shouldPlayBassNote,
bool* shouldPlayRandomStep,
252 bool* shouldPlayReverseNote,
bool* shouldPlayChordNote);
253 void increasePatternIndexes(uint8_t numStepRepeats);
254 void increaseSequenceIndexes(uint32_t maxSequenceLength, uint32_t rhythm);
256 bool evaluateRhythm(uint32_t rhythm,
bool isRatchet);
257 bool evaluateNoteProbability(
bool isRatchet);
258 bool evaluateBassProbability(
bool isRatchet);
259 bool evaluateSwapProbability(
bool isRatchet);
260 bool evaluateReverseProbability(
bool isRatchet);
261 bool evaluateChordProbability(
bool isRatchet);
262 uint32_t calculateSpreadVelocity(uint8_t velocity, int32_t spreadVelocityForCurrentStep);
266 bool getRandomProbabilityResult(uint32_t value);
267 int8_t getRandomBipolarProbabilityAmount(uint32_t value);
268 int8_t getRandomWeighted2BitsAmount(uint32_t value);
270 bool gateCurrentlyActive =
false;
272 bool playedFirstArpeggiatedNoteYet =
false;
275 uint32_t notesPlayedFromSequence = 0;
276 uint32_t randomNotesPlayedFromOctave = 0;
279 int16_t whichNoteCurrentlyOnPostArp;
280 int8_t currentOctave = 0;
281 int8_t currentDirection = 1;
282 int8_t currentOctaveDirection = 1;
285 uint32_t notesPlayedFromRhythm = 0;
286 uint32_t lastNormalNotePlayedFromRhythm = 0;
289 uint32_t notesPlayedFromLockedRandomizer = 0;
292 bool lastNormalNotePlayedFromNoteProbability =
true;
295 bool lastNormalNotePlayedFromBassProbability =
false;
298 bool lastNormalNotePlayedFromSwapProbability =
false;
301 bool lastNormalNotePlayedFromReverseProbability =
false;
304 bool lastNormalNotePlayedFromChordProbability =
false;
307 int32_t stepRepeatIndex = 0;
310 uint32_t ratchetNotesIndex = 0;
311 uint32_t ratchetNotesMultiplier = 0;
312 uint32_t ratchetNotesCount = 0;
313 bool isRatcheting =
false;
316 bool glideOnNextNoteOff =
false;
319 bool isPlayNoteForCurrentStep =
true;
320 bool isPlayBassForCurrentStep =
false;
321 bool isPlayRandomStepForCurrentStep =
false;
322 bool isPlayReverseForCurrentStep =
false;
323 bool isPlayChordForCurrentStep =
false;
324 bool isPlayRatchetForCurrentStep =
false;
325 bool isPlayGlideForCurrentStep =
false;
326 int32_t spreadVelocityForCurrentStep = 0;
327 int32_t spreadGateForCurrentStep = 0;
328 int32_t spreadOctaveForCurrentStep = 0;
329 bool resetLockedRandomizerValuesNextTime =
false;