In this tutorial, we're going to analyze the structure of an audio effect application, by applying distortion to a looping audio file.
- Adding custom plugin classes
- Customizing an Edit using plugin objects.
- Registering plugins with the Engine so that they can be added to an Edit's PluginList
- Adding plugins to a Track.
- Writing a custom ValueSource to bind a Slider/Value to an AutomatableParameter.
We'll show you how to register plugins with the engine in order to add them to an Edit's pluginList. We'll also cover adding plugins to a track
Plugin
is the base class for plugins that can be loaded onto a Track
.
PluginManager
is a class for the management, registration and loading of plugins by the Engine
.
We need to construct a class that wraps a te::AutomatableParameter
as a juce::ValueSource
so that it can be used as a Value
in a Slider
.
param
is a pointer to anAutomatableParameter
In the ParameterValueSource
constructor, we attach a listener to
the AutomatableParameter
ParameterValueSource (AutomatableParameter::Ptr p)
: param (p)
{
param->addListener (this);
}
setValue()
is automatically called whenever the value changes. We need to override the function to update param
to reflect the new value.
void setValue (const var& newValue) override
{
param->setParameter (static_cast<float> (newValue), juce::sendNotification);
}
We need to override the function getValue()
to return the value from param
.
var getValue() const override
{
return param->getCurrentValue();
}
Now that we have created a class to wrap an AutomatableParameter as a ValueSource, we need to bind the AutomatableParameter to a Slider so that changes in either are reflected across each other.
- We can do this by getting the valueRange from the AutomatableParameter and using it to construct a NormalisableRange object
range
. We then set the NormalisableRange of the Slider torange
.
const auto v = p.valueRange;
const auto range = NormalisableRange<double> (static_cast<double> (v.start),
static_cast<double> (v.end),
static_cast<double> (v.interval),
static_cast<double> (v.skew),
v.symmetricSkew);
s.setNormalisableRange (range);
- We can then create a new ParameterValueSource object from AutomatableParameter
p
, cast it to theValue
type and refer the slider'sValue
object to it.
s.getValueObject().referTo (juce::Value (new ParameterValueSource (p)));
- In the constructor, we refer the gainValue CachedValue member to the plugin's
state
gain property, by using the Edit's undomanager and providing a default value of 1.
auto um = getUndoManager();
gainValue.referTo (state, IDs::gain, um, 1.0f);
- We create an AutomatableParameter and assign it to our gainParam member.
addParam
registers the parameter with thePlugin
. Next,attachToCurrentValue
attaches thegainValue
CachedValue
member to the gain parameter.
gainParam = addParam ("gain", TRANS("Gain"), { 0.1f, 20.0f });
gainParam->attachToCurrentValue (gainValue);
- In the Destructor, we notify listeners that the object has been deleted and detach all parameters from their current values.
notifyListenersOfDeletion();
gainParam->detachFromCurrentValue();
-
The
applyToBuffer
function is where plugin processing takes place. The destination buffer can be obtained from thePluginRenderContext
in order to apply digital signal processing. -
If the plugin is disabled, we return the function early to prevent processing.
if (! isEnabled())
return;
- We iterate through each sample of each channel and apply the tanh waveshaping function to the output audio.
for (int channel = 0; channel < fc.destBuffer->getNumChannels(); ++channel)
{
auto dest = fc.destBuffer->getWritePointer (channel);
for (int i = 0; i < fc.bufferNumSamples; ++i)
dest[i] = std::tanh (gainValue * dest[i]);
}
The DistortionEffectDemo is defined similarly to the PitchAndTimeComponent class
class DistortionEffectDemo : public Component,
private ChangeListener
- If you look in the private member section of the
DistortionEffectDemo
again, you'll see similarities to the PitchAndTimeComponent and PlayBackDemo. There is anEngine
to set up Tracktion Engine, and anEdit
for editing.
te::Edit edit { engine, te::createEmptyEdit(), te::Edit::forEditing, nullptr, 0 };
te::TransportControl& transport { edit.getTransport() };
- We've added a slider called
gainSlider
to represent distortion gain.
TextButton settingsButton { "Settings" }, playPauseButton { "Play" }, loadFileButton { "Load file" };
Thumbnail thumbnail { transport };
Slider gainSlider;
- In the constructor, we'll start by registering the
DistortionPlugin
class with Tracktion Engine.
engine.getPluginManager().createBuiltInType<DistortionPlugin>();
- We'll add the sliders and buttons to the GUI.
Helpers::addAndMakeVisible (*this, { &gainSlider, &settingsButton, &playPauseButton });
- Next, we'll load the PlayBackDemo audio example into file f.
oggTempFile = std::make_unique<TemporaryFile> (".ogg");
auto f = oggTempFile->getFile();
f.replaceWithData (PlaybackDemoAudio::BITs_Export_2_ogg, PlaybackDemoAudio::BITs_Export_2_oggSize);
- We'll create a clip from the audio file. Then we'll create a track on the edit. Finally, we'll load the clip onto the track.
auto track = EngineHelpers::getOrInsertAudioTrackAt (edit, 0);
jassert (track != nullptr);
te::AudioFile audioFile (edit.engine, f);
auto clip = track->insertWaveClip (f.getFileNameWithoutExtension(), f,
{ { 0.0, audioFile.getLength() }, 0.0 }, false);
jassert (clip != nullptr);
- Now that the track has been created and the clip is on the track, we need to create a new instance of DistortionPlugin and load it onto the track.
auto gainPlugin = edit.getPluginCache().createNewPlugin (DistortionPlugin::xmlTypeName, {});
track->pluginList.insertPlugin (gainPlugin, 0, nullptr);
- Finally, in order to hear the effect being played, we must set the transport loop range to the start/end of the clip, set it to loop, and initialise play.
edit.getTransport().addChangeListener (this);
EngineHelpers::loopAroundClip (*clip);
- In order to control the gain parameter with the slider, we need to bind them together. This ensures that parameter value changes are reflected by the slider. When the slider is moved, the plugin parameter values are updated. Similarly, when the plugin parameter values are updated, the slider position changes.
auto gainParam = gainPlugin->getAutomatableParameterByID ("gain");
bindSliderToParameter (gainSlider, *gainParam);