From e0924c3df397f8e1f16390366c6843a5feba1e18 Mon Sep 17 00:00:00 2001 From: Paul Date: Thu, 9 Sep 2021 12:33:01 -0400 Subject: [PATCH] Handle Midi at the sample where it is realized (#5035) My original rough draft handled midi all up front. This now interlaves the midi iteration with the sample iteration so midi should arrive at the right block. Closes #4983 --- src/surge_synth_juce/SurgeSynthProcessor.cpp | 114 ++++++++++++------- src/surge_synth_juce/SurgeSynthProcessor.h | 1 + 2 files changed, 72 insertions(+), 43 deletions(-) diff --git a/src/surge_synth_juce/SurgeSynthProcessor.cpp b/src/surge_synth_juce/SurgeSynthProcessor.cpp index 4d0e002ed6d..a358bf873f7 100644 --- a/src/surge_synth_juce/SurgeSynthProcessor.cpp +++ b/src/surge_synth_juce/SurgeSynthProcessor.cpp @@ -216,49 +216,6 @@ void SurgeSynthProcessor::processBlock(AudioBuffer &buffer, MidiBuffer &m surge->resetStateFromTimeData(); } - for (const MidiMessageMetadata it : midiMessages) - { - MidiMessage m = it.getMessage(); - const int ch = m.getChannel() - 1; - juce::ScopedValueSetter midiAdd(isAddingFromMidi, true); - midiKeyboardState.processNextMidiEvent(m); - - if (m.isNoteOn()) - { - surge->playNote(ch, m.getNoteNumber(), m.getVelocity(), 0); - } - else if (m.isNoteOff()) - { - surge->releaseNote(ch, m.getNoteNumber(), m.getVelocity()); - } - else if (m.isChannelPressure()) - { - surge->channelAftertouch(ch, m.getChannelPressureValue()); - } - else if (m.isAftertouch()) - { - surge->polyAftertouch(ch, m.getNoteNumber(), m.getAfterTouchValue()); - } - else if (m.isPitchWheel()) - { - surge->pitchBend(ch, m.getPitchWheelValue() - 8192); - } - else if (m.isController()) - { - surge->channelController(ch, m.getControllerNumber(), m.getControllerValue()); - } - else if (m.isProgramChange()) - { - // apparently this is not enough to actually execute SurgeSynthesizer::programChange - // for whatever reason! - surge->programChange(ch, m.getProgramChangeNumber()); - } - else - { - // std::cout << "Ignoring message " << std::endl; - } - } - midiR rec; while (midiFromGUI.pop(rec)) { @@ -280,8 +237,29 @@ void SurgeSynthProcessor::processBlock(AudioBuffer &buffer, MidiBuffer &m auto mainInput = getBusBuffer(buffer, true, 0); auto sceneAOutput = getBusBuffer(buffer, false, 1); auto sceneBOutput = getBusBuffer(buffer, false, 2); + + auto midiIt = midiMessages.findNextSamplePosition(0); + int nextMidi = -1; + if (midiIt != midiMessages.cend()) + { + nextMidi = (*midiIt).samplePosition; + } + for (int i = 0; i < buffer.getNumSamples(); i++) { + if (i == nextMidi) + { + applyMidi(*midiIt); + midiIt++; + if (midiIt == midiMessages.cend()) + { + nextMidi = -1; + } + else + { + nextMidi = (*midiIt).samplePosition; + } + } auto outL = mainOutput.getWritePointer(0, i); auto outR = mainOutput.getWritePointer(1, i); @@ -333,6 +311,13 @@ void SurgeSynthProcessor::processBlock(AudioBuffer &buffer, MidiBuffer &m blockPos = (blockPos + 1) & (BLOCK_SIZE - 1); } + // This should, in theory, never happen, but better safe than sorry + while (midiIt != midiMessages.cend()) + { + applyMidi(*midiIt); + midiIt++; + } + if (checkNamesEvery++ > 10) { checkNamesEvery = 0; @@ -343,6 +328,49 @@ void SurgeSynthProcessor::processBlock(AudioBuffer &buffer, MidiBuffer &m } } +void SurgeSynthProcessor::applyMidi(const juce::MidiMessageMetadata &it) +{ + MidiMessage m = it.getMessage(); + const int ch = m.getChannel() - 1; + juce::ScopedValueSetter midiAdd(isAddingFromMidi, true); + midiKeyboardState.processNextMidiEvent(m); + + if (m.isNoteOn()) + { + surge->playNote(ch, m.getNoteNumber(), m.getVelocity(), 0); + } + else if (m.isNoteOff()) + { + surge->releaseNote(ch, m.getNoteNumber(), m.getVelocity()); + } + else if (m.isChannelPressure()) + { + surge->channelAftertouch(ch, m.getChannelPressureValue()); + } + else if (m.isAftertouch()) + { + surge->polyAftertouch(ch, m.getNoteNumber(), m.getAfterTouchValue()); + } + else if (m.isPitchWheel()) + { + surge->pitchBend(ch, m.getPitchWheelValue() - 8192); + } + else if (m.isController()) + { + surge->channelController(ch, m.getControllerNumber(), m.getControllerValue()); + } + else if (m.isProgramChange()) + { + // apparently this is not enough to actually execute SurgeSynthesizer::programChange + // for whatever reason! + surge->programChange(ch, m.getProgramChangeNumber()); + } + else + { + // std::cout << "Ignoring message " << std::endl; + } +} + //============================================================================== bool SurgeSynthProcessor::hasEditor() const { diff --git a/src/surge_synth_juce/SurgeSynthProcessor.h b/src/surge_synth_juce/SurgeSynthProcessor.h index c15cc59ee9e..fa1f5417e78 100644 --- a/src/surge_synth_juce/SurgeSynthProcessor.h +++ b/src/surge_synth_juce/SurgeSynthProcessor.h @@ -163,6 +163,7 @@ class SurgeSynthProcessor : public juce::AudioProcessor, bool canRemoveBusValue = false; bool canRemoveBus(bool isInput) const override { return canRemoveBusValue; } void processBlock(juce::AudioBuffer &, juce::MidiBuffer &) override; + void applyMidi(const juce::MidiMessageMetadata &); //============================================================================== juce::AudioProcessorEditor *createEditor() override;