Deluge Firmware 1.3.0
Build date: 2026.05.13
Loading...
Searching...
No Matches
song.h
1/*
2 * Copyright © 2014-2023 Synthstrom Audible Limited
3 *
4 * This file is part of The Synthstrom Audible Deluge Firmware.
5 *
6 * The Synthstrom Audible Deluge Firmware is free software: you can redistribute it and/or modify it under the
7 * terms of the GNU General Public License as published by the Free Software Foundation,
8 * either version 3 of the License, or (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
11 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12 * See the GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License along with this program.
15 * If not, see <https://www.gnu.org/licenses/>.
16 */
17
18#pragma once
19
20#include "definitions_cxx.hpp"
21#include "gui/menu_item/reverb/model.h"
22#include "io/midi/learned_midi.h"
23#include "model/clip/clip.h"
24#include "model/clip/clip_array.h"
25#include "model/global_effectable/global_effectable_for_song.h"
26#include "model/instrument/instrument.h"
27#include "model/output.h"
28#include "model/scale/musical_key.h"
29#include "model/scale/note_set.h"
30#include "model/scale/preset_scales.h"
31#include "model/scale/scale_change.h"
32#include "model/scale/scale_mapper.h"
33#include "model/sync.h"
34#include "model/timeline_counter.h"
35#include "modulation/params/param.h"
36#include "modulation/params/param_manager.h"
37#include "storage/flash_storage.h"
38#include "util/container/array/ordered_resizeable_array_with_multi_word_key.h"
39#include "util/d_string.h"
40
41class MidiCommand;
42class Clip;
43class AudioClip;
44class Instrument;
45class InstrumentClip;
46class Synth;
48class Instrument;
50class Drum;
51class SoundInstrument;
52class SoundDrum;
53class Action;
54class ArrangementRow;
57class Kit;
58class MIDIInstrument;
59class NoteRow;
60class Output;
61class AudioOutput;
62class ModelStack;
64Clip* getCurrentClip();
65InstrumentClip* getCurrentInstrumentClip();
66AudioClip* getCurrentAudioClip();
67Output* getCurrentOutput();
68Kit* getCurrentKit();
69Instrument* getCurrentInstrument();
70OutputType getCurrentOutputType();
71
72class Section {
73public:
74 LearnedMIDI launchMIDICommand;
75 int16_t numRepetitions;
76
77 Section() { numRepetitions = 0; }
78};
79
81 ModControllableAudio* modControllable;
82 Clip* clip;
83 ParamManager paramManager;
84};
85
86#define MAX_NOTES_CHORD_MEM 10
87
88enum SessionMacroKind : int8_t {
89 NO_MACRO = 0,
90 CLIP_LAUNCH,
91 OUTPUT_CYCLE,
92 SECTION,
93 NUM_KINDS,
94};
95
97 SessionMacroKind kind{NO_MACRO};
98 Clip* clip{nullptr};
99 Output* output{nullptr};
100 uint8_t section{0};
101};
102
103class Song final : public TimelineCounter {
104public:
105 Song();
106 ~Song() override;
107 bool mayDoubleTempo();
108 bool ensureAtLeastOneSessionClip();
109 void transposeAllScaleModeClips(int32_t interval);
110 void transposeAllScaleModeClips(int32_t offset, bool chromatic);
111 bool anyScaleModeClips();
112 void changeMusicalMode(uint8_t yVisualWithinOctave, int8_t change);
113 void rotateMusicalMode(int8_t change);
114 void replaceMusicalMode(const ScaleChange& changes, bool affectMIDITranspose);
115 int32_t getYVisualFromYNote(int32_t yNote, bool inKeyMode);
116 int32_t getYVisualFromYNote(int32_t yNote, bool inKeyMode, const MusicalKey& key);
117 int32_t incrementYNoteInKey(int32_t yNote, int32_t increment, bool inOctave) const;
118 static int32_t incrementYNoteInKey(int32_t yNote, int32_t increment, bool inOctave, const MusicalKey& key);
119 int32_t getYNoteFromYVisual(int32_t yVisual, bool inKeyMode);
120 int32_t getYNoteFromYVisual(int32_t yVisual, bool inKeyMode, const MusicalKey& key);
121 bool mayMoveModeNote(int16_t yVisualWithinOctave, int8_t newOffset);
122 ParamManagerForTimeline* findParamManagerForDrum(Kit* kit, Drum* drum, Clip* stopTraversalAtClip = nullptr);
123 void setupPatchingForAllParamManagersForDrum(SoundDrum* drum);
124 void setupPatchingForAllParamManagersForInstrument(SoundInstrument* sound);
125 void grabVelocityToLevelFromMIDICableAndSetupPatchingForAllParamManagersForInstrument(MIDICable& cable,
126 SoundInstrument* instrument);
127 void grabVelocityToLevelFromMIDICableAndSetupPatchingForAllParamManagersForDrum(MIDICable& cable, SoundDrum* drum,
128 Kit* kit);
129 void grabVelocityToLevelFromMIDICableAndSetupPatchingForEverything(MIDICable& cable);
130 void getCurrentRootNoteAndScaleName(StringBuf& buffer);
131 void displayCurrentRootNoteAndScaleName();
132
133 // Scale-related methods
134
136 Scale cycleThroughScales();
138 Scale getCurrentScale();
141 Scale setScale(Scale newScale);
144 bool setScaleNotes(NoteSet newScale);
148 bool hasUserScale();
151 void setRootNote(int32_t newRootNote, InstrumentClip* clipToAvoidAdjustingScrollFor = nullptr);
154
155 void setTempoFromNumSamples(double newTempoSamples, bool shouldLogAction);
156 void setupDefault();
157 void setBPM(float tempoBPM, bool shouldLogAction);
158 void setTempoFromParams(int32_t magnitude, int8_t whichValue, bool shouldLogAction);
159 void deleteSoundsWhichWontSound();
160 void writeTemplateSong(const char* templateSong);
161 void
162 deleteClipObject(Clip* clip, bool songBeingDestroyedToo = false,
163 InstrumentRemoval instrumentRemovalInstruction = InstrumentRemoval::DELETE_OR_HIBERNATE_IF_UNUSED);
164 int32_t getMaxMIDIChannelSuffix(int32_t channel);
165 void addOutput(Output* output, bool atStart = true);
166 void deleteOutputThatIsInMainList(Output* output, bool stopAnyAuditioningFirst = true);
167 void markAllInstrumentsAsEdited();
168 Instrument* getInstrumentFromPresetSlot(OutputType outputType, int32_t presetNumber, int32_t presetSubSlotNumber,
169 char const* name, char const* dirPath, bool searchHibernatingToo = true,
170 bool searchNonHibernating = true);
171 AudioOutput* getAudioOutputFromName(std::string_view name);
172 void setupPatchingForAllParamManagers();
173 void replaceInstrument(Instrument* oldInstrument, Instrument* newInstrument, bool keepNoteRowsWithMIDIInput = true);
174 void stopAllMIDIAndGateNotesPlaying();
175 void stopAllAuditioning();
176 void deleteOrHibernateOutput(Output* output);
177 Instrument* getNonAudioInstrumentToSwitchTo(OutputType newOutputType, Availability availabilityRequirement,
178 int16_t newSlot, int8_t newSubSlot, bool* instrumentWasAlreadyInSong);
179 void removeSessionClipLowLevel(Clip* clip, int32_t clipIndex);
180 void changeSwingInterval(int32_t newValue);
181 int32_t convertSyncLevelFromFileValueToInternalValue(int32_t fileValue);
182 int32_t convertSyncLevelFromInternalValueToFileValue(int32_t internalValue);
183 String getSongFullPath();
184 void setSongFullPath(const char* fullPath);
185 int32_t getInputTickMagnitude() const { return insideWorldTickMagnitude + insideWorldTickMagnitudeOffsetFromBPM; }
186
187 GlobalEffectableForSong globalEffectable;
188
189 ClipArray sessionClips;
190 ClipArray arrangementOnlyClips;
191
192 Output* firstOutput;
194 firstHibernatingInstrument; // All Instruments have inValidState set to false when they're added to this list
195
196 OrderedResizeableArrayWithMultiWordKey backedUpParamManagers;
197
198 uint32_t xZoom[2]; // Set default zoom at max zoom-out;
199 int32_t xScroll[2]; // Leave this as signed
200 int32_t xScrollForReturnToSongView;
201 int32_t xZoomForReturnToSongView;
202 bool tripletsOn;
203 uint32_t tripletsLevel; // The number of ticks in one of the three triplets
204
205 uint64_t timePerTimerTickBig;
206 int32_t divideByTimePerTimerTick;
207
208 // How many orders of magnitude faster internal ticks are going than input ticks. Used in combination with
209 // inputTickScale, which is usually 1, but is different if there's an inputTickScaleClip. So, e.g. if
210 // insideWorldTickMagnitude is 1, this means the inside world is spinning twice as fast as the external world, so
211 // MIDI sync coming in representing an 8th-note would be interpreted internally as a quarter-note (because two
212 // internal 8th-notes would have happened, twice as fast, making a quarter-note)
213 int32_t insideWorldTickMagnitude;
214
215 // Sometimes, we'll do weird stuff to insideWorldTickMagnitude for sync-scaling, which would make BPM values look
216 // weird. So, we keep insideWorldTickMagnitudeOffsetFromBPM
217 int32_t insideWorldTickMagnitudeOffsetFromBPM;
218
219 int8_t swingAmount;
220 uint8_t swingInterval;
221
222 Section sections[kMaxNumSections];
223
224 MusicalKey key;
225 std::bitset<NUM_PRESET_SCALES> disabledPresetScales;
226
227 uint16_t slot;
228 int8_t subSlot;
229 String name;
230
231 bool affectEntire;
232
233 SessionLayoutType sessionLayout = FlashStorage::defaultSessionLayout;
234 int32_t songGridScrollX = 0;
235 int32_t songGridScrollY = 0;
236 int32_t songViewYScroll;
237 int32_t arrangementYScroll;
238
239 uint8_t sectionToReturnToAfterSongEnd;
240
241 bool wasLastInArrangementEditor;
242 int32_t lastClipInstanceEnteredStartPos; // -1 means we are not "inside" an arrangement. While we're in the
243 // ArrangementEditor, it's 0
244
245 bool arrangerAutoScrollModeActive;
246
247 MIDIInstrument* hibernatingMIDIInstrument;
248
249 bool outputClipInstanceListIsCurrentlyInvalid; // Set to true during scenarios like replaceInstrument(), to warn
250 // other functions not to look at Output::clipInstances
251
252 bool paramsInAutomationMode;
253
254 bool inClipMinderViewOnLoad; // Temp variable only valid while loading Song
255
256 int32_t unautomatedParamValues[deluge::modulation::params::kMaxNumUnpatchedParams];
257
258 String dirPath;
259
260 std::array<SessionMacro, 8> sessionMacros{};
261
262 bool getAnyClipsSoloing() const;
263 Clip* getCurrentClip();
264 void setCurrentClip(Clip* clip) {
265 if (currentClip != nullptr) {
266 previousClip = currentClip;
267 }
268 currentClip = clip;
269 }
270 uint32_t getInputTickScale();
271 Clip* getSyncScalingClip();
272 void setInputTickScaleClip(Clip* clip);
273 inline bool isFillModeActive() { return fillModeActive; }
274 void changeFillMode(bool on);
275 void loadNextSong();
276 void setClipLength(Clip* clip, uint32_t newLength, Action* action, bool mayReSyncClip = true);
277 void doubleClipLength(InstrumentClip* clip, Action* action = nullptr);
278 Clip* getClipWithOutput(Output* output, bool mustBeActive = false, Clip* excludeClip = nullptr);
279 Error readFromFile(Deserializer& reader);
280 void writeToFile();
281 void loadAllSamples(bool mayActuallyReadFiles = true);
282 void renderAudio(std::span<StereoSample> outputBuffer, int32_t* reverbBuffer, int32_t sideChainHitPending);
283 bool isYNoteAllowed(int32_t yNote, bool inKeyMode);
284 Clip* syncScalingClip;
285 void setTimePerTimerTick(uint64_t newTimeBig, bool shouldLogAction = false);
286 bool hasAnySwing();
287 void resyncLFOs();
288 void ensureInaccessibleParamPresetValuesWithoutKnobsAreZero(Sound* sound);
289 bool areAllClipsInSectionPlaying(int32_t section);
290 void removeYNoteFromMode(int32_t yNoteWithinOctave);
291 void turnSoloingIntoJustPlaying(bool getRidOfArmingToo = true);
292 void reassessWhetherAnyClipsSoloing();
293 float getTimePerTimerTickFloat();
294 uint32_t getTimePerTimerTickRounded();
295 int32_t getNumOutputs();
296 Clip* getNextSessionClipWithOutput(int32_t offset, Output* output, Clip* prevClip);
297 bool anyClipsSoloing;
298
299 ParamManager* getBackedUpParamManagerForExactClip(ModControllableAudio* modControllable, Clip* clip,
300 ParamManager* stealInto = nullptr);
301 ParamManager* getBackedUpParamManagerPreferablyWithClip(ModControllableAudio* modControllable, Clip* clip,
302 ParamManager* stealInto = nullptr);
303 void backUpParamManager(ModControllableAudio* modControllable, Clip* clip, ParamManagerForTimeline* paramManager,
304 bool shouldStealExpressionParamsToo = false);
305 void moveInstrumentToHibernationList(Instrument* instrument);
306 void deleteOrHibernateOutputIfNoClips(Output* output);
307 void removeInstrumentFromHibernationList(Instrument* instrument);
308 bool doesOutputHaveActiveClipInSession(Output* output);
309 bool doesNonAudioSlotHaveActiveClipInSession(OutputType outputType, int32_t slot, int32_t subSlot = -1);
310 bool doesNonAudioSlotHaveClipInSession(OutputType outputType, int32_t slot, int32_t subSlot);
311 bool doesOutputHaveAnyClips(Output* output);
312 void deleteBackedUpParamManagersForClip(Clip* clip);
313 void deleteBackedUpParamManagersForModControllable(ModControllableAudio* modControllable);
314 void deleteHibernatingInstrumentWithSlot(OutputType outputType, char const* name);
315 void loadCrucialSamplesOnly();
316 Clip* getSessionClipWithOutput(Output* output, int32_t requireSection = -1, Clip* excludeClip = nullptr,
317 int32_t* clipIndex = nullptr, bool excludePendingOverdubs = false);
318 void restoreClipStatesBeforeArrangementPlay();
319 void deleteOrAddToHibernationListOutput(Output* output);
320 int32_t getLowestSectionWithNoSessionClipForOutput(Output* output);
321 void assertActiveness(ModelStackWithTimelineCounter* modelStack, int32_t endInstanceAtTime = -1);
322 [[nodiscard]] bool isClipActive(Clip const* clip) const;
323 void sendAllMIDIPGMs();
324 void sortOutWhichClipsAreActiveWithoutSendingPGMs(ModelStack* modelStack, int32_t playbackWillStartInArrangerAtPos);
325 void deactivateAnyArrangementOnlyClips();
326 Clip* getLongestClip(bool includePlayDisabled, bool includeArrangementOnly);
327 Clip* getLongestActiveClipWithMultipleOrFactorLength(int32_t targetLength, bool revertToAnyActiveClipIfNone = true,
328 Clip* excludeClip = nullptr);
329 int32_t getOutputIndex(Output* output);
330 void setHibernatingMIDIInstrument(MIDIInstrument* newInstrument);
331 void deleteHibernatingMIDIInstrument();
332 MIDIInstrument* grabHibernatingMIDIInstrument(int32_t newSlot, int32_t newSubSlot);
333 NoteRow* findNoteRowForDrum(Kit* kit, Drum* drum, Clip* stopTraversalAtClip = nullptr);
334
335 bool anyOutputsSoloingInArrangement;
336 bool getAnyOutputsSoloingInArrangement();
337 void reassessWhetherAnyOutputsSoloingInArrangement();
338 bool isOutputActiveInArrangement(Output* output);
339 Output* getOutputFromIndex(int32_t index);
340 void ensureAllInstrumentsHaveAClipOrBackedUpParamManager(char const* errorMessageNormal,
341 char const* errorMessageHibernating);
342 Error placeFirstInstancesOfActiveClips(int32_t pos);
343 void endInstancesOfActiveClips(int32_t pos, bool detachClipsToo = false);
344 void clearArrangementBeyondPos(int32_t pos, Action* action);
345 void deletingClipInstanceForClip(Output* output, Clip* clip, Action* action, bool shouldPickNewActiveClip);
346 bool arrangementHasAnyClipInstances();
347 void resumeClipsClonedForArrangementRecording();
348 void setParamsInAutomationMode(bool newState);
349 bool shouldOldOutputBeReplaced(Clip* clip, Availability* availabilityRequirement = nullptr);
350 Output* navigateThroughPresetsForInstrument(Output* output, int32_t offset);
351 void instrumentSwapped(Instrument* newInstrument);
352 Instrument* changeOutputType(Instrument* oldInstrument, OutputType newOutputType);
353 AudioOutput* getFirstAudioOutput();
354 AudioOutput* createNewAudioOutput(Output* replaceOutput = nullptr);
356 void getNoteLengthName(StringBuf& buffer, uint32_t noteLength, char const* notesString = "-notes",
357 bool clarifyPerColumn = false) const;
358 void replaceOutputLowLevel(Output* newOutput, Output* oldOutput);
359 void removeSessionClip(Clip* clip, int32_t clipIndex, bool forceClipsAboveToMoveVertically = false);
360 bool deletePendingOverdubs(Output* onlyWithOutput = nullptr, int32_t* originalClipIndex = nullptr,
361 bool createConsequencesForOtherLinearlyRecordingClips = false);
362 Clip* getPendingOverdubWithOutput(Output* output);
363 Clip* getClipWithOutputAboutToBeginLinearRecording(Output* output);
364 Clip* createPendingNextOverdubBelowClip(Clip* clip, int32_t clipIndex, OverDubType newOverdubNature);
365 bool hasAnyPendingNextOverdubs();
366 Output* getNextAudioOutput(int32_t offset, Output* oldOutput, Availability availabilityRequirement);
367 void deleteOutput(Output* output);
368 void cullAudioClipVoice();
369 int32_t getYScrollSongViewWithoutPendingOverdubs();
370 int32_t removeOutputFromMainList(Output* output, bool stopAnyAuditioningFirst = true);
371 void swapClips(Clip* newClip, Clip* oldClip, int32_t clipIndex);
372 Clip* replaceInstrumentClipWithAudioClip(Clip* oldClip, int32_t clipIndex);
373 void setDefaultVelocityForAllInstruments(uint8_t newDefaultVelocity);
374 void midiCableBendRangeUpdatedViaMessage(ModelStack* modelStack, MIDICable& cable, int32_t channelOrZone,
375 int32_t whichBendRange, int32_t bendSemitones);
376 Error addInstrumentsToFileItems(OutputType outputType);
377
378 uint32_t getQuarterNoteLength();
379 uint32_t getBarLength();
380 uint32_t getSixteenthNoteLength();
381 ModelStackWithThreeMainThings* setupModelStackWithSongAsTimelineCounter(void* memory);
382 ModelStackWithTimelineCounter* setupModelStackWithCurrentClip(void* memory);
383 ModelStackWithThreeMainThings* addToModelStack(ModelStack* modelStack);
386 ModelStackWithAutoParam* getModelStackWithParam(ModelStackWithThreeMainThings* modelStack, int32_t paramID);
387
388 // TimelineCounter implementation
389 [[nodiscard]] int32_t getLastProcessedPos() const override;
390 [[nodiscard]] uint32_t getLivePos() const override;
391 [[nodiscard]] int32_t getLoopLength() const override;
392 [[nodiscard]] bool isPlayingAutomationNow() const override;
393 [[nodiscard]] bool backtrackingCouldLoopBackToEnd() const override;
394 [[nodiscard]] int32_t getPosAtWhichPlaybackWillCut(ModelStackWithTimelineCounter const* modelStack) const override;
395 void getActiveModControllable(ModelStackWithTimelineCounter* modelStack) override;
396 void expectEvent() override;
397 TimelineCounter* getTimelineCounterToRecordTo() override;
398
399 // Reverb params to be stored here between loading and song being made the active one
400 dsp::Reverb::Model model;
401 float reverbRoomSize;
402 float reverbHPF;
403 float reverbLPF;
404 float reverbDamp;
405 float reverbWidth;
406 int32_t reverbPan;
407 int32_t reverbSidechainVolume;
408 int32_t reverbSidechainShape;
409 int32_t reverbSidechainAttack;
410 int32_t reverbSidechainRelease;
411 SyncLevel reverbSidechainSync;
412
413 // START ~ new Automation Arranger View Variables
414 int32_t lastSelectedParamID; // last selected Parameter to be edited in Automation Arranger View
416 lastSelectedParamKind; // 0 = patched, 1 = unpatched, 2 = global effectable, 3 = none
417 int32_t lastSelectedParamShortcutX;
418 int32_t lastSelectedParamShortcutY;
419 int32_t lastSelectedParamArrayPosition;
420 // END ~ new Automation Arranger View Variables
421
422 // Song level transpose control (encoder actions)
423 void commandTranspose(int32_t interval);
424 int32_t masterTransposeInterval = 0;
425 void transpose(int32_t interval);
426 void adjustMasterTransposeInterval(int32_t interval);
427 void displayMasterTransposeInterval();
428
429 // MIDI controlled song transpose
430 bool hasBeenTransposed = 0;
431 int16_t transposeOffset = 0;
432
433 int32_t countAudioClips() const;
434
435 // Chord memory
436 uint8_t chordMemNoteCount[kDisplayHeight] = {0};
437 uint8_t chordMem[kDisplayHeight][MAX_NOTES_CHORD_MEM] = {0};
438
439 // Tempo automation
440 void clearTempoAutomation();
441 void updateBPMFromAutomation();
442
443 float calculateBPM() {
444 float timePerTimerTick = getTimePerTimerTickFloat();
445 return calculateBPM(timePerTimerTick);
446 }
447 float calculateBPM(float timePerTimerTick) {
448
449 if (insideWorldTickMagnitude > 0) {
450 timePerTimerTick *= ((uint32_t)1 << (insideWorldTickMagnitude));
451 }
452 float tempoBPM = (float)110250 / timePerTimerTick;
453 if (insideWorldTickMagnitude < 0) {
454 tempoBPM *= ((uint32_t)1 << (-insideWorldTickMagnitude));
455 }
456 return tempoBPM;
457 }
458
459 int8_t defaultAudioClipOverdubOutputCloning = -1; // -1 means no default set
460
461 // Threshold
462 void changeThresholdRecordingMode(int8_t offset);
463 void displayThresholdRecordingMode();
464 ThresholdRecordingMode thresholdRecordingMode;
465
466private:
467 ScaleMapper scaleMapper;
468 NoteSet userScaleNotes;
469 bool fillModeActive;
470 Clip* currentClip = nullptr;
471 Clip* previousClip = nullptr; // for future use, maybe finding an instrument clip or something
472 void inputTickScalePotentiallyJustChanged(uint32_t oldScale);
473 Error readClipsFromFile(Deserializer& reader, ClipArray* clipArray);
474 void addInstrumentToHibernationList(Instrument* instrument);
475 void deleteAllBackedUpParamManagers(bool shouldAlsoEmptyVector = true);
476 void deleteAllBackedUpParamManagersWithClips();
477 void deleteAllOutputs(Output** prevPointer);
478 void setupClipIndexesForSaving();
479 void setBPMInner(float tempoBPM, bool shouldLogAction);
480 void clearTempoAutomation(float tempoBPM);
481 int32_t intBPM{0};
482};
483
484extern Song* currentSong;
485extern Song* preLoadedSong;
Definition action.h:75
Definition arpeggiator.h:48
Definition audio_clip.h:35
Definition audio_output.h:37
Definition clip_array.h:25
Definition clip.h:46
Definition drum.h:44
Definition global_effectable_for_song.h:22
Definition instrument_clip.h:48
Definition instrument.h:45
Definition kit.h:34
Definition learned_midi.h:30
A MIDI cable connection. Stores all state specific to a given cable and its contained ports and chann...
Definition midi_device.h:96
Definition midi_instrument.h:37
Definition mod_controllable_audio.h:47
Definition model_stack.h:129
Definition model_stack.h:123
Definition musical_key.h:10
Definition note_row.h:99
Definition note_set.h:20
Definition ordered_resizeable_array_with_multi_word_key.h:25
Definition output.h:81
Definition param_manager.h:174
Definition param_manager.h:45
Definition scale_change.h:5
Definition song.h:72
Definition song.h:103
Scale getCurrentScale()
Returns current scale.
Definition song.cpp:3040
Scale setScale(Scale newScale)
Returns CUSTOM_SCALE_WITH_MORE_THAN_7_NOTES we can't use the newScale.
Definition song.cpp:2992
void learnScaleFromCurrentNotes()
Learns a user scale from notes in scale mode clips.
ModelStackWithAutoParam * getModelStackWithParam(ModelStackWithThreeMainThings *modelStack, int32_t paramID)
Definition song.cpp:5964
void setRootNote(int32_t newRootNote, InstrumentClip *clipToAvoidAdjustingScrollFor=nullptr)
Definition song.cpp:582
bool hasUserScale()
Returns true if the song has a user scale.
Definition song.cpp:3036
void getNoteLengthName(StringBuf &buffer, uint32_t noteLength, char const *notesString="-notes", bool clarifyPerColumn=false) const
buffer must have at least 5 characters on 7seg, or 30 for OLED
Definition song.cpp:5214
NoteSet notesInScaleModeClips()
Returns a NoteSet with all notes currently in used in scale mdoe clips.
Definition song.cpp:572
int32_t getLastProcessedPos() const override
Get the tick at which this timeline counter last did anything.
Definition song.cpp:5724
bool setScaleNotes(NoteSet newScale)
Definition song.cpp:3005
uint32_t getLivePos() const override
Get the current tick of this timeline counter relative to the playback handler.
Definition song.cpp:5712
Scale cycleThroughScales()
Changes to next applicable scale.
Definition song.cpp:2976
Definition sound_drum.h:28
Definition sound_instrument.h:30
Definition d_stringbuf.h:16
Definition d_string.h:41
Kind
Definition param.h:42
Definition song.h:80
Definition song.h:96