audacia/src/effects/FindClipping.cpp

278 lines
6.6 KiB
C++
Raw Normal View History

/**********************************************************************
Audacity: A Digital Audio Editor
FindClipping.cpp
Leland Lucius
*******************************************************************//**
\class EffectFindClipping
\brief Locates clipping and inserts labels when found
*//****************************************************************//**
\class FindClippingDialog
\brief FindClippingDialog used with EffectFindClipping
*//*******************************************************************/
#include "FindClipping.h"
#include "LoadEffects.h"
#include <math.h>
#include <wx/intl.h>
2019-02-06 18:44:52 +00:00
#include "../Shuttle.h"
#include "../ShuttleGui.h"
#include "../widgets/valnum.h"
#include "../widgets/AudacityMessageBox.h"
2015-07-04 17:58:01 +00:00
#include "../LabelTrack.h"
2015-07-28 14:48:21 +00:00
#include "../WaveTrack.h"
2015-07-03 04:20:21 +00:00
// Define keys, defaults, minimums, and maximums for the effect parameters
//
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
// Name Type Key Def Min Max Scale
Param( Start, int, wxT("Duty Cycle Start"), 3, 1, INT_MAX, 1 );
Param( Stop, int, wxT("Duty Cycle End"), 3, 1, INT_MAX, 1 );
const ComponentInterfaceSymbol EffectFindClipping::Symbol
{ XO("Find Clipping") };
namespace{ BuiltinEffectsModule::Registration< EffectFindClipping > reg; }
EffectFindClipping::EffectFindClipping()
{
mStart = DEF_Start;
mStop = DEF_Stop;
}
EffectFindClipping::~EffectFindClipping()
{
2014-06-03 20:30:19 +00:00
}
// ComponentInterface implementation
ComponentInterfaceSymbol EffectFindClipping::GetSymbol()
{
return Symbol;
}
TranslatableString EffectFindClipping::GetDescription()
{
return XO("Creates labels where clipping is detected");
}
ManualPageID EffectFindClipping::ManualPage()
{
return L"Find_Clipping";
}
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 EffectFindClipping::GetType()
{
return EffectTypeAnalyze;
}
// 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 EffectFindClipping::DefineParams( ShuttleParams & S ){
S.SHUTTLE_PARAM( mStart, Start );
S.SHUTTLE_PARAM( mStop, Stop );
return true;
}
2018-02-21 14:24:25 +00:00
bool EffectFindClipping::GetAutomationParameters(CommandParameters & parms)
{
parms.Write(KEY_Start, mStart);
parms.Write(KEY_Stop, mStop);
return true;
}
2018-02-21 14:24:25 +00:00
bool EffectFindClipping::SetAutomationParameters(CommandParameters & parms)
2014-06-03 20:30:19 +00:00
{
ReadAndVerifyInt(Start);
ReadAndVerifyInt(Stop);
mStart = Start;
mStop = Stop;
return true;
}
// Effect implementation
bool EffectFindClipping::Process()
{
std::shared_ptr<AddedAnalysisTrack> addedTrack;
Optional<ModifiedAnalysisTrack> modifiedTrack;
const wxString name{ _("Clipping") };
auto clt = *inputTracks()->Any< const LabelTrack >().find_if(
[&]( const Track *track ){ return track->GetName() == name; } );
2014-06-03 20:30:19 +00:00
LabelTrack *lt{};
if (!clt)
addedTrack = (AddAnalysisTrack(name)), lt = addedTrack->get();
else
modifiedTrack.emplace(ModifyAnalysisTrack(clt, name)),
lt = modifiedTrack->get();
int count = 0;
// JC: Only process selected tracks.
for (auto t : inputTracks()->Selected< const WaveTrack >()) {
double trackStart = t->GetStartTime();
double trackEnd = t->GetEndTime();
double t0 = mT0 < trackStart ? trackStart : mT0;
double t1 = mT1 > trackEnd ? trackEnd : mT1;
if (t1 > t0) {
auto start = t->TimeToLongSamples(t0);
auto end = t->TimeToLongSamples(t1);
auto len = end - start;
if (!ProcessOne(lt, count, t, start, len)) {
return false;
}
}
count++;
}
// No cancellation, so commit the addition of the track.
if (addedTrack)
addedTrack->Commit();
if (modifiedTrack)
modifiedTrack->Commit();
return true;
}
bool EffectFindClipping::ProcessOne(LabelTrack * lt,
int count,
const WaveTrack * wt,
sampleCount start,
sampleCount len)
{
bool bGoodResult = true;
size_t blockSize = (mStart * 1000);
2014-06-03 20:30:19 +00:00
if (len < mStart) {
return true;
}
2016-04-14 16:35:15 +00:00
Floats buffer;
try {
// mStart should be positive.
// if we are throwing bad_alloc and mStart is negative, find out why.
if (mStart < 0 || (int)blockSize < mStart)
// overflow
throw std::bad_alloc{};
2016-04-14 16:35:15 +00:00
buffer.reinit(blockSize);
}
catch( const std::bad_alloc & ) {
Effect::MessageBox( XO("Requested value exceeds memory capacity.") );
return false;
}
2016-04-14 16:35:15 +00:00
float *ptr = buffer.get();
decltype(len) s = 0, startrun = 0, stoprun = 0, samps = 0;
decltype(blockSize) block = 0;
double startTime = -1.0;
2014-06-03 20:30:19 +00:00
while (s < len) {
if (block == 0) {
if (TrackProgress(count,
s.as_double() /
len.as_double() )) {
bGoodResult = false;
break;
}
block = limitSampleBufferSize( blockSize, len - s );
wt->GetFloats(buffer.get(), start + s, block);
2016-04-14 16:35:15 +00:00
ptr = buffer.get();
}
float v = fabs(*ptr++);
if (v >= MAX_AUDIO) {
if (startrun == 0) {
startTime = wt->LongSamplesToTime(start + s);
samps = 0;
}
else {
stoprun = 0;
}
startrun++;
samps++;
}
else {
if (startrun >= mStart) {
stoprun++;
samps++;
if (stoprun >= mStop) {
lt->AddLabel(SelectedRegion(startTime,
wt->LongSamplesToTime(start + s - mStop)),
wxString::Format(wxT("%lld of %lld"), startrun.as_long_long(), (samps - mStop).as_long_long()));
startrun = 0;
stoprun = 0;
samps = 0;
}
}
else {
startrun = 0;
}
}
s++;
block--;
}
return bGoodResult;
}
void EffectFindClipping::PopulateOrExchange(ShuttleGui & S)
{
S.StartMultiColumn(2, wxALIGN_CENTER);
{
2017-10-30 16:23:41 +00:00
S.Validator<IntegerValidator<int>>(
&mStart, NumValidatorStyle::DEFAULT, MIN_Start)
.TieTextBox(XXO("&Start threshold (samples):"), mStart, 10);
2017-10-30 16:23:41 +00:00
S.Validator<IntegerValidator<int>>(
&mStop, NumValidatorStyle::DEFAULT, MIN_Stop)
.TieTextBox(XXO("St&op threshold (samples):"), mStop, 10);
}
S.EndMultiColumn();
}
bool EffectFindClipping::TransferDataToWindow()
{
ShuttleGui S(mUIParent, eIsSettingToDialog);
PopulateOrExchange(S);
return true;
}
bool EffectFindClipping::TransferDataFromWindow()
{
if (!mUIParent->Validate())
{
return false;
}
ShuttleGui S(mUIParent, eIsGettingFromDialog);
PopulateOrExchange(S);
return true;
}