25bd5d8adb
subrepo: subdir: "deps/juce" merged: "b13f9084e" upstream: origin: "https://github.com/essej/JUCE.git" branch: "sono6good" commit: "b13f9084e" git-subrepo: version: "0.4.3" origin: "https://github.com/ingydotnet/git-subrepo.git" commit: "2f68596"
338 lines
12 KiB
C++
338 lines
12 KiB
C++
/*
|
|
==============================================================================
|
|
|
|
This file is part of the JUCE examples.
|
|
Copyright (c) 2020 - Raw Material Software Limited
|
|
|
|
The code included in this file is provided under the terms of the ISC license
|
|
http://www.isc.org/downloads/software-support-policy/isc-license. Permission
|
|
To use, copy, modify, and/or distribute this software for any purpose with or
|
|
without fee is hereby granted provided that the above copyright notice and
|
|
this permission notice appear in all copies.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES,
|
|
WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR
|
|
PURPOSE, ARE DISCLAIMED.
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
/*******************************************************************************
|
|
The block below describes the properties of this PIP. A PIP is a short snippet
|
|
of code that can be read by the Projucer and used to generate a JUCE project.
|
|
|
|
BEGIN_JUCE_PIP_METADATA
|
|
|
|
name: SurroundPlugin
|
|
version: 1.0.0
|
|
vendor: JUCE
|
|
website: http://juce.com
|
|
description: Surround audio plugin.
|
|
|
|
dependencies: juce_audio_basics, juce_audio_devices, juce_audio_formats,
|
|
juce_audio_plugin_client, juce_audio_processors,
|
|
juce_audio_utils, juce_core, juce_data_structures,
|
|
juce_events, juce_graphics, juce_gui_basics, juce_gui_extra
|
|
exporters: xcode_mac, vs2019, linux_make
|
|
|
|
moduleFlags: JUCE_STRICT_REFCOUNTEDPOINTER=1
|
|
|
|
type: AudioProcessor
|
|
mainClass: SurroundProcessor
|
|
|
|
useLocalCopy: 1
|
|
|
|
END_JUCE_PIP_METADATA
|
|
|
|
*******************************************************************************/
|
|
|
|
#pragma once
|
|
|
|
|
|
//==============================================================================
|
|
class ChannelClickListener
|
|
{
|
|
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())
|
|
{
|
|
layoutTitle.setJustificationType (Justification::centred);
|
|
addAndMakeVisible (layoutTitle);
|
|
addAndMakeVisible (noChannelsLabel);
|
|
|
|
setSize (600, 100);
|
|
|
|
lastSuspended = ! getAudioProcessor()->isSuspended();
|
|
timerCallback();
|
|
startTimer (500);
|
|
}
|
|
|
|
void resized() override
|
|
{
|
|
auto r = getLocalBounds();
|
|
|
|
layoutTitle.setBounds (r.removeFromBottom (16));
|
|
|
|
noChannelsLabel.setBounds (r);
|
|
|
|
if (channelButtons.size() > 0)
|
|
{
|
|
auto buttonWidth = r.getWidth() / channelButtons.size();
|
|
for (auto channelButton : channelButtons)
|
|
channelButton->setBounds (r.removeFromLeft (buttonWidth));
|
|
}
|
|
}
|
|
|
|
void paint (Graphics& g) override
|
|
{
|
|
g.fillAll (getLookAndFeel().findColour (ResizableWindow::backgroundColourId));
|
|
}
|
|
|
|
void updateButton (Button* btn)
|
|
{
|
|
if (auto* textButton = dynamic_cast<TextButton*> (btn))
|
|
{
|
|
auto channelIndex = channelButtons.indexOf (textButton);
|
|
|
|
if (auto* listener = dynamic_cast<ChannelClickListener*> (getAudioProcessor()))
|
|
listener->channelButtonClicked (channelIndex);
|
|
}
|
|
}
|
|
|
|
void updateGUI()
|
|
{
|
|
const auto& channelSet = getAudioProcessor()->getChannelLayoutOfBus (false, 0);
|
|
|
|
if (channelSet != currentChannelLayout)
|
|
{
|
|
currentChannelLayout = channelSet;
|
|
|
|
layoutTitle.setText (currentChannelLayout.getDescription(), NotificationType::dontSendNotification);
|
|
channelButtons.clear();
|
|
activeChannels.resize (currentChannelLayout.size());
|
|
|
|
if (currentChannelLayout == AudioChannelSet::disabled())
|
|
{
|
|
noChannelsLabel.setVisible (true);
|
|
}
|
|
else
|
|
{
|
|
auto numChannels = currentChannelLayout.size();
|
|
|
|
for (auto i = 0; i < numChannels; ++i)
|
|
{
|
|
auto channelName =
|
|
AudioChannelSet::getAbbreviatedChannelTypeName (currentChannelLayout.getTypeOfChannel (i));
|
|
|
|
TextButton* newButton;
|
|
channelButtons.add (newButton = new TextButton (channelName, channelName));
|
|
|
|
newButton->onClick = [this, newButton] { updateButton (newButton); };
|
|
addAndMakeVisible (newButton);
|
|
}
|
|
|
|
noChannelsLabel.setVisible (false);
|
|
resized();
|
|
}
|
|
|
|
if (auto* listener = dynamic_cast<ChannelClickListener*> (getAudioProcessor()))
|
|
{
|
|
auto activeColour = getLookAndFeel().findColour (Slider::thumbColourId);
|
|
auto inactiveColour = getLookAndFeel().findColour (Slider::trackColourId);
|
|
|
|
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();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
String getLayoutName() const
|
|
{
|
|
if (auto* p = getAudioProcessor())
|
|
return p->getChannelLayoutOfBus (false, 0).getDescription();
|
|
|
|
return "Unknown";
|
|
}
|
|
|
|
void timerCallback() override
|
|
{
|
|
if (getAudioProcessor()->isSuspended() != lastSuspended)
|
|
{
|
|
lastSuspended = getAudioProcessor()->isSuspended();
|
|
updateGUI();
|
|
}
|
|
|
|
if (! lastSuspended)
|
|
{
|
|
if (auto* listener = dynamic_cast<ChannelClickListener*> (getAudioProcessor()))
|
|
{
|
|
auto activeColour = getLookAndFeel().findColour (Slider::thumbColourId);
|
|
auto inactiveColour = getLookAndFeel().findColour (Slider::trackColourId);
|
|
|
|
for (auto i = 0; i < activeChannels.size(); ++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();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
AudioChannelSet currentChannelLayout;
|
|
Label noChannelsLabel { "noChannelsLabel", "Input disabled" },
|
|
layoutTitle;
|
|
OwnedArray<TextButton> channelButtons;
|
|
Array<bool> activeChannels;
|
|
|
|
bool lastSuspended;
|
|
};
|
|
|
|
//==============================================================================
|
|
class SurroundProcessor : public AudioProcessor,
|
|
public ChannelClickListener,
|
|
private AsyncUpdater
|
|
{
|
|
public:
|
|
SurroundProcessor()
|
|
: AudioProcessor(BusesProperties().withInput ("Input", AudioChannelSet::stereo())
|
|
.withOutput ("Output", AudioChannelSet::stereo()))
|
|
{}
|
|
|
|
//==============================================================================
|
|
void prepareToPlay (double sampleRate, int samplesPerBlock) override
|
|
{
|
|
channelClicked = 0;
|
|
sampleOffset = static_cast<int> (std::ceil (sampleRate));
|
|
|
|
auto numChannels = getChannelCountOfBus (true, 0);
|
|
channelActive.resize (numChannels);
|
|
alphaCoeffs .resize (numChannels);
|
|
reset();
|
|
|
|
triggerAsyncUpdate();
|
|
|
|
ignoreUnused (samplesPerBlock);
|
|
}
|
|
|
|
void releaseResources() override { reset(); }
|
|
|
|
void processBlock (AudioBuffer<float>& buffer, MidiBuffer&) override
|
|
{
|
|
for (auto ch = 0; ch < buffer.getNumChannels(); ++ch)
|
|
{
|
|
auto& channelTime = channelActive.getReference (ch);
|
|
auto& alpha = alphaCoeffs .getReference (ch);
|
|
|
|
for (auto j = 0; j < buffer.getNumSamples(); ++j)
|
|
{
|
|
auto sample = buffer.getReadPointer (ch)[j];
|
|
alpha = (0.8f * alpha) + (0.2f * sample);
|
|
|
|
if (std::abs (alpha) >= 0.1f)
|
|
channelTime = static_cast<int> (getSampleRate() / 2.0);
|
|
}
|
|
|
|
channelTime = jmax (0, channelTime - buffer.getNumSamples());
|
|
}
|
|
|
|
auto fillSamples = jmin (static_cast<int> (std::ceil (getSampleRate())) - sampleOffset,
|
|
buffer.getNumSamples());
|
|
|
|
if (isPositiveAndBelow (channelClicked, buffer.getNumChannels()))
|
|
{
|
|
auto* channelBuffer = buffer.getWritePointer (channelClicked);
|
|
auto freq = (float) (440.0 / getSampleRate());
|
|
|
|
for (auto i = 0; i < fillSamples; ++i)
|
|
channelBuffer[i] += std::sin (MathConstants<float>::twoPi * freq * static_cast<float> (sampleOffset++));
|
|
}
|
|
}
|
|
|
|
using AudioProcessor::processBlock;
|
|
|
|
//==============================================================================
|
|
AudioProcessorEditor* createEditor() override { return new SurroundEditor (*this); }
|
|
bool hasEditor() const override { return true; }
|
|
|
|
//==============================================================================
|
|
bool isBusesLayoutSupported (const BusesLayout& layouts) const override
|
|
{
|
|
return ((! layouts.getMainInputChannelSet() .isDiscreteLayout())
|
|
&& (! layouts.getMainOutputChannelSet().isDiscreteLayout())
|
|
&& (layouts.getMainInputChannelSet() == layouts.getMainOutputChannelSet())
|
|
&& (! layouts.getMainInputChannelSet().isDisabled()));
|
|
}
|
|
|
|
void reset() override
|
|
{
|
|
for (auto& channel : channelActive)
|
|
channel = 0;
|
|
}
|
|
|
|
//==============================================================================
|
|
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; }
|
|
|
|
//==============================================================================
|
|
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 {}
|
|
|
|
//==============================================================================
|
|
void getStateInformation (MemoryBlock&) override {}
|
|
void setStateInformation (const void*, int) override {}
|
|
|
|
void channelButtonClicked (int channelIndex) override
|
|
{
|
|
channelClicked = channelIndex;
|
|
sampleOffset = 0;
|
|
}
|
|
|
|
bool isChannelActive (int channelIndex) override
|
|
{
|
|
return channelActive[channelIndex] > 0;
|
|
}
|
|
|
|
void handleAsyncUpdate() override
|
|
{
|
|
if (auto* editor = getActiveEditor())
|
|
if (auto* surroundEditor = dynamic_cast<SurroundEditor*> (editor))
|
|
surroundEditor->updateGUI();
|
|
}
|
|
|
|
private:
|
|
Array<int> channelActive;
|
|
Array<float> alphaCoeffs;
|
|
int channelClicked;
|
|
int sampleOffset;
|
|
|
|
//==============================================================================
|
|
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SurroundProcessor)
|
|
};
|