1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
/*!********************************************************************
Audacity: A Digital Audio Editor
DownwardMeterValueProvider.cpp
Matthieu Hodgkinson
**********************************************************************/
#include "DownwardMeterValueProvider.h"
#include <algorithm>
#include <cassert>
namespace
{
constexpr auto decayPerSecondDb = 10.f;
constexpr auto decayPerTickDb =
decayPerSecondDb * compressorMeterUpdatePeriodMs / 1000.f;
constexpr auto maxDelayMs = 5000;
constexpr auto maxTickCount = maxDelayMs / compressorMeterUpdatePeriodMs;
} // namespace
DownwardMeterValueProvider::DownwardMeterValueProvider(float upperValue)
: mUpperValue { upperValue }
, mGlobalMin { upperValue }
, mCurrentMin { upperValue }
, mFiveSecMinState { upperValue }
{
mRingBuffer.fill(upperValue);
}
void DownwardMeterValueProvider::Update(float newValue, bool alsoFiveSecondMax)
{
++mTimerCount;
const auto value = mRingBuffer[mRingBufferIndex];
mRingBuffer[mRingBufferIndex] = newValue;
mRingBufferIndex = (mRingBufferIndex + 1) % ringBufferLength;
if (value < mCurrentMin)
{
mCurrentMin = value;
mGlobalMin = std::min(mGlobalMin, value);
}
else
mCurrentMin = std::min(mCurrentMin + decayPerTickDb, mUpperValue);
mLastFiveSeconds.emplace_back(mTimerCount, value);
while (!mLastFiveSeconds.empty() &&
mLastFiveSeconds.front().first < mTimerCount - maxTickCount)
mLastFiveSeconds.erase(mLastFiveSeconds.begin());
if (!mLastFiveSeconds.empty() && alsoFiveSecondMax)
{
const auto rawMin =
std::min_element(
mLastFiveSeconds.begin(), mLastFiveSeconds.end(),
[](const auto& a, const auto& b) { return a.second < b.second; })
->second;
if (rawMin <= mFiveSecMinState)
mFiveSecMinState = rawMin;
else
mFiveSecMinState =
std::min(mFiveSecMinState + decayPerTickDb, mUpperValue);
}
}
float DownwardMeterValueProvider::GetGlobalMax() const
{
return mGlobalMin;
}
float DownwardMeterValueProvider::GetFiveSecMax() const
{
return mFiveSecMinState;
}
float DownwardMeterValueProvider::GetCurrentMax() const
{
return mCurrentMin;
}
DownwardMeterValueProvider::Direction
DownwardMeterValueProvider::GetDirection() const
{
return Direction::Downwards;
}
bool DownwardMeterValueProvider::IsInvisible() const
{
return mCurrentMin >= mUpperValue;
}
|