AnalogTapeModel

Physical modelling signal processing for analog tape recording
Log | Files | Refs | Submodules | README | LICENSE

commit 81e8acb65682d35c1b82cfecadbc85b487f9fbd0
parent cc357a1dd6a3b686fd4eb77a0eb8a9c20e6d2b82
Author: jatinchowdhury18 <[email protected]>
Date:   Sun, 13 Sep 2020 15:57:58 -0700

Implement mix groups feature (#87)

* Set up basic mix group controlling with OSC (WIP)

* Completed first pass of mix groups feature

Co-authored-by: jatinchowdhury18 <[email protected]>
Diffstat:
MPlugin/CHOWTapeModel.jucer | 14++++++++++++++
MPlugin/Source/GUI/Assets/gui.xml | 6+++++-
APlugin/Source/MixGroups/MixGroupsController.cpp | 171+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
APlugin/Source/MixGroups/MixGroupsController.h | 45+++++++++++++++++++++++++++++++++++++++++++++
APlugin/Source/MixGroups/MixGroupsParamReceiver.cpp | 38++++++++++++++++++++++++++++++++++++++
APlugin/Source/MixGroups/MixGroupsParamReceiver.h | 29+++++++++++++++++++++++++++++
MPlugin/Source/PluginProcessor.cpp | 4+++-
MPlugin/Source/PluginProcessor.h | 3+++
8 files changed, 308 insertions(+), 2 deletions(-)

diff --git a/Plugin/CHOWTapeModel.jucer b/Plugin/CHOWTapeModel.jucer @@ -31,6 +31,16 @@ <FILE id="oFI05X" name="TooltipComp.cpp" compile="1" resource="0" file="Source/GUI/TooltipComp.cpp"/> <FILE id="BpGwTA" name="TooltipComp.h" compile="0" resource="0" file="Source/GUI/TooltipComp.h"/> </GROUP> + <GROUP id="{DCA33FEB-7151-818B-57ED-905929FEF150}" name="MixGroups"> + <FILE id="plu32m" name="MixGroupsController.cpp" compile="1" resource="0" + file="Source/MixGroups/MixGroupsController.cpp"/> + <FILE id="vwS6BX" name="MixGroupsController.h" compile="0" resource="0" + file="Source/MixGroups/MixGroupsController.h"/> + <FILE id="yc5D3k" name="MixGroupsParamReceiver.cpp" compile="1" resource="0" + file="Source/MixGroups/MixGroupsParamReceiver.cpp"/> + <FILE id="hGI0mn" name="MixGroupsParamReceiver.h" compile="0" resource="0" + file="Source/MixGroups/MixGroupsParamReceiver.h"/> + </GROUP> <GROUP id="{71C1FCA8-E7B0-3B66-1340-F140C452FF6F}" name="Presets"> <GROUP id="{AB6F221D-98B5-9782-2241-321BA5DFB83C}" name="PresetConfigs"> <FILE id="AymGjK" name="Default.xml" compile="0" resource="1" file="Source/Presets/PresetConfigs/Default.xml"/> @@ -118,6 +128,7 @@ <MODULEPATH id="juce_dsp" path="Juce/modules"/> <MODULEPATH id="foleys_gui_magic" path="."/> <MODULEPATH id="juce_opengl" path="Juce/modules"/> + <MODULEPATH id="juce_osc" path="Juce/modules"/> </MODULEPATHS> </XCODE_MAC> <VS2017 targetFolder="Builds/VisualStudio2017"> @@ -142,6 +153,7 @@ <MODULEPATH id="juce_dsp" path="Juce/modules"/> <MODULEPATH id="foleys_gui_magic" path="."/> <MODULEPATH id="juce_opengl" path="Juce/modules"/> + <MODULEPATH id="juce_osc" path="Juce/modules"/> </MODULEPATHS> </VS2017> <LINUX_MAKE targetFolder="Builds/LinuxMakefile"> @@ -165,6 +177,7 @@ <MODULEPATH id="juce_dsp" path="Juce/modules"/> <MODULEPATH id="foleys_gui_magic" path="."/> <MODULEPATH id="juce_opengl" path="Juce/modules"/> + <MODULEPATH id="juce_osc" path="Juce/modules"/> </MODULEPATHS> </LINUX_MAKE> </EXPORTFORMATS> @@ -185,6 +198,7 @@ <MODULE id="juce_gui_basics" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> <MODULE id="juce_gui_extra" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> <MODULE id="juce_opengl" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_osc" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> </MODULES> <LIVE_SETTINGS> <WINDOWS/> diff --git a/Plugin/Source/GUI/Assets/gui.xml b/Plugin/Source/GUI/Assets/gui.xml @@ -141,9 +141,13 @@ name="Hysteresis Mode" caption="Hysteresis Mode" caption-size="0" combo-text="FFEAA92C" caption-color="FFFFFFFF" max-height="100" margin="" parameter="mode" combo-background="00000000" tooltip="Selects the mode to use for hysteresis processing. Choose between 2nd/4th order Runge-Kutta method, 4 or 8 Newton-Raphson iterations, or revert to version 1.0."/> + <ComboBox lookAndFeel="ComboBoxLNF" padding="0" border="0" background-color="00000000" + name="Mix Group" caption="Mix Group" caption-size="0" flex-grow="0.9" + combo-text="FFEAA92C" caption-color="FFFFFFFF" max-height="100" margin="0" + parameter="mix_group" combo-background="00000000" tooltip="Adds this plugin to a mix group. If you add this plugin to a group, the parameter settings will be copied to all other plugins in the group, and their parameters will remain in sync."/> <presets margin="5" padding="0" background-color="00000000" border-color="595C6B" radius="" border="" lookAndFeel="ComboBoxLNF" tooltip="Selects a preset for the plugin." - flex-grow="1.8" max-height="100"/> + flex-grow="1.75" max-height="100"/> </View> </View> </magic> diff --git a/Plugin/Source/MixGroups/MixGroupsController.cpp b/Plugin/Source/MixGroups/MixGroupsController.cpp @@ -0,0 +1,171 @@ +#include "MixGroupsController.h" + +using namespace MixGroupsConstants; + +MixGroupsController::MixGroupsController (AudioProcessorValueTreeState& vts, + AudioProcessor* proc) : + vts (vts) +{ + // connect sender + sender.connect ("127.0.0.1", portNum); + + // load parameters + auto params = proc->getParameters(); + loadParameterList (params); + + // set up receiver + paramReceiver->loadReceiverListeners (paramList); + paramReceiver->addChangeListener (this); + + mixGroupParam = vts.getRawParameterValue (mixGroupParamID); +} + +void MixGroupsController::createParameterLayout (std::vector<std::unique_ptr<RangedAudioParameter>>& params) +{ + params.push_back (std::make_unique<AudioParameterChoice> (mixGroupParamID, "Mix Group", StringArray ({"N/A", "1", "2", "3", "4"}), 0)); +} + +void MixGroupsController::loadParameterList (Array<AudioProcessorParameter*>& params) +{ + // iterate over parameters + for (auto* param : params) + { + auto paramWithID = dynamic_cast<AudioProcessorParameterWithID*> (param); + + if (paramWithID == nullptr) + continue; + + auto paramID = paramWithID->paramID; + vts.addParameterListener (paramID, this); + + if (paramID != mixGroupParamID) + paramList.addIfNotAlreadyThere (paramID); + } +} + +void MixGroupsController::parameterChanged (const String& parameterID, float newValue) +{ + int mixGroup = (int) mixGroupParam->load(); + if (mixGroup == 0) // no mix group, don't bother sending + return; + + if (parameterID == mixGroupParamID) // mix group was changed + { + OSCMessage message (allParamsAddress); + message.addArgument (uuid.toString()); + message.addArgument (mixGroup); + + for (const auto& paramID : paramList) + { + auto param = vts.getParameter (paramID); + auto value = param->convertFrom0to1 (param->getValue()); + + message.addArgument (paramID); + message.addArgument (value); + } + + sender.send (message); + + return; + } + + if (! paramList.contains (parameterID)) // parameter is not in list + return; + + sendParameter (parameterID, mixGroup, newValue); +} + +void MixGroupsController::sendParameter (const String& paramID, int mixGroup, float value) +{ + String address = oscAddressPrefix + paramID; + String pluginID = uuid.toString(); + + sender.send (address, pluginID, mixGroup, value); +} + +void MixGroupsController::changeListenerCallback (ChangeBroadcaster* source) +{ + if (source != paramReceiver) + return; + + // get message from receiver, we already know this will be a valid message! + auto& message = paramReceiver->getOSCMessage(); + + // get address + auto address = message.getAddressPattern().toString(); + + // special case: all parameters + if (address == allParamsAddress) + { + parseAllParams (message); + return; + } + + // get paramID from address + auto paramID = address.fromFirstOccurrenceOf (oscAddressPrefix, false, false); + auto param = vts.getParameter (paramID); + + if (param == nullptr) // invalid parameter + return; + + String pluginID = message[0].getString(); + if (uuid == pluginID) // this message came from me! + return; + + int mixGroup = message[1].getInt32(); + if (mixGroup != (int) mixGroupParam->load()) // received message does not apply to this mix group + return; + + auto value = message[2].getFloat32(); + param->setValueNotifyingHost (param->convertTo0to1 (value)); +} + +void MixGroupsController::parseAllParams (const OSCMessage& message) +{ + if (message.size() % 2 != 0) // must have even number of arguments! + return; + + String pluginID = message[0].getString(); + if (uuid == pluginID) // this message came from me! + return; + + int mixGroup = message[1].getInt32(); + if (mixGroup != (int) mixGroupParam->load()) // received message does not apply to this mix group + return; + + for (int argIdx = 2; argIdx < message.size(); argIdx += 2) + { + auto temp1 = message[argIdx]; + auto temp2 = message[argIdx+1]; + if (! (message[argIdx].isString() && message[argIdx + 1].isFloat32())) // incorrect format... + continue; + + String paramID = message[argIdx].getString(); + auto param = vts.getParameter (paramID); + + if (param == nullptr) // invalid parameter + continue; + + auto value = message[argIdx + 1].getFloat32(); + param->setValueNotifyingHost (param->convertTo0to1 (value)); + } +} + +bool MixGroupsController::isValidOSCMessage (const OSCMessage& message) +{ + // special case: all parameters + if (message.getAddressPattern() == allParamsAddress && message.size() > 2) + return true; + + // other valid messages will be of the form: string, int, float + if (message.size() != 3) + return false; + + bool isValid = true; + + isValid &= message[0].isString(); + isValid &= message[1].isInt32(); + isValid &= message[2].isFloat32(); + + return isValid; +} diff --git a/Plugin/Source/MixGroups/MixGroupsController.h b/Plugin/Source/MixGroups/MixGroupsController.h @@ -0,0 +1,45 @@ +#ifndef MIXGROUPSCONTROLLER_H_INCLUDED +#define MIXGROUPSCONTROLLER_H_INCLUDED + +#include "MixGroupsParamReceiver.h" + +namespace MixGroupsConstants +{ + constexpr int portNum = 1818; + const String oscAddressPrefix = "/chowdsp/tape/"; + const String allParamsAddress = oscAddressPrefix + "all"; + const String mixGroupParamID = "mix_group"; +} + +/** Class to control syncing parameters between multiple mix groups */ +class MixGroupsController : private AudioProcessorValueTreeState::Listener, + private ChangeListener +{ +public: + MixGroupsController (AudioProcessorValueTreeState& vts, AudioProcessor* proc); + + static void createParameterLayout (std::vector<std::unique_ptr<RangedAudioParameter>>& params); + + void parameterChanged (const String& parameterID, float newValue) override; + void changeListenerCallback (ChangeBroadcaster* source) override; + void sendParameter (const String& paramID, int mixGroup, float value); + + void parseAllParams (const OSCMessage& message); + + static bool isValidOSCMessage (const OSCMessage& message); + +private: + void loadParameterList (Array<AudioProcessorParameter*>& params); + + AudioProcessorValueTreeState& vts; + std::atomic<float>* mixGroupParam = nullptr; + Array<String> paramList; + OSCSender sender; + Uuid uuid; + + SharedResourcePointer<MixGroupsParamReceiver> paramReceiver; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (MixGroupsController) +}; + +#endif // MIXGROUPSCONTROLLER_H_INCLUDED diff --git a/Plugin/Source/MixGroups/MixGroupsParamReceiver.cpp b/Plugin/Source/MixGroups/MixGroupsParamReceiver.cpp @@ -0,0 +1,38 @@ +#include "MixGroupsParamReceiver.h" +#include "MixGroupsController.h" + +MixGroupsParamReceiver::MixGroupsParamReceiver() +{ + // connect datagram socket + socket.bindToPort (MixGroupsConstants::portNum); + socket.setEnablePortReuse (true); + connectToSocket (socket); +} + +void MixGroupsParamReceiver::loadReceiverListeners (Array<String>& paramList) +{ + if (listenersInitialized) // listeners are already set up! + return; + + for (auto& paramID : paramList) + addListener (this, MixGroupsConstants::oscAddressPrefix + paramID); + + addListener (this, MixGroupsConstants::oscAddressPrefix + "all"); + + listenersInitialized = true; +} + +void MixGroupsParamReceiver::oscMessageReceived (const OSCMessage& message) +{ + // clear existing OSC message + oscMessage.clear(); + + if (message.isEmpty()) // no message! + return; + + if (! MixGroupsController::isValidOSCMessage (message)) + return; + + oscMessage = message; + sendChangeMessage(); +} diff --git a/Plugin/Source/MixGroups/MixGroupsParamReceiver.h b/Plugin/Source/MixGroups/MixGroupsParamReceiver.h @@ -0,0 +1,29 @@ +#ifndef MIXGROUPSPARAMRECEIVER +#define MIXGROUPSPARAMRECEIVER + +#include <JuceHeader.h> + +/** + * Class to receive parameter info from other instances of the plugin. + * Create instances of this class with a SharedResourcePointer, otherwise + * the messages won't sync correctly between instances of the plugin. + */ +class MixGroupsParamReceiver : public ChangeBroadcaster, + private OSCReceiver, + private OSCReceiver::ListenerWithOSCAddress<OSCReceiver::MessageLoopCallback> +{ +public: + MixGroupsParamReceiver(); + + void oscMessageReceived (const OSCMessage& message) override; + void loadReceiverListeners (Array<String>& paramList); + const OSCMessage& getOSCMessage() { return oscMessage; } + +private: + DatagramSocket socket; + OSCMessage oscMessage { "/noaddress" }; + bool listenersInitialized = false; +}; + +#endif // MIXGROUPSPARAMRECEIVER + diff --git a/Plugin/Source/PluginProcessor.cpp b/Plugin/Source/PluginProcessor.cpp @@ -30,7 +30,8 @@ ChowtapeModelAudioProcessor::ChowtapeModelAudioProcessor() hysteresis (vts), degrade (vts), chewer (vts), - flutter (vts) + flutter (vts), + mixGroupsController (vts, this) { for (int ch = 0; ch < 2; ++ch) lossFilter[ch].reset (new LossFilter (vts)); @@ -61,6 +62,7 @@ AudioProcessorValueTreeState::ParameterLayout ChowtapeModelAudioProcessor::creat Flutter::createParameterLayout (params); DegradeProcessor::createParameterLayout (params); ChewProcessor::createParameterLayout (params); + MixGroupsController::createParameterLayout (params); return { params.begin(), params.end() }; } diff --git a/Plugin/Source/PluginProcessor.h b/Plugin/Source/PluginProcessor.h @@ -22,6 +22,7 @@ #include "Presets/PresetManager.h" #include "GUI/MyLNF.h" #include "GUI/AutoUpdating.h" +#include "MixGroups/MixGroupsController.h" //============================================================================== /** @@ -97,6 +98,8 @@ private: AutoUpdater updater; bool needsUpdate = false; + MixGroupsController mixGroupsController; + //============================================================================== JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ChowtapeModelAudioProcessor) };