audacia/src/effects/Compressor.cpp

816 lines
26 KiB
C++
Raw Normal View History

/**********************************************************************
Audacity: A Digital Audio Editor
Compressor.cpp
Dominic Mazzoni
Martyn Shaw
Steve Jolly
*******************************************************************//**
\class EffectCompressor
\brief An Effect derived from EffectTwoPassSimpleMono
- Martyn Shaw made it inherit from EffectTwoPassSimpleMono 10/2005.
- Steve Jolly made it inherit from EffectSimpleMono.
- GUI added and implementation improved by Dominic Mazzoni, 5/11/2003.
*//****************************************************************//**
\class CompressorPanel
\brief Panel used within the EffectCompressor for EffectCompressor.
*//*******************************************************************/
#include "Compressor.h"
#include "LoadEffects.h"
#include <math.h>
#include <wx/brush.h>
2018-11-11 22:30:55 +00:00
#include <wx/checkbox.h>
2015-05-16 05:29:53 +00:00
#include <wx/dcclient.h>
#include <wx/intl.h>
2018-11-11 21:18:23 +00:00
#include <wx/slider.h>
2018-11-11 21:41:45 +00:00
#include <wx/stattext.h>
#include "../AColor.h"
#include "../Prefs.h"
2019-02-06 18:44:52 +00:00
#include "../Shuttle.h"
#include "../ShuttleGui.h"
#include "../Theme.h"
#include "../float_cast.h"
#include "../widgets/Ruler.h"
2015-07-03 04:20:21 +00:00
#include "../WaveTrack.h"
2017-04-26 21:32:09 +00:00
#include "../AllThemeResources.h"
2015-07-03 04:20:21 +00:00
enum
{
ID_Threshold = 10000,
ID_NoiseFloor,
ID_Ratio,
ID_Attack,
ID_Decay
};
// Define keys, defaults, minimums, and maximums for the effect parameters
//
// Name Type Key Def Min Max Scale
Param( Threshold, double, wxT("Threshold"), -12.0, -60.0, -1.0, 1 );
Param( NoiseFloor, double, wxT("NoiseFloor"), -40.0, -80.0, -20.0, 0.2 );
Param( Ratio, double, wxT("Ratio"), 2.0, 1.1, 10.0, 10 );
Param( AttackTime, double, wxT("AttackTime"), 0.2, 0.1, 5.0, 100 );
Param( ReleaseTime, double, wxT("ReleaseTime"), 1.0, 1.0, 30.0, 10 );
Param( Normalize, bool, wxT("Normalize"), true, false, true, 1 );
Param( UsePeak, bool, wxT("UsePeak"), false, false, true, 1 );
//----------------------------------------------------------------------------
// EffectCompressor
//----------------------------------------------------------------------------
const ComponentInterfaceSymbol EffectCompressor::Symbol
{ XO("Compressor") };
namespace{ BuiltinEffectsModule::Registration< EffectCompressor > reg; }
BEGIN_EVENT_TABLE(EffectCompressor, wxEvtHandler)
EVT_SLIDER(wxID_ANY, EffectCompressor::OnSlider)
END_EVENT_TABLE()
EffectCompressor::EffectCompressor()
{
mThresholdDB = DEF_Threshold;
mNoiseFloorDB = DEF_NoiseFloor;
mAttackTime = DEF_AttackTime; // seconds
mDecayTime = DEF_ReleaseTime; // seconds
mRatio = DEF_Ratio; // positive number > 1.0
mNormalize = DEF_Normalize;
mUsePeak = DEF_UsePeak;
mThreshold = 0.25;
mNoiseFloor = 0.01;
mCompression = 0.5;
mFollowLen = 0;
SetLinearEffectFlag(false);
}
EffectCompressor::~EffectCompressor()
{
}
// ComponentInterface implementation
ComponentInterfaceSymbol EffectCompressor::GetSymbol()
{
return Symbol;
}
TranslatableString EffectCompressor::GetDescription()
{
return XO("Compresses the dynamic range of audio");
}
ManualPageID EffectCompressor::ManualPage()
{
return L"Compressor";
}
Automation: AudacityCommand This is a squash of 50 commits. This merges the capabilities of BatchCommands and Effects using a new AudacityCommand class. AudacityCommand provides one function to specify the parameters, and then we leverage that one function in automation, whether by chains, mod-script-pipe or (future) Nyquist. - Now have AudacityCommand which is using the same mechanism as Effect - Has configurable parameters - Has data-entry GUI (built using shuttle GUI) - Registers with PluginManager. - Menu commands now provided in chains, and to python batch. - Tested with Zoom Toggle. - ShuttleParams now can set, get, set defaults, validate and specify the parameters. - Bugfix: Don't overwrite values with defaults first time out. - Add DefineParams function for all built-in effects. - Extend CommandContext to carry output channels for results. We abuse EffectsManager. It handles both Effects and AudacityCommands now. In time an Effect should become a special case of AudacityCommand and we'll split and rename the EffectManager class. - Don't use 'default' as a parameter name. - Massive renaming for CommandDefinitionInterface - EffectIdentInterface becomes EffectDefinitionInterface - EffectAutomationParameters becomes CommandAutomationParameters - PluginType is now a bit field. This way we can search for related types at the same time. - Most old batch commands made into AudacityCommands. The ones that weren't are for a reason. They are used by mod-script-pipe to carry commands and responses across from a non-GUI thread to the GUI thread. - Major tidy up of ScreenshotCommand - Reworking of SelectCommand - GetPreferenceCommand and SetPreferenceCommand - GetTrackInfo and SetTrackInfo - GetInfoCommand - Help, Open, Save, Import and Export commands. - Removed obsolete commands ExecMenu, GetProjectInfo and SetProjectInfo which are now better handled by other commands. - JSONify "GetInfo: Commands" output, i.e. commas in the right places. - General work on better Doxygen. - Lyrics -> LyricsPanel - Meter -> MeterPanel - Updated Linux makefile. - Scripting commands added into Extra menu. - Distinct names for previously duplicated find-clipping parameters. - Fixed longstanding error with erroneous status field number which previously caused an ASSERT in debug. - Sensible formatting of numbers in Chains, 0.1 not 0.1000000000137
2018-01-14 18:51:41 +00:00
// EffectDefinitionInterface implementation
EffectType EffectCompressor::GetType()
{
return EffectTypeProcess;
}
// EffectClientInterface implementation
Automation: AudacityCommand This is a squash of 50 commits. This merges the capabilities of BatchCommands and Effects using a new AudacityCommand class. AudacityCommand provides one function to specify the parameters, and then we leverage that one function in automation, whether by chains, mod-script-pipe or (future) Nyquist. - Now have AudacityCommand which is using the same mechanism as Effect - Has configurable parameters - Has data-entry GUI (built using shuttle GUI) - Registers with PluginManager. - Menu commands now provided in chains, and to python batch. - Tested with Zoom Toggle. - ShuttleParams now can set, get, set defaults, validate and specify the parameters. - Bugfix: Don't overwrite values with defaults first time out. - Add DefineParams function for all built-in effects. - Extend CommandContext to carry output channels for results. We abuse EffectsManager. It handles both Effects and AudacityCommands now. In time an Effect should become a special case of AudacityCommand and we'll split and rename the EffectManager class. - Don't use 'default' as a parameter name. - Massive renaming for CommandDefinitionInterface - EffectIdentInterface becomes EffectDefinitionInterface - EffectAutomationParameters becomes CommandAutomationParameters - PluginType is now a bit field. This way we can search for related types at the same time. - Most old batch commands made into AudacityCommands. The ones that weren't are for a reason. They are used by mod-script-pipe to carry commands and responses across from a non-GUI thread to the GUI thread. - Major tidy up of ScreenshotCommand - Reworking of SelectCommand - GetPreferenceCommand and SetPreferenceCommand - GetTrackInfo and SetTrackInfo - GetInfoCommand - Help, Open, Save, Import and Export commands. - Removed obsolete commands ExecMenu, GetProjectInfo and SetProjectInfo which are now better handled by other commands. - JSONify "GetInfo: Commands" output, i.e. commas in the right places. - General work on better Doxygen. - Lyrics -> LyricsPanel - Meter -> MeterPanel - Updated Linux makefile. - Scripting commands added into Extra menu. - Distinct names for previously duplicated find-clipping parameters. - Fixed longstanding error with erroneous status field number which previously caused an ASSERT in debug. - Sensible formatting of numbers in Chains, 0.1 not 0.1000000000137
2018-01-14 18:51:41 +00:00
bool EffectCompressor::DefineParams( ShuttleParams & S ){
S.SHUTTLE_PARAM( mThresholdDB, Threshold );
S.SHUTTLE_PARAM( mNoiseFloorDB, NoiseFloor );
S.SHUTTLE_PARAM( mRatio, Ratio);
S.SHUTTLE_PARAM( mAttackTime, AttackTime);
S.SHUTTLE_PARAM( mDecayTime, ReleaseTime);
S.SHUTTLE_PARAM( mNormalize, Normalize);
S.SHUTTLE_PARAM( mUsePeak, UsePeak);
return true;
}
2018-02-21 14:24:25 +00:00
bool EffectCompressor::GetAutomationParameters(CommandParameters & parms)
{
parms.Write(KEY_Threshold, mThresholdDB);
parms.Write(KEY_NoiseFloor, mNoiseFloorDB);
parms.Write(KEY_Ratio, mRatio);
parms.Write(KEY_AttackTime, mAttackTime);
parms.Write(KEY_ReleaseTime, mDecayTime);
parms.Write(KEY_Normalize, mNormalize);
parms.Write(KEY_UsePeak, mUsePeak);
return true;
}
2018-02-21 14:24:25 +00:00
bool EffectCompressor::SetAutomationParameters(CommandParameters & parms)
{
ReadAndVerifyDouble(Threshold);
ReadAndVerifyDouble(NoiseFloor);
ReadAndVerifyDouble(Ratio);
ReadAndVerifyDouble(AttackTime);
ReadAndVerifyDouble(ReleaseTime);
ReadAndVerifyBool(Normalize);
ReadAndVerifyBool(UsePeak);
mThresholdDB = Threshold;
mNoiseFloorDB = NoiseFloor;
mRatio = Ratio;
mAttackTime = AttackTime;
mDecayTime = ReleaseTime;
mNormalize = Normalize;
mUsePeak = UsePeak;
return true;
}
2020-04-11 07:08:33 +00:00
// Effect Implementation
bool EffectCompressor::Startup()
{
wxString base = wxT("/Effects/Compressor/");
// Migrate settings from 2.1.0 or before
// Already migrated, so bail
if (gPrefs->Exists(base + wxT("Migrated")))
{
return true;
}
// Load the old "current" settings
if (gPrefs->Exists(base))
{
gPrefs->Read(base + wxT("ThresholdDB"), &mThresholdDB, -12.0f );
gPrefs->Read(base + wxT("NoiseFloorDB"), &mNoiseFloorDB, -40.0f );
gPrefs->Read(base + wxT("Ratio"), &mRatio, 2.0f );
gPrefs->Read(base + wxT("AttackTime"), &mAttackTime, 0.2f );
gPrefs->Read(base + wxT("DecayTime"), &mDecayTime, 1.0f );
gPrefs->Read(base + wxT("Normalize"), &mNormalize, true );
gPrefs->Read(base + wxT("UsePeak"), &mUsePeak, false );
SaveUserPreset(GetCurrentSettingsGroup());
// Do not migrate again
gPrefs->Write(base + wxT("Migrated"), true);
gPrefs->Flush();
}
return true;
}
namespace {
TranslatableString ThresholdFormat( int value )
/* i18n-hint: usually leave this as is as dB doesn't get translated*/
{ return XO("%3d dB").Format(value); }
TranslatableString AttackTimeFormat( double value )
{ return XO("%.2f secs").Format( value ); }
TranslatableString DecayTimeFormat( double value )
{ return XO("%.1f secs").Format( value ); }
TranslatableString RatioTextFormat( int sliderValue, double value )
{
auto format = (sliderValue % 10 == 0)
/* i18n-hint: Unless your language has a different convention for ratios,
* like 8:1, leave as is.*/
? XO("%.0f:1")
/* i18n-hint: Unless your language has a different convention for ratios,
* like 8:1, leave as is.*/
: XO("%.1f:1");
return format.Format( value );
}
TranslatableString RatioLabelFormat( int sliderValue, double value )
{
auto format = (sliderValue % 10 == 0)
? XO("Ratio %.0f to 1")
: XO("Ratio %.1f to 1");
return format.Format( value );
}
}
void EffectCompressor::PopulateOrExchange(ShuttleGui & S)
{
S.SetBorder(5);
S.StartHorizontalLay(wxEXPAND, true);
{
S.SetBorder(10);
mPanel = safenew EffectCompressorPanel(S.GetParent(), wxID_ANY,
mThresholdDB,
mNoiseFloorDB,
mRatio);
S.Prop(true)
.Position(wxEXPAND | wxALL)
.MinSize( { 400, 200 } )
.AddWindow(mPanel);
S.SetBorder(5);
}
S.EndHorizontalLay();
S.StartStatic( {} );
{
S.StartMultiColumn(3, wxEXPAND);
{
S.SetStretchyCol(1);
mThresholdLabel = S.AddVariableText(XO("&Threshold:"), true,
wxALIGN_RIGHT | wxALIGN_CENTER_VERTICAL);
mThresholdSlider = S.Id(ID_Threshold)
.Name(XO("Threshold"))
.Style(wxSL_HORIZONTAL)
.AddSlider( {},
DEF_Threshold * SCL_Threshold,
MAX_Threshold * SCL_Threshold,
MIN_Threshold * SCL_Threshold);
mThresholdText = S.AddVariableText(ThresholdFormat(999), true,
wxALIGN_LEFT | wxALIGN_CENTER_VERTICAL);
mNoiseFloorLabel = S.AddVariableText(XO("&Noise Floor:"), true,
wxALIGN_RIGHT | wxALIGN_CENTER_VERTICAL);
mNoiseFloorSlider = S.Id(ID_NoiseFloor)
.Name(XO("Noise Floor"))
.Style(wxSL_HORIZONTAL)
.AddSlider( {},
DEF_NoiseFloor * SCL_NoiseFloor,
MAX_NoiseFloor * SCL_NoiseFloor,
MIN_NoiseFloor * SCL_NoiseFloor);
mNoiseFloorText = S.AddVariableText(ThresholdFormat(999),
true, wxALIGN_LEFT | wxALIGN_CENTER_VERTICAL);
mRatioLabel = S.AddVariableText(XO("&Ratio:"), true,
wxALIGN_RIGHT | wxALIGN_CENTER_VERTICAL);
mRatioSlider = S.Id(ID_Ratio)
.Name(XO("Ratio"))
.Style(wxSL_HORIZONTAL)
.AddSlider( {},
DEF_Ratio * SCL_Ratio,
MAX_Ratio * SCL_Ratio,
MIN_Ratio * SCL_Ratio);
mRatioSlider->SetPageSize(5);
mRatioText = S.AddVariableText(RatioTextFormat( 1, 99.9 ), true,
wxALIGN_LEFT | wxALIGN_CENTER_VERTICAL);
/* i18n-hint: Particularly in percussion, sounds can be regarded as having
* an 'attack' phase where the sound builds up and a 'decay' where the
* sound dies away. So this means 'onset duration'. */
mAttackLabel = S.AddVariableText(XO("&Attack Time:"), true,
wxALIGN_RIGHT | wxALIGN_CENTER_VERTICAL);
mAttackSlider = S.Id(ID_Attack)
/* i18n-hint: Particularly in percussion, sounds can be regarded as having
* an 'attack' phase where the sound builds up and a 'decay' where the
* sound dies away. So this means 'onset duration'. */
.Name(XO("Attack Time"))
.Style(wxSL_HORIZONTAL)
.AddSlider( {},
DEF_AttackTime * SCL_AttackTime,
MAX_AttackTime * SCL_AttackTime,
MIN_AttackTime * SCL_AttackTime);
mAttackText = S.AddVariableText(
AttackTimeFormat(9.99),
true, wxALIGN_LEFT | wxALIGN_CENTER_VERTICAL);
/* i18n-hint: Particularly in percussion, sounds can be regarded as having
* an 'attack' phase where the sound builds up and a 'decay' or 'release' where the
* sound dies away. */
mDecayLabel = S.AddVariableText(XO("R&elease Time:"), true,
wxALIGN_RIGHT | wxALIGN_CENTER_VERTICAL);
mDecaySlider = S.Id(ID_Decay)
/* i18n-hint: Particularly in percussion, sounds can be regarded as having
* an 'attack' phase where the sound builds up and a 'decay' or 'release' where the
* sound dies away. */
.Name(XO("Release Time"))
.Style(wxSL_HORIZONTAL)
.AddSlider( {},
DEF_ReleaseTime * SCL_ReleaseTime,
MAX_ReleaseTime * SCL_ReleaseTime,
MIN_ReleaseTime * SCL_ReleaseTime);
mDecayText = S.AddVariableText(
DecayTimeFormat(99.9),
true, wxALIGN_LEFT | wxALIGN_CENTER_VERTICAL);
}
S.EndMultiColumn();
}
S.EndStatic();
S.StartHorizontalLay(wxCENTER, false);
{
/* i18n-hint: Make-up, i.e. correct for any reduction, rather than fabricate it.*/
mGainCheckBox = S.AddCheckBox(XXO("Ma&ke-up gain for 0 dB after compressing"),
DEF_Normalize);
/* i18n-hint: "Compress" here means reduce variations of sound volume,
NOT related to file-size compression; Peaks means extremes in volume */
mPeakCheckBox = S.AddCheckBox(XXO("C&ompress based on Peaks"),
DEF_UsePeak);
}
S.EndHorizontalLay();
}
bool EffectCompressor::TransferDataToWindow()
{
mThresholdSlider->SetValue(lrint(mThresholdDB));
mNoiseFloorSlider->SetValue(lrint(mNoiseFloorDB * SCL_NoiseFloor));
mRatioSlider->SetValue(lrint(mRatio * SCL_Ratio));
mAttackSlider->SetValue(lrint(mAttackTime * SCL_AttackTime));
mDecaySlider->SetValue(lrint(mDecayTime * SCL_ReleaseTime));
mGainCheckBox->SetValue(mNormalize);
mPeakCheckBox->SetValue(mUsePeak);
UpdateUI();
return true;
}
bool EffectCompressor::TransferDataFromWindow()
{
if (!mUIParent->Validate())
{
return false;
}
mThresholdDB = (double) mThresholdSlider->GetValue();
mNoiseFloorDB = (double) mNoiseFloorSlider->GetValue() / SCL_NoiseFloor;
mRatio = (double) mRatioSlider->GetValue() / SCL_Ratio;
mAttackTime = (double) mAttackSlider->GetValue() / 100.0; //SCL_AttackTime;
mDecayTime = (double) mDecaySlider->GetValue() / SCL_ReleaseTime;
mNormalize = mGainCheckBox->GetValue();
mUsePeak = mPeakCheckBox->GetValue();
return true;
}
// EffectTwoPassSimpleMono implementation
bool EffectCompressor::NewTrackPass1()
{
mThreshold = DB_TO_LINEAR(mThresholdDB);
mNoiseFloor = DB_TO_LINEAR(mNoiseFloorDB);
mNoiseCounter = 100;
mAttackInverseFactor = exp(log(mThreshold) / (mCurRate * mAttackTime + 0.5));
mAttackFactor = 1.0 / mAttackInverseFactor;
mDecayFactor = exp(log(mThreshold) / (mCurRate * mDecayTime + 0.5));
if(mRatio > 1)
mCompression = 1.0-1.0/mRatio;
else
mCompression = 0.0;
mLastLevel = mThreshold;
mCircleSize = 100;
2016-04-14 16:35:15 +00:00
mCircle.reinit( mCircleSize, true );
mCirclePos = 0;
mRMSSum = 0.0;
return true;
}
bool EffectCompressor::InitPass1()
{
mMax=0.0;
if (!mNormalize)
DisableSecondPass();
// Find the maximum block length required for any track
size_t maxlen = inputTracks()->Selected< const WaveTrack >().max(
&WaveTrack::GetMaxBlockSize
);
2016-04-14 16:35:15 +00:00
mFollow1.reset();
mFollow2.reset();
// Allocate buffers for the envelope
if(maxlen > 0) {
2016-04-14 16:35:15 +00:00
mFollow1.reinit(maxlen);
mFollow2.reinit(maxlen);
}
mFollowLen = maxlen;
return true;
}
bool EffectCompressor::InitPass2()
{
// Actually, this should not even be called, because we call
// DisableSecondPass() before, if mNormalize is false.
return mNormalize;
}
// Process the input with 2 buffers available at a time
// buffer1 will be written upon return
// buffer2 will be passed as buffer1 on the next call
bool EffectCompressor::TwoBufferProcessPass1
(float *buffer1, size_t len1, float *buffer2, size_t len2)
{
// If buffers are bigger than allocated, then abort
// (this should never happen, but if it does, we don't want to crash)
if((len1 > mFollowLen) || (len2 > mFollowLen))
return false;
// This makes sure that the initial value is well-chosen
// buffer1 == NULL on the first and only the first call
if (buffer1 == NULL) {
// Initialize the mLastLevel to the peak level in the first buffer
// This avoids problems with large spike events near the beginning of the track
mLastLevel = mThreshold;
for(size_t i=0; i<len2; i++) {
if(mLastLevel < fabs(buffer2[i]))
mLastLevel = fabs(buffer2[i]);
}
}
// buffer2 is NULL on the last and only the last call
if(buffer2 != NULL) {
2016-04-14 16:35:15 +00:00
Follow(buffer2, mFollow2.get(), len2, mFollow1.get(), len1);
}
if(buffer1 != NULL) {
for (size_t i = 0; i < len1; i++) {
buffer1[i] = DoCompression(buffer1[i], mFollow1[i]);
}
}
#if 0
// Copy the envelope over the track data (for debug purposes)
memcpy(buffer1, mFollow1, len1*sizeof(float));
#endif
// Rotate the buffer pointers
2016-04-14 16:35:15 +00:00
mFollow1.swap(mFollow2);
return true;
}
bool EffectCompressor::ProcessPass2(float *buffer, size_t len)
{
if (mMax != 0)
{
for (size_t i = 0; i < len; i++)
buffer[i] /= mMax;
}
return true;
}
void EffectCompressor::FreshenCircle()
{
// Recompute the RMS sum periodically to prevent accumulation of rounding errors
// during long waveforms
mRMSSum = 0;
2016-04-14 16:35:15 +00:00
for(size_t i=0; i<mCircleSize; i++)
mRMSSum += mCircle[i];
}
float EffectCompressor::AvgCircle(float value)
{
float level;
// Calculate current level from root-mean-squared of
// circular buffer ("RMS")
mRMSSum -= mCircle[mCirclePos];
mCircle[mCirclePos] = value*value;
mRMSSum += mCircle[mCirclePos];
level = sqrt(mRMSSum/mCircleSize);
mCirclePos = (mCirclePos+1)%mCircleSize;
return level;
}
void EffectCompressor::Follow(float *buffer, float *env, size_t len, float *previous, size_t previous_len)
{
/*
"Follow"ing algorithm by Roger B. Dannenberg, taken from
Nyquist. His description follows. -DMM
Description: this is a sophisticated envelope follower.
The input is an envelope, e.g. something produced with
the AVG function. The purpose of this function is to
generate a smooth envelope that is generally not less
than the input signal. In other words, we want to "ride"
the peaks of the signal with a smooth function. The
algorithm is as follows: keep a current output value
(called the "value"). The value is allowed to increase
by at most rise_factor and decrease by at most fall_factor.
Therefore, the next value should be between
value * rise_factor and value * fall_factor. If the input
is in this range, then the next value is simply the input.
If the input is less than value * fall_factor, then the
next value is just value * fall_factor, which will be greater
than the input signal. If the input is greater than value *
rise_factor, then we compute a rising envelope that meets
the input value by working bacwards in time, changing the
previous values to input / rise_factor, input / rise_factor^2,
input / rise_factor^3, etc. until this NEW envelope intersects
the previously computed values. There is only a limited buffer
in which we can work backwards, so if the NEW envelope does not
intersect the old one, then make yet another pass, this time
from the oldest buffered value forward, increasing on each
sample by rise_factor to produce a maximal envelope. This will
still be less than the input.
The value has a lower limit of floor to make sure value has a
reasonable positive value from which to begin an attack.
*/
double level,last;
if(!mUsePeak) {
// Update RMS sum directly from the circle buffer
// to avoid accumulation of rounding errors
FreshenCircle();
}
// First apply a peak detect with the requested decay rate
last = mLastLevel;
for(size_t i=0; i<len; i++) {
if(mUsePeak)
level = fabs(buffer[i]);
else // use RMS
level = AvgCircle(buffer[i]);
// Don't increase gain when signal is continuously below the noise floor
if(level < mNoiseFloor) {
mNoiseCounter++;
} else {
mNoiseCounter = 0;
}
if(mNoiseCounter < 100) {
last *= mDecayFactor;
if(last < mThreshold)
last = mThreshold;
if(level > last)
last = level;
}
env[i] = last;
}
mLastLevel = last;
// Next do the same process in reverse direction to get the requested attack rate
last = mLastLevel;
for(size_t i = len; i--;) {
last *= mAttackInverseFactor;
if(last < mThreshold)
last = mThreshold;
if(env[i] < last)
env[i] = last;
else
last = env[i];
}
if((previous != NULL) && (previous_len > 0)) {
// If the previous envelope was passed, propagate the rise back until we intersect
for(size_t i = previous_len; i--;) {
last *= mAttackInverseFactor;
if(last < mThreshold)
last = mThreshold;
if(previous[i] < last)
previous[i] = last;
else // Intersected the previous envelope buffer, so we are finished
return;
}
// If we can't back up far enough, project the starting level forward
// until we intersect the desired envelope
last = previous[0];
for(size_t i=1; i<previous_len; i++) {
last *= mAttackFactor;
if(previous[i] > last)
previous[i] = last;
else // Intersected the desired envelope, so we are finished
return;
}
// If we still didn't intersect, then continue ramp up into current buffer
for(size_t i=0; i<len; i++) {
last *= mAttackFactor;
if(buffer[i] > last)
buffer[i] = last;
else // Finally got an intersect
return;
}
// If we still didn't intersect, then reset mLastLevel
mLastLevel = last;
}
}
float EffectCompressor::DoCompression(float value, double env)
{
float out;
if(mUsePeak) {
// Peak values map 1.0 to 1.0 - 'upward' compression
out = value * pow(1.0/env, mCompression);
} else {
// With RMS-based compression don't change values below mThreshold - 'downward' compression
out = value * pow(mThreshold/env, mCompression);
}
// Retain the maximum value for use in the normalization pass
if(mMax < fabs(out))
mMax = fabs(out);
return out;
}
void EffectCompressor::OnSlider(wxCommandEvent & WXUNUSED(evt))
{
TransferDataFromWindow();
UpdateUI();
}
void EffectCompressor::UpdateUI()
{
mThresholdLabel->SetName(wxString::Format(_("Threshold %d dB"), (int) mThresholdDB));
mThresholdText->SetLabel(ThresholdFormat((int) mThresholdDB).Translation());
mThresholdText->SetName(mThresholdText->GetLabel()); // fix for bug 577 (NVDA/Narrator screen readers do not read static text in dialogs)
mNoiseFloorLabel->SetName(wxString::Format(_("Noise Floor %d dB"), (int) mNoiseFloorDB));
mNoiseFloorText->SetLabel(ThresholdFormat((int) mNoiseFloorDB).Translation());
mNoiseFloorText->SetName(mNoiseFloorText->GetLabel()); // fix for bug 577 (NVDA/Narrator screen readers do not read static text in dialogs)
mRatioLabel->SetName(
RatioLabelFormat(mRatioSlider->GetValue(), mRatio).Translation());
mRatioText->SetLabel(
RatioTextFormat(mRatioSlider->GetValue(), mRatio).Translation());
mRatioText->SetName(mRatioText->GetLabel()); // fix for bug 577 (NVDA/Narrator screen readers do not read static text in dialogs)
mAttackLabel->SetName(wxString::Format(_("Attack Time %.2f secs"), mAttackTime));
mAttackText->SetLabel(AttackTimeFormat(mAttackTime).Translation());
mAttackText->SetName(mAttackText->GetLabel()); // fix for bug 577 (NVDA/Narrator screen readers do not read static text in dialogs)
mDecayLabel->SetName(wxString::Format(_("Release Time %.1f secs"), mDecayTime));
mDecayText->SetLabel(DecayTimeFormat(mDecayTime).Translation());
mDecayText->SetName(mDecayText->GetLabel()); // fix for bug 577 (NVDA/Narrator screen readers do not read static text in dialogs)
mPanel->Refresh(false);
return;
}
//----------------------------------------------------------------------------
// EffectCompressorPanel
//----------------------------------------------------------------------------
BEGIN_EVENT_TABLE(EffectCompressorPanel, wxPanelWrapper)
EVT_PAINT(EffectCompressorPanel::OnPaint)
EVT_SIZE(EffectCompressorPanel::OnSize)
END_EVENT_TABLE()
EffectCompressorPanel::EffectCompressorPanel(wxWindow *parent, wxWindowID winid,
double & threshold,
double & noiseFloor,
double & ratio)
: wxPanelWrapper(parent, winid),
threshold(threshold),
noiseFloor(noiseFloor),
ratio(ratio)
{
}
void EffectCompressorPanel::OnPaint(wxPaintEvent & WXUNUSED(evt))
{
wxPaintDC dc(this);
int width, height;
GetSize(&width, &height);
double rangeDB = 60;
// Ruler
int w = 0;
int h = 0;
Ruler vRuler;
2015-05-19 07:37:26 +00:00
vRuler.SetBounds(0, 0, width, height);
vRuler.SetOrientation(wxVERTICAL);
vRuler.SetRange(0, -rangeDB);
vRuler.SetFormat(Ruler::LinearDBFormat);
vRuler.SetUnits(XO("dB"));
vRuler.GetMaxSize(&w, NULL);
2014-06-03 20:30:19 +00:00
Ruler hRuler;
2015-05-19 07:37:26 +00:00
hRuler.SetBounds(0, 0, width, height);
hRuler.SetOrientation(wxHORIZONTAL);
hRuler.SetRange(-rangeDB, 0);
hRuler.SetFormat(Ruler::LinearDBFormat);
hRuler.SetUnits(XO("dB"));
hRuler.SetFlip(true);
hRuler.GetMaxSize(NULL, &h);
2015-05-19 07:37:26 +00:00
vRuler.SetBounds(0, 0, w, height - h);
hRuler.SetBounds(w, height - h, width, height);
2017-04-26 21:32:09 +00:00
vRuler.SetTickColour( theTheme.Colour( clrGraphLabels ));
hRuler.SetTickColour( theTheme.Colour( clrGraphLabels ));
2015-05-19 07:37:26 +00:00
#if defined(__WXMSW__)
dc.Clear();
#endif
wxRect border;
border.x = w;
border.y = 0;
2015-05-19 07:37:26 +00:00
border.width = width - w;
border.height = height - h + 1;
2015-05-19 07:37:26 +00:00
dc.SetBrush(*wxWHITE_BRUSH);
dc.SetPen(*wxBLACK_PEN);
dc.DrawRectangle(border);
2015-05-19 07:37:26 +00:00
wxRect envRect = border;
envRect.Deflate( 2, 2 );
2015-05-19 07:37:26 +00:00
int kneeX = lrint((rangeDB+threshold)*envRect.width/rangeDB);
int kneeY = lrint((rangeDB+threshold/ratio)*envRect.height/rangeDB);
2015-05-19 07:37:26 +00:00
int finalY = envRect.height;
int startY = lrint((threshold*(1.0/ratio-1.0))*envRect.height/rangeDB);
// Yellow line for threshold
2015-05-19 07:37:26 +00:00
/* dc.SetPen(wxPen(wxColour(220, 220, 0), 1, wxSOLID));
AColor::Line(dc,
envRect.x,
envRect.y + envRect.height - kneeY,
envRect.x + envRect.width - 1,
envRect.y + envRect.height - kneeY);*/
// Was: Nice dark red line for the compression diagram
2015-05-19 07:37:26 +00:00
// dc.SetPen(wxPen(wxColour(180, 40, 40), 3, wxSOLID));
// Nice blue line for compressor, same color as used in the waveform envelope.
2015-05-19 07:37:26 +00:00
dc.SetPen( AColor::WideEnvelopePen) ;
2015-05-19 07:37:26 +00:00
AColor::Line(dc,
envRect.x,
envRect.y + envRect.height - startY,
envRect.x + kneeX - 1,
envRect.y + envRect.height - kneeY);
2015-05-19 07:37:26 +00:00
AColor::Line(dc,
envRect.x + kneeX,
envRect.y + envRect.height - kneeY,
envRect.x + envRect.width - 1,
envRect.y + envRect.height - finalY);
// Paint border again
2015-05-19 07:37:26 +00:00
dc.SetBrush(*wxTRANSPARENT_BRUSH);
dc.SetPen(*wxBLACK_PEN);
dc.DrawRectangle(border);
2015-05-19 07:37:26 +00:00
vRuler.Draw(dc);
hRuler.Draw(dc);
}
void EffectCompressorPanel::OnSize(wxSizeEvent & WXUNUSED(evt))
{
Refresh(false);
}