|
|
|
@@ -50,288 +50,412 @@ |
|
|
|
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
class ChannelClickListener
|
|
|
|
class ProcessorWithLevels : public AudioProcessor,
|
|
|
|
private AsyncUpdater,
|
|
|
|
private Timer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual ~ChannelClickListener() {}
|
|
|
|
virtual void channelButtonClicked (int channelIndex) = 0;
|
|
|
|
virtual bool isChannelActive (int channelIndex) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
class SurroundEditor : public AudioProcessorEditor,
|
|
|
|
private Timer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SurroundEditor (AudioProcessor& parent)
|
|
|
|
: AudioProcessorEditor (parent),
|
|
|
|
currentChannelLayout (AudioChannelSet::disabled()),
|
|
|
|
layoutTitle ("LayoutTitleLabel", getLayoutName())
|
|
|
|
ProcessorWithLevels()
|
|
|
|
: AudioProcessor (BusesProperties().withInput ("Input", AudioChannelSet::stereo())
|
|
|
|
.withInput ("Aux", AudioChannelSet::stereo(), false)
|
|
|
|
.withOutput ("Output", AudioChannelSet::stereo())
|
|
|
|
.withOutput ("Aux", AudioChannelSet::stereo(), false))
|
|
|
|
{
|
|
|
|
layoutTitle.setJustificationType (Justification::centred);
|
|
|
|
addAndMakeVisible (layoutTitle);
|
|
|
|
addAndMakeVisible (noChannelsLabel);
|
|
|
|
|
|
|
|
setSize (600, 100);
|
|
|
|
|
|
|
|
lastSuspended = ! getAudioProcessor()->isSuspended();
|
|
|
|
timerCallback();
|
|
|
|
startTimer (500);
|
|
|
|
startTimerHz (60);
|
|
|
|
applyBusLayouts (getBusesLayout());
|
|
|
|
}
|
|
|
|
|
|
|
|
void resized() override
|
|
|
|
~ProcessorWithLevels() override
|
|
|
|
{
|
|
|
|
auto r = getLocalBounds();
|
|
|
|
stopTimer();
|
|
|
|
cancelPendingUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
layoutTitle.setBounds (r.removeFromBottom (16));
|
|
|
|
void prepareToPlay (double, int) override
|
|
|
|
{
|
|
|
|
samplesToPlay = (int) getSampleRate();
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
noChannelsLabel.setBounds (r);
|
|
|
|
void processBlock (AudioBuffer<float>& audio, MidiBuffer&) override { processAudio (audio); }
|
|
|
|
void processBlock (AudioBuffer<double>& audio, MidiBuffer&) override { processAudio (audio); }
|
|
|
|
|
|
|
|
if (channelButtons.size() > 0)
|
|
|
|
{
|
|
|
|
auto buttonWidth = r.getWidth() / channelButtons.size();
|
|
|
|
for (auto channelButton : channelButtons)
|
|
|
|
channelButton->setBounds (r.removeFromLeft (buttonWidth));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void releaseResources() override { reset(); }
|
|
|
|
|
|
|
|
void paint (Graphics& g) override
|
|
|
|
float getLevel (int bus, int channel) const
|
|
|
|
{
|
|
|
|
g.fillAll (getLookAndFeel().findColour (ResizableWindow::backgroundColourId));
|
|
|
|
return readableLevels[(size_t) getChannelIndexInProcessBlockBuffer (true, bus, channel)];
|
|
|
|
}
|
|
|
|
|
|
|
|
void updateButton (Button* btn)
|
|
|
|
bool isBusesLayoutSupported (const BusesLayout& layouts) const override
|
|
|
|
{
|
|
|
|
if (auto* textButton = dynamic_cast<TextButton*> (btn))
|
|
|
|
const auto isSetValid = [] (const AudioChannelSet& set)
|
|
|
|
{
|
|
|
|
auto channelIndex = channelButtons.indexOf (textButton);
|
|
|
|
return ! set.isDisabled()
|
|
|
|
&& ! (set.isDiscreteLayout() && set.getChannelIndexForType (AudioChannelSet::discreteChannel0) == -1);
|
|
|
|
};
|
|
|
|
|
|
|
|
if (auto* listener = dynamic_cast<ChannelClickListener*> (getAudioProcessor()))
|
|
|
|
listener->channelButtonClicked (channelIndex);
|
|
|
|
}
|
|
|
|
return isSetValid (layouts.getMainOutputChannelSet())
|
|
|
|
&& isSetValid (layouts.getMainInputChannelSet());
|
|
|
|
}
|
|
|
|
|
|
|
|
void updateGUI()
|
|
|
|
void reset() override
|
|
|
|
{
|
|
|
|
const auto& channelSet = getAudioProcessor()->getChannelLayoutOfBus (false, 0);
|
|
|
|
channelClicked = 0;
|
|
|
|
samplesPlayed = samplesToPlay;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (channelSet != currentChannelLayout)
|
|
|
|
{
|
|
|
|
currentChannelLayout = channelSet;
|
|
|
|
bool applyBusLayouts (const BusesLayout& layouts) override
|
|
|
|
{
|
|
|
|
// Some very badly-behaved hosts will call this during processing!
|
|
|
|
const SpinLock::ScopedLockType lock (levelMutex);
|
|
|
|
|
|
|
|
layoutTitle.setText (currentChannelLayout.getDescription(), NotificationType::dontSendNotification);
|
|
|
|
channelButtons.clear();
|
|
|
|
activeChannels.resize (currentChannelLayout.size());
|
|
|
|
const auto result = AudioProcessor::applyBusLayouts (layouts);
|
|
|
|
|
|
|
|
if (currentChannelLayout == AudioChannelSet::disabled())
|
|
|
|
{
|
|
|
|
noChannelsLabel.setVisible (true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto numChannels = currentChannelLayout.size();
|
|
|
|
size_t numInputChannels = 0;
|
|
|
|
|
|
|
|
for (auto i = 0; i < numChannels; ++i)
|
|
|
|
{
|
|
|
|
auto channelName =
|
|
|
|
AudioChannelSet::getAbbreviatedChannelTypeName (currentChannelLayout.getTypeOfChannel (i));
|
|
|
|
for (auto i = 0; i < getBusCount (true); ++i)
|
|
|
|
numInputChannels += (size_t) getBus (true, i)->getLastEnabledLayout().size();
|
|
|
|
|
|
|
|
TextButton* newButton;
|
|
|
|
channelButtons.add (newButton = new TextButton (channelName, channelName));
|
|
|
|
incomingLevels = readableLevels = std::vector<float> (numInputChannels, 0.0f);
|
|
|
|
|
|
|
|
newButton->onClick = [this, newButton] { updateButton (newButton); };
|
|
|
|
addAndMakeVisible (newButton);
|
|
|
|
}
|
|
|
|
triggerAsyncUpdate();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
noChannelsLabel.setVisible (false);
|
|
|
|
resized();
|
|
|
|
}
|
|
|
|
//==============================================================================
|
|
|
|
const String getName() const override { return "Surround PlugIn"; }
|
|
|
|
bool acceptsMidi() const override { return false; }
|
|
|
|
bool producesMidi() const override { return false; }
|
|
|
|
double getTailLengthSeconds() const override { return 0; }
|
|
|
|
|
|
|
|
if (auto* listener = dynamic_cast<ChannelClickListener*> (getAudioProcessor()))
|
|
|
|
{
|
|
|
|
auto activeColour = getLookAndFeel().findColour (Slider::thumbColourId);
|
|
|
|
auto inactiveColour = getLookAndFeel().findColour (Slider::trackColourId);
|
|
|
|
//==============================================================================
|
|
|
|
int getNumPrograms() override { return 1; }
|
|
|
|
int getCurrentProgram() override { return 0; }
|
|
|
|
void setCurrentProgram (int) override {}
|
|
|
|
const String getProgramName (int) override { return "None"; }
|
|
|
|
void changeProgramName (int, const String&) override {}
|
|
|
|
|
|
|
|
for (auto i = 0; i < activeChannels.size(); ++i)
|
|
|
|
{
|
|
|
|
auto isActive = listener->isChannelActive (i);
|
|
|
|
activeChannels.getReference (i) = isActive;
|
|
|
|
channelButtons[i]->setColour (TextButton::buttonColourId, isActive ? activeColour : inactiveColour);
|
|
|
|
channelButtons[i]->repaint();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//==============================================================================
|
|
|
|
void getStateInformation (MemoryBlock&) override {}
|
|
|
|
void setStateInformation (const void*, int) override {}
|
|
|
|
|
|
|
|
void channelButtonClicked (int bus, int channelIndex)
|
|
|
|
{
|
|
|
|
channelClicked = getChannelIndexInProcessBlockBuffer (false, bus, channelIndex);
|
|
|
|
samplesPlayed = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::function<void()> updateEditor;
|
|
|
|
|
|
|
|
private:
|
|
|
|
String getLayoutName() const
|
|
|
|
void handleAsyncUpdate() override
|
|
|
|
{
|
|
|
|
if (auto* p = getAudioProcessor())
|
|
|
|
return p->getChannelLayoutOfBus (false, 0).getDescription();
|
|
|
|
|
|
|
|
return "Unknown";
|
|
|
|
NullCheckedInvocation::invoke (updateEditor);
|
|
|
|
}
|
|
|
|
|
|
|
|
void timerCallback() override
|
|
|
|
template <typename Float>
|
|
|
|
void processAudio (AudioBuffer<Float>& audio)
|
|
|
|
{
|
|
|
|
if (getAudioProcessor()->isSuspended() != lastSuspended)
|
|
|
|
{
|
|
|
|
lastSuspended = getAudioProcessor()->isSuspended();
|
|
|
|
updateGUI();
|
|
|
|
}
|
|
|
|
SpinLock::ScopedTryLockType lock (levelMutex);
|
|
|
|
|
|
|
|
if (! lastSuspended)
|
|
|
|
{
|
|
|
|
if (auto* listener = dynamic_cast<ChannelClickListener*> (getAudioProcessor()))
|
|
|
|
if (lock.isLocked())
|
|
|
|
{
|
|
|
|
auto activeColour = getLookAndFeel().findColour (Slider::thumbColourId);
|
|
|
|
auto inactiveColour = getLookAndFeel().findColour (Slider::trackColourId);
|
|
|
|
const auto numInputChannels = (size_t) getTotalNumInputChannels();
|
|
|
|
|
|
|
|
for (auto i = 0; i < activeChannels.size(); ++i)
|
|
|
|
for (size_t i = 0; i < numInputChannels; ++i)
|
|
|
|
{
|
|
|
|
auto isActive = listener->isChannelActive (i);
|
|
|
|
if (activeChannels.getReference (i) != isActive)
|
|
|
|
{
|
|
|
|
activeChannels.getReference (i) = isActive;
|
|
|
|
channelButtons[i]->setColour (TextButton::buttonColourId, isActive ? activeColour : inactiveColour);
|
|
|
|
channelButtons[i]->repaint();
|
|
|
|
}
|
|
|
|
const auto minMax = audio.findMinMax ((int) i, 0, audio.getNumSamples());
|
|
|
|
const auto newMax = (float) std::max (std::abs (minMax.getStart()), std::abs (minMax.getEnd()));
|
|
|
|
|
|
|
|
auto& toUpdate = incomingLevels[i];
|
|
|
|
toUpdate = jmax (toUpdate, newMax);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
audio.clear (0, audio.getNumSamples());
|
|
|
|
|
|
|
|
auto fillSamples = jmin (samplesToPlay - samplesPlayed, audio.getNumSamples());
|
|
|
|
|
|
|
|
if (isPositiveAndBelow (channelClicked, audio.getNumChannels()))
|
|
|
|
{
|
|
|
|
auto* channelBuffer = audio.getWritePointer (channelClicked);
|
|
|
|
auto freq = (float) (440.0 / getSampleRate());
|
|
|
|
|
|
|
|
for (auto i = 0; i < fillSamples; ++i)
|
|
|
|
channelBuffer[i] += std::sin (MathConstants<float>::twoPi * freq * (float) samplesPlayed++);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
AudioChannelSet currentChannelLayout;
|
|
|
|
Label noChannelsLabel { "noChannelsLabel", "Input disabled" },
|
|
|
|
layoutTitle;
|
|
|
|
OwnedArray<TextButton> channelButtons;
|
|
|
|
Array<bool> activeChannels;
|
|
|
|
void timerCallback() override
|
|
|
|
{
|
|
|
|
const SpinLock::ScopedLockType lock (levelMutex);
|
|
|
|
|
|
|
|
bool lastSuspended;
|
|
|
|
for (size_t i = 0; i < readableLevels.size(); ++i)
|
|
|
|
readableLevels[i] = std::max (readableLevels[i] * 0.95f, std::exchange (incomingLevels[i], 0.0f));
|
|
|
|
}
|
|
|
|
|
|
|
|
SpinLock levelMutex;
|
|
|
|
std::vector<float> incomingLevels;
|
|
|
|
std::vector<float> readableLevels;
|
|
|
|
|
|
|
|
int channelClicked;
|
|
|
|
int samplesPlayed;
|
|
|
|
int samplesToPlay;
|
|
|
|
};
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
class SurroundProcessor : public AudioProcessor,
|
|
|
|
public ChannelClickListener,
|
|
|
|
private AsyncUpdater
|
|
|
|
const Colour textColour = Colours::white.withAlpha (0.8f);
|
|
|
|
|
|
|
|
inline void drawBackground (Component& comp, Graphics& g)
|
|
|
|
{
|
|
|
|
g.setColour (comp.getLookAndFeel().findColour (ResizableWindow::backgroundColourId).darker (0.8f));
|
|
|
|
g.fillRoundedRectangle (comp.getLocalBounds().toFloat(), 4.0f);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void configureLabel (Label& label, const AudioProcessor::Bus* layout)
|
|
|
|
{
|
|
|
|
const auto text = layout != nullptr
|
|
|
|
? (layout->getName() + ": " + layout->getCurrentLayout().getDescription())
|
|
|
|
: "";
|
|
|
|
label.setText (text, dontSendNotification);
|
|
|
|
label.setJustificationType (Justification::centred);
|
|
|
|
label.setColour (Label::textColourId, textColour);
|
|
|
|
}
|
|
|
|
|
|
|
|
class InputBusViewer : public Component,
|
|
|
|
private Timer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SurroundProcessor()
|
|
|
|
: AudioProcessor(BusesProperties().withInput ("Input", AudioChannelSet::stereo())
|
|
|
|
.withOutput ("Output", AudioChannelSet::stereo()))
|
|
|
|
{}
|
|
|
|
InputBusViewer (ProcessorWithLevels& proc, int busNumber)
|
|
|
|
: processor (proc),
|
|
|
|
bus (busNumber)
|
|
|
|
{
|
|
|
|
configureLabel (layoutName, processor.getBus (true, bus));
|
|
|
|
addAndMakeVisible (layoutName);
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
void prepareToPlay (double sampleRate, int samplesPerBlock) override
|
|
|
|
startTimerHz (60);
|
|
|
|
}
|
|
|
|
|
|
|
|
~InputBusViewer() override
|
|
|
|
{
|
|
|
|
channelClicked = 0;
|
|
|
|
sampleOffset = static_cast<int> (std::ceil (sampleRate));
|
|
|
|
stopTimer();
|
|
|
|
}
|
|
|
|
|
|
|
|
auto numChannels = getChannelCountOfBus (true, 0);
|
|
|
|
channelActive.resize (numChannels);
|
|
|
|
alphaCoeffs .resize (numChannels);
|
|
|
|
reset();
|
|
|
|
void paint (Graphics& g) override
|
|
|
|
{
|
|
|
|
drawBackground (*this, g);
|
|
|
|
|
|
|
|
triggerAsyncUpdate();
|
|
|
|
auto* layout = processor.getBus (true, bus);
|
|
|
|
|
|
|
|
ignoreUnused (samplesPerBlock);
|
|
|
|
}
|
|
|
|
if (layout == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
void releaseResources() override { reset(); }
|
|
|
|
const auto channelSet = layout->getCurrentLayout();
|
|
|
|
const auto numChannels = channelSet.size();
|
|
|
|
|
|
|
|
void processBlock (AudioBuffer<float>& buffer, MidiBuffer&) override
|
|
|
|
{
|
|
|
|
for (auto ch = 0; ch < buffer.getNumChannels(); ++ch)
|
|
|
|
Grid grid;
|
|
|
|
|
|
|
|
grid.autoFlow = Grid::AutoFlow::column;
|
|
|
|
grid.autoColumns = grid.autoRows = Grid::TrackInfo (Grid::Fr (1));
|
|
|
|
grid.items.insertMultiple (0, GridItem(), numChannels);
|
|
|
|
grid.performLayout (getLocalBounds());
|
|
|
|
|
|
|
|
const auto minDb = -50.0f;
|
|
|
|
const auto maxDb = 6.0f;
|
|
|
|
|
|
|
|
for (auto i = 0; i < numChannels; ++i)
|
|
|
|
{
|
|
|
|
auto& channelTime = channelActive.getReference (ch);
|
|
|
|
auto& alpha = alphaCoeffs .getReference (ch);
|
|
|
|
g.setColour (Colours::orange.darker());
|
|
|
|
|
|
|
|
for (auto j = 0; j < buffer.getNumSamples(); ++j)
|
|
|
|
{
|
|
|
|
auto sample = buffer.getReadPointer (ch)[j];
|
|
|
|
alpha = (0.8f * alpha) + (0.2f * sample);
|
|
|
|
const auto levelInDb = Decibels::gainToDecibels (processor.getLevel (bus, i), minDb);
|
|
|
|
const auto fractionOfHeight = jmap (levelInDb, minDb, maxDb, 0.0f, 1.0f);
|
|
|
|
const auto bounds = grid.items[i].currentBounds;
|
|
|
|
const auto trackBounds = bounds.withSizeKeepingCentre (16, bounds.getHeight() - 10).toFloat();
|
|
|
|
g.fillRect (trackBounds.withHeight (trackBounds.proportionOfHeight (fractionOfHeight)).withBottomY (trackBounds.getBottom()));
|
|
|
|
|
|
|
|
if (std::abs (alpha) >= 0.1f)
|
|
|
|
channelTime = static_cast<int> (getSampleRate() / 2.0);
|
|
|
|
}
|
|
|
|
g.setColour (textColour);
|
|
|
|
|
|
|
|
channelTime = jmax (0, channelTime - buffer.getNumSamples());
|
|
|
|
g.drawText (channelSet.getAbbreviatedChannelTypeName (channelSet.getTypeOfChannel (i)),
|
|
|
|
bounds,
|
|
|
|
Justification::centredBottom);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
auto fillSamples = jmin (static_cast<int> (std::ceil (getSampleRate())) - sampleOffset,
|
|
|
|
buffer.getNumSamples());
|
|
|
|
void resized() override
|
|
|
|
{
|
|
|
|
layoutName.setBounds (getLocalBounds().removeFromTop (20));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isPositiveAndBelow (channelClicked, buffer.getNumChannels()))
|
|
|
|
{
|
|
|
|
auto* channelBuffer = buffer.getWritePointer (channelClicked);
|
|
|
|
auto freq = (float) (440.0 / getSampleRate());
|
|
|
|
int getNumChannels() const
|
|
|
|
{
|
|
|
|
if (auto* b = processor.getBus (true, bus))
|
|
|
|
return b->getCurrentLayout().size();
|
|
|
|
|
|
|
|
for (auto i = 0; i < fillSamples; ++i)
|
|
|
|
channelBuffer[i] += std::sin (MathConstants<float>::twoPi * freq * static_cast<float> (sampleOffset++));
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
using AudioProcessor::processBlock;
|
|
|
|
private:
|
|
|
|
void timerCallback() override { repaint(); }
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
AudioProcessorEditor* createEditor() override { return new SurroundEditor (*this); }
|
|
|
|
bool hasEditor() const override { return true; }
|
|
|
|
ProcessorWithLevels& processor;
|
|
|
|
int bus = 0;
|
|
|
|
Label layoutName;
|
|
|
|
};
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
bool isBusesLayoutSupported (const BusesLayout& layouts) const override
|
|
|
|
//==============================================================================
|
|
|
|
class OutputBusViewer : public Component
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
OutputBusViewer (ProcessorWithLevels& proc, int busNumber)
|
|
|
|
: processor (proc),
|
|
|
|
bus (busNumber)
|
|
|
|
{
|
|
|
|
return ((! layouts.getMainInputChannelSet() .isDiscreteLayout())
|
|
|
|
&& (! layouts.getMainOutputChannelSet().isDiscreteLayout())
|
|
|
|
&& (layouts.getMainInputChannelSet() == layouts.getMainOutputChannelSet())
|
|
|
|
&& (! layouts.getMainInputChannelSet().isDisabled()));
|
|
|
|
auto* layout = processor.getBus (false, bus);
|
|
|
|
|
|
|
|
configureLabel (layoutName, layout);
|
|
|
|
addAndMakeVisible (layoutName);
|
|
|
|
|
|
|
|
if (layout == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto& channelSet = layout->getCurrentLayout();
|
|
|
|
|
|
|
|
const auto numChannels = channelSet.size();
|
|
|
|
|
|
|
|
for (auto i = 0; i < numChannels; ++i)
|
|
|
|
{
|
|
|
|
const auto channelName = channelSet.getAbbreviatedChannelTypeName (channelSet.getTypeOfChannel (i));
|
|
|
|
|
|
|
|
channelButtons.emplace_back (channelName, channelName);
|
|
|
|
|
|
|
|
auto& newButton = channelButtons.back();
|
|
|
|
newButton.onClick = [&proc = processor, bus = bus, i] { proc.channelButtonClicked (bus, i); };
|
|
|
|
addAndMakeVisible (newButton);
|
|
|
|
}
|
|
|
|
|
|
|
|
resized();
|
|
|
|
}
|
|
|
|
|
|
|
|
void reset() override
|
|
|
|
void paint (Graphics& g) override
|
|
|
|
{
|
|
|
|
for (auto& channel : channelActive)
|
|
|
|
channel = 0;
|
|
|
|
drawBackground (*this, g);
|
|
|
|
}
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
const String getName() const override { return "Surround PlugIn"; }
|
|
|
|
bool acceptsMidi() const override { return false; }
|
|
|
|
bool producesMidi() const override { return false; }
|
|
|
|
double getTailLengthSeconds() const override { return 0; }
|
|
|
|
void resized() override
|
|
|
|
{
|
|
|
|
auto b = getLocalBounds();
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
int getNumPrograms() override { return 1; }
|
|
|
|
int getCurrentProgram() override { return 0; }
|
|
|
|
void setCurrentProgram (int) override {}
|
|
|
|
const String getProgramName (int) override { return "None"; }
|
|
|
|
void changeProgramName (int, const String&) override {}
|
|
|
|
layoutName.setBounds (b.removeFromBottom (20));
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
void getStateInformation (MemoryBlock&) override {}
|
|
|
|
void setStateInformation (const void*, int) override {}
|
|
|
|
Grid grid;
|
|
|
|
grid.autoFlow = Grid::AutoFlow::column;
|
|
|
|
grid.autoColumns = grid.autoRows = Grid::TrackInfo (Grid::Fr (1));
|
|
|
|
|
|
|
|
void channelButtonClicked (int channelIndex) override
|
|
|
|
for (auto& channelButton : channelButtons)
|
|
|
|
grid.items.add (GridItem (channelButton));
|
|
|
|
|
|
|
|
grid.performLayout (b.reduced (2));
|
|
|
|
}
|
|
|
|
|
|
|
|
int getNumChannels() const
|
|
|
|
{
|
|
|
|
channelClicked = channelIndex;
|
|
|
|
sampleOffset = 0;
|
|
|
|
if (auto* b = processor.getBus (false, bus))
|
|
|
|
return b->getCurrentLayout().size();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isChannelActive (int channelIndex) override
|
|
|
|
private:
|
|
|
|
ProcessorWithLevels& processor;
|
|
|
|
int bus = 0;
|
|
|
|
Label layoutName;
|
|
|
|
std::list<TextButton> channelButtons;
|
|
|
|
};
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
class SurroundEditor : public AudioProcessorEditor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit SurroundEditor (ProcessorWithLevels& parent)
|
|
|
|
: AudioProcessorEditor (parent),
|
|
|
|
customProcessor (parent),
|
|
|
|
scopedUpdateEditor (customProcessor.updateEditor, [this] { updateGUI(); })
|
|
|
|
{
|
|
|
|
return channelActive[channelIndex] > 0;
|
|
|
|
updateGUI();
|
|
|
|
setResizable (true, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void handleAsyncUpdate() override
|
|
|
|
void resized() override
|
|
|
|
{
|
|
|
|
if (auto* editor = getActiveEditor())
|
|
|
|
if (auto* surroundEditor = dynamic_cast<SurroundEditor*> (editor))
|
|
|
|
surroundEditor->updateGUI();
|
|
|
|
auto r = getLocalBounds();
|
|
|
|
doLayout (inputViewers, r.removeFromTop (proportionOfHeight (0.5f)));
|
|
|
|
doLayout (outputViewers, r);
|
|
|
|
}
|
|
|
|
|
|
|
|
void paint (Graphics& g) override
|
|
|
|
{
|
|
|
|
g.fillAll (getLookAndFeel().findColour (ResizableWindow::backgroundColourId));
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Array<int> channelActive;
|
|
|
|
Array<float> alphaCoeffs;
|
|
|
|
int channelClicked;
|
|
|
|
int sampleOffset;
|
|
|
|
template <typename Range>
|
|
|
|
void doLayout (Range& range, Rectangle<int> bounds) const
|
|
|
|
{
|
|
|
|
FlexBox fb;
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SurroundProcessor)
|
|
|
|
for (auto& viewer : range)
|
|
|
|
{
|
|
|
|
if (viewer.getNumChannels() != 0)
|
|
|
|
{
|
|
|
|
fb.items.add (FlexItem (viewer)
|
|
|
|
.withFlex ((float) viewer.getNumChannels())
|
|
|
|
.withMargin (4.0f));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fb.performLayout (bounds);
|
|
|
|
}
|
|
|
|
|
|
|
|
void updateGUI()
|
|
|
|
{
|
|
|
|
inputViewers.clear();
|
|
|
|
outputViewers.clear();
|
|
|
|
|
|
|
|
const auto inputBuses = getAudioProcessor()->getBusCount (true);
|
|
|
|
|
|
|
|
for (auto i = 0; i < inputBuses; ++i)
|
|
|
|
{
|
|
|
|
inputViewers.emplace_back (customProcessor, i);
|
|
|
|
addAndMakeVisible (inputViewers.back());
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto outputBuses = getAudioProcessor()->getBusCount (false);
|
|
|
|
|
|
|
|
for (auto i = 0; i < outputBuses; ++i)
|
|
|
|
{
|
|
|
|
outputViewers.emplace_back (customProcessor, i);
|
|
|
|
addAndMakeVisible (outputViewers.back());
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto channels = jmax (processor.getTotalNumInputChannels(),
|
|
|
|
processor.getTotalNumOutputChannels());
|
|
|
|
setSize (jmax (150, channels * 40), 200);
|
|
|
|
|
|
|
|
resized();
|
|
|
|
}
|
|
|
|
|
|
|
|
ProcessorWithLevels& customProcessor;
|
|
|
|
ScopedValueSetter<std::function<void()>> scopedUpdateEditor;
|
|
|
|
std::list<InputBusViewer> inputViewers;
|
|
|
|
std::list<OutputBusViewer> outputViewers;
|
|
|
|
};
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
struct SurroundProcessor : public ProcessorWithLevels
|
|
|
|
{
|
|
|
|
AudioProcessorEditor* createEditor() override { return new SurroundEditor (*this); }
|
|
|
|
bool hasEditor() const override { return true; }
|
|
|
|
};
|