2016-02-23 23:16:51 +00:00
|
|
|
#include "volume-control.hpp"
|
|
|
|
#include "qt-wrappers.hpp"
|
2017-04-19 19:54:15 +00:00
|
|
|
#include "obs-app.hpp"
|
2016-02-23 23:16:51 +00:00
|
|
|
#include "mute-checkbox.hpp"
|
|
|
|
#include "slider-absoluteset-style.hpp"
|
2018-02-19 19:54:37 +00:00
|
|
|
#include <obs-audio-controls.h>
|
2016-02-23 23:16:51 +00:00
|
|
|
#include <util/platform.h>
|
2017-04-19 19:54:15 +00:00
|
|
|
#include <util/threading.h>
|
2018-02-19 19:54:37 +00:00
|
|
|
#include <QFontDatabase>
|
2016-02-23 23:16:51 +00:00
|
|
|
#include <QHBoxLayout>
|
|
|
|
#include <QVBoxLayout>
|
|
|
|
#include <QPushButton>
|
|
|
|
#include <QVariant>
|
|
|
|
#include <QSlider>
|
|
|
|
#include <QLabel>
|
|
|
|
#include <QPainter>
|
|
|
|
#include <QTimer>
|
|
|
|
#include <string>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
#define CLAMP(x, min, max) ((x) < min ? min : ((x) > max ? max : (x)))
|
|
|
|
|
2017-04-19 19:54:15 +00:00
|
|
|
QWeakPointer<VolumeMeterTimer> VolumeMeter::updateTimer;
|
|
|
|
|
2016-02-23 23:16:51 +00:00
|
|
|
void VolControl::OBSVolumeChanged(void *data, float db)
|
|
|
|
{
|
|
|
|
Q_UNUSED(db);
|
|
|
|
VolControl *volControl = static_cast<VolControl*>(data);
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(volControl, "VolumeChanged");
|
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolControl::OBSVolumeLevel(void *data,
|
|
|
|
const float magnitude[MAX_AUDIO_CHANNELS],
|
|
|
|
const float peak[MAX_AUDIO_CHANNELS],
|
|
|
|
const float inputPeak[MAX_AUDIO_CHANNELS])
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
|
|
|
VolControl *volControl = static_cast<VolControl*>(data);
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
volControl->volMeter->setLevels(magnitude, peak, inputPeak);
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::OBSVolumeMuted(void *data, calldata_t *calldata)
|
|
|
|
{
|
|
|
|
VolControl *volControl = static_cast<VolControl*>(data);
|
|
|
|
bool muted = calldata_bool(calldata, "muted");
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(volControl, "VolumeMuted",
|
|
|
|
Q_ARG(bool, muted));
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::VolumeChanged()
|
|
|
|
{
|
|
|
|
slider->blockSignals(true);
|
|
|
|
slider->setValue((int) (obs_fader_get_deflection(obs_fader) * 100.0f));
|
|
|
|
slider->blockSignals(false);
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
updateText();
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::VolumeMuted(bool muted)
|
|
|
|
{
|
|
|
|
if (mute->isChecked() != muted)
|
|
|
|
mute->setChecked(muted);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::SetMuted(bool checked)
|
|
|
|
{
|
|
|
|
obs_source_set_muted(source, checked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::SliderChanged(int vol)
|
|
|
|
{
|
|
|
|
obs_fader_set_deflection(obs_fader, float(vol) * 0.01f);
|
|
|
|
updateText();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::updateText()
|
|
|
|
{
|
2017-04-19 19:54:15 +00:00
|
|
|
QString db = QString::number(obs_fader_get_db(obs_fader), 'f', 1)
|
|
|
|
.append(" dB");
|
|
|
|
volLabel->setText(db);
|
|
|
|
|
|
|
|
bool muted = obs_source_muted(source);
|
|
|
|
const char *accTextLookup = muted
|
|
|
|
? "VolControl.SliderMuted"
|
|
|
|
: "VolControl.SliderUnmuted";
|
|
|
|
|
|
|
|
QString sourceName = obs_source_get_name(source);
|
|
|
|
QString accText = QTStr(accTextLookup).arg(sourceName, db);
|
|
|
|
|
|
|
|
slider->setAccessibleName(accText);
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QString VolControl::GetName() const
|
|
|
|
{
|
|
|
|
return nameLabel->text();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::SetName(const QString &newName)
|
|
|
|
{
|
|
|
|
nameLabel->setText(newName);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::EmitConfigClicked()
|
|
|
|
{
|
|
|
|
emit ConfigClicked();
|
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolControl::SetMeterDecayRate(qreal q)
|
|
|
|
{
|
|
|
|
volMeter->setPeakDecayRate(q);
|
|
|
|
}
|
|
|
|
|
2016-02-23 23:16:51 +00:00
|
|
|
VolControl::VolControl(OBSSource source_, bool showConfig)
|
|
|
|
: source (source_),
|
|
|
|
levelTotal (0.0f),
|
|
|
|
levelCount (0.0f),
|
|
|
|
obs_fader (obs_fader_create(OBS_FADER_CUBIC)),
|
|
|
|
obs_volmeter (obs_volmeter_create(OBS_FADER_LOG))
|
|
|
|
{
|
|
|
|
QHBoxLayout *volLayout = new QHBoxLayout();
|
|
|
|
QVBoxLayout *mainLayout = new QVBoxLayout();
|
|
|
|
QHBoxLayout *textLayout = new QHBoxLayout();
|
|
|
|
QHBoxLayout *botLayout = new QHBoxLayout();
|
|
|
|
|
|
|
|
nameLabel = new QLabel();
|
|
|
|
volLabel = new QLabel();
|
2018-02-19 19:54:37 +00:00
|
|
|
volMeter = new VolumeMeter(0, obs_volmeter);
|
2016-02-23 23:16:51 +00:00
|
|
|
mute = new MuteCheckBox();
|
|
|
|
slider = new QSlider(Qt::Horizontal);
|
|
|
|
|
|
|
|
QFont font = nameLabel->font();
|
|
|
|
font.setPointSize(font.pointSize()-1);
|
|
|
|
|
2017-04-19 19:54:15 +00:00
|
|
|
QString sourceName = obs_source_get_name(source);
|
|
|
|
|
|
|
|
nameLabel->setText(sourceName);
|
2016-02-23 23:16:51 +00:00
|
|
|
nameLabel->setFont(font);
|
|
|
|
volLabel->setFont(font);
|
|
|
|
slider->setMinimum(0);
|
|
|
|
slider->setMaximum(100);
|
|
|
|
|
|
|
|
// slider->setMaximumHeight(13);
|
|
|
|
|
|
|
|
textLayout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
textLayout->addWidget(nameLabel);
|
|
|
|
textLayout->addWidget(volLabel);
|
|
|
|
textLayout->setAlignment(nameLabel, Qt::AlignLeft);
|
|
|
|
textLayout->setAlignment(volLabel, Qt::AlignRight);
|
|
|
|
|
2017-04-19 19:54:15 +00:00
|
|
|
bool muted = obs_source_muted(source);
|
|
|
|
mute->setChecked(muted);
|
|
|
|
mute->setAccessibleName(
|
|
|
|
QTStr("VolControl.Mute").arg(sourceName));
|
2016-02-23 23:16:51 +00:00
|
|
|
|
|
|
|
volLayout->addWidget(slider);
|
|
|
|
volLayout->addWidget(mute);
|
|
|
|
volLayout->setSpacing(5);
|
|
|
|
|
|
|
|
botLayout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
botLayout->setSpacing(0);
|
|
|
|
botLayout->addLayout(volLayout);
|
|
|
|
|
|
|
|
if (showConfig) {
|
|
|
|
config = new QPushButton(this);
|
|
|
|
config->setProperty("themeID", "configIconSmall");
|
|
|
|
config->setFlat(true);
|
|
|
|
config->setSizePolicy(QSizePolicy::Maximum,
|
|
|
|
QSizePolicy::Maximum);
|
|
|
|
config->setMaximumSize(22, 22);
|
|
|
|
config->setAutoDefault(false);
|
|
|
|
|
2017-04-19 19:54:15 +00:00
|
|
|
config->setAccessibleName(QTStr("VolControl.Properties")
|
|
|
|
.arg(sourceName));
|
|
|
|
|
2016-02-23 23:16:51 +00:00
|
|
|
connect(config, &QAbstractButton::clicked,
|
|
|
|
this, &VolControl::EmitConfigClicked);
|
|
|
|
|
|
|
|
botLayout->addWidget(config);
|
|
|
|
}
|
|
|
|
|
|
|
|
mainLayout->setContentsMargins(4, 4, 4, 4);
|
|
|
|
mainLayout->setSpacing(2);
|
|
|
|
mainLayout->addItem(textLayout);
|
|
|
|
mainLayout->addWidget(volMeter);
|
|
|
|
mainLayout->addItem(botLayout);
|
|
|
|
|
|
|
|
setLayout(mainLayout);
|
|
|
|
|
|
|
|
obs_fader_add_callback(obs_fader, OBSVolumeChanged, this);
|
|
|
|
obs_volmeter_add_callback(obs_volmeter, OBSVolumeLevel, this);
|
|
|
|
|
|
|
|
signal_handler_connect(obs_source_get_signal_handler(source),
|
|
|
|
"mute", OBSVolumeMuted, this);
|
|
|
|
|
|
|
|
QWidget::connect(slider, SIGNAL(valueChanged(int)),
|
|
|
|
this, SLOT(SliderChanged(int)));
|
|
|
|
QWidget::connect(mute, SIGNAL(clicked(bool)),
|
|
|
|
this, SLOT(SetMuted(bool)));
|
|
|
|
|
|
|
|
obs_fader_attach_source(obs_fader, source);
|
|
|
|
obs_volmeter_attach_source(obs_volmeter, source);
|
|
|
|
|
|
|
|
slider->setStyle(new SliderAbsoluteSetStyle(slider->style()));
|
|
|
|
|
|
|
|
/* Call volume changed once to init the slider position and label */
|
|
|
|
VolumeChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
VolControl::~VolControl()
|
|
|
|
{
|
|
|
|
obs_fader_remove_callback(obs_fader, OBSVolumeChanged, this);
|
|
|
|
obs_volmeter_remove_callback(obs_volmeter, OBSVolumeLevel, this);
|
|
|
|
|
|
|
|
signal_handler_disconnect(obs_source_get_signal_handler(source),
|
|
|
|
"mute", OBSVolumeMuted, this);
|
|
|
|
|
|
|
|
obs_fader_destroy(obs_fader);
|
|
|
|
obs_volmeter_destroy(obs_volmeter);
|
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
QColor VolumeMeter::getBackgroundNominalColor() const
|
|
|
|
{
|
|
|
|
return backgroundNominalColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setBackgroundNominalColor(QColor c)
|
|
|
|
{
|
|
|
|
backgroundNominalColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getBackgroundWarningColor() const
|
|
|
|
{
|
|
|
|
return backgroundWarningColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setBackgroundWarningColor(QColor c)
|
|
|
|
{
|
|
|
|
backgroundWarningColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getBackgroundErrorColor() const
|
|
|
|
{
|
|
|
|
return backgroundErrorColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setBackgroundErrorColor(QColor c)
|
|
|
|
{
|
|
|
|
backgroundErrorColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getForegroundNominalColor() const
|
|
|
|
{
|
|
|
|
return foregroundNominalColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setForegroundNominalColor(QColor c)
|
|
|
|
{
|
|
|
|
foregroundNominalColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getForegroundWarningColor() const
|
|
|
|
{
|
|
|
|
return foregroundWarningColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setForegroundWarningColor(QColor c)
|
|
|
|
{
|
|
|
|
foregroundWarningColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getForegroundErrorColor() const
|
|
|
|
{
|
|
|
|
return foregroundErrorColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setForegroundErrorColor(QColor c)
|
|
|
|
{
|
|
|
|
foregroundErrorColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getClipColor() const
|
|
|
|
{
|
|
|
|
return clipColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setClipColor(QColor c)
|
|
|
|
{
|
|
|
|
clipColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getMagnitudeColor() const
|
|
|
|
{
|
|
|
|
return magnitudeColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setMagnitudeColor(QColor c)
|
|
|
|
{
|
|
|
|
magnitudeColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getMajorTickColor() const
|
|
|
|
{
|
|
|
|
return majorTickColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setMajorTickColor(QColor c)
|
|
|
|
{
|
|
|
|
majorTickColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getMinorTickColor() const
|
|
|
|
{
|
|
|
|
return minorTickColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setMinorTickColor(QColor c)
|
|
|
|
{
|
|
|
|
minorTickColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
qreal VolumeMeter::getMinimumLevel() const
|
|
|
|
{
|
|
|
|
return minimumLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setMinimumLevel(qreal v)
|
|
|
|
{
|
|
|
|
minimumLevel = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
qreal VolumeMeter::getWarningLevel() const
|
|
|
|
{
|
|
|
|
return warningLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setWarningLevel(qreal v)
|
|
|
|
{
|
|
|
|
warningLevel = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
qreal VolumeMeter::getErrorLevel() const
|
|
|
|
{
|
|
|
|
return errorLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setErrorLevel(qreal v)
|
|
|
|
{
|
|
|
|
errorLevel = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
qreal VolumeMeter::getClipLevel() const
|
|
|
|
{
|
|
|
|
return clipLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setClipLevel(qreal v)
|
|
|
|
{
|
|
|
|
clipLevel = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
qreal VolumeMeter::getMinimumInputLevel() const
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
return minimumInputLevel;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolumeMeter::setMinimumInputLevel(qreal v)
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
minimumInputLevel = v;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
qreal VolumeMeter::getPeakDecayRate() const
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
return peakDecayRate;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolumeMeter::setPeakDecayRate(qreal v)
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
peakDecayRate = v;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
qreal VolumeMeter::getMagnitudeIntegrationTime() const
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
return magnitudeIntegrationTime;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolumeMeter::setMagnitudeIntegrationTime(qreal v)
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
magnitudeIntegrationTime = v;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
qreal VolumeMeter::getPeakHoldDuration() const
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
return peakHoldDuration;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolumeMeter::setPeakHoldDuration(qreal v)
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
peakHoldDuration = v;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
qreal VolumeMeter::getInputPeakHoldDuration() const
|
|
|
|
{
|
|
|
|
return inputPeakHoldDuration;
|
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolumeMeter::setInputPeakHoldDuration(qreal v)
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2018-02-19 19:54:37 +00:00
|
|
|
inputPeakHoldDuration = v;
|
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
VolumeMeter::VolumeMeter(QWidget *parent, obs_volmeter_t *obs_volmeter)
|
|
|
|
: QWidget(parent), obs_volmeter(obs_volmeter)
|
|
|
|
{
|
|
|
|
// Use a font that can be rendered small.
|
|
|
|
tickFont = QFont("Arial");
|
|
|
|
tickFont.setPixelSize(7);
|
|
|
|
// Default meter color settings, they only show if
|
|
|
|
// there is no stylesheet, do not remove.
|
|
|
|
backgroundNominalColor.setRgb(0x26, 0x7f, 0x26); // Dark green
|
|
|
|
backgroundWarningColor.setRgb(0x7f, 0x7f, 0x26); // Dark yellow
|
|
|
|
backgroundErrorColor.setRgb(0x7f, 0x26, 0x26); // Dark red
|
|
|
|
foregroundNominalColor.setRgb(0x4c, 0xff, 0x4c); // Bright green
|
|
|
|
foregroundWarningColor.setRgb(0xff, 0xff, 0x4c); // Bright yellow
|
|
|
|
foregroundErrorColor.setRgb(0xff, 0x4c, 0x4c); // Bright red
|
|
|
|
clipColor.setRgb(0xff, 0xff, 0xff); // Bright white
|
|
|
|
magnitudeColor.setRgb(0x00, 0x00, 0x00); // Black
|
|
|
|
majorTickColor.setRgb(0xff, 0xff, 0xff); // Black
|
|
|
|
minorTickColor.setRgb(0xcc, 0xcc, 0xcc); // Black
|
|
|
|
minimumLevel = -60.0; // -60 dB
|
|
|
|
warningLevel = -20.0; // -20 dB
|
|
|
|
errorLevel = -9.0; // -9 dB
|
|
|
|
clipLevel = -0.5; // -0.5 dB
|
|
|
|
minimumInputLevel = -50.0; // -50 dB
|
|
|
|
peakDecayRate = 11.76; // 20 dB / 1.7 sec
|
|
|
|
magnitudeIntegrationTime = 0.3; // 99% in 300 ms
|
|
|
|
peakHoldDuration = 20.0; // 20 seconds
|
|
|
|
inputPeakHoldDuration = 1.0; // 1 second
|
|
|
|
|
|
|
|
handleChannelCofigurationChange();
|
2017-04-19 19:54:15 +00:00
|
|
|
updateTimerRef = updateTimer.toStrongRef();
|
|
|
|
if (!updateTimerRef) {
|
|
|
|
updateTimerRef = QSharedPointer<VolumeMeterTimer>::create();
|
|
|
|
updateTimerRef->start(34);
|
|
|
|
updateTimer = updateTimerRef;
|
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2017-04-19 19:54:15 +00:00
|
|
|
updateTimerRef->AddVolControl(this);
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 19:54:15 +00:00
|
|
|
VolumeMeter::~VolumeMeter()
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2017-04-19 19:54:15 +00:00
|
|
|
updateTimerRef->RemoveVolControl(this);
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolumeMeter::setLevels(
|
|
|
|
const float magnitude[MAX_AUDIO_CHANNELS],
|
|
|
|
const float peak[MAX_AUDIO_CHANNELS],
|
|
|
|
const float inputPeak[MAX_AUDIO_CHANNELS])
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2017-04-19 19:54:15 +00:00
|
|
|
uint64_t ts = os_gettime_ns();
|
|
|
|
QMutexLocker locker(&dataMutex);
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
currentLastUpdateTime = ts;
|
|
|
|
for (int channelNr = 0; channelNr < MAX_AUDIO_CHANNELS; channelNr++) {
|
|
|
|
currentMagnitude[channelNr] = magnitude[channelNr];
|
|
|
|
currentPeak[channelNr] = peak[channelNr];
|
|
|
|
currentInputPeak[channelNr] = inputPeak[channelNr];
|
|
|
|
}
|
|
|
|
|
|
|
|
// In case there are more updates then redraws we must make sure
|
|
|
|
// that the ballistics of peak and hold are recalculated.
|
|
|
|
locker.unlock();
|
|
|
|
calculateBallistics(ts);
|
2017-04-19 19:54:15 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
inline void VolumeMeter::resetLevels()
|
|
|
|
{
|
|
|
|
currentLastUpdateTime = 0;
|
|
|
|
for (int channelNr = 0; channelNr < MAX_AUDIO_CHANNELS; channelNr++) {
|
|
|
|
currentMagnitude[channelNr] = -M_INFINITE;
|
|
|
|
currentPeak[channelNr] = -M_INFINITE;
|
|
|
|
currentInputPeak[channelNr] = -M_INFINITE;
|
|
|
|
|
|
|
|
displayMagnitude[channelNr] = -M_INFINITE;
|
|
|
|
displayPeak[channelNr] = -M_INFINITE;
|
|
|
|
displayPeakHold[channelNr] = -M_INFINITE;
|
|
|
|
displayPeakHoldLastUpdateTime[channelNr] = 0;
|
|
|
|
displayInputPeakHold[channelNr] = -M_INFINITE;
|
|
|
|
displayInputPeakHoldLastUpdateTime[channelNr] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void VolumeMeter::handleChannelCofigurationChange()
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&dataMutex);
|
|
|
|
|
|
|
|
int currentNrAudioChannels = obs_volmeter_get_nr_channels(obs_volmeter);
|
|
|
|
if (displayNrAudioChannels != currentNrAudioChannels) {
|
|
|
|
displayNrAudioChannels = currentNrAudioChannels;
|
|
|
|
|
|
|
|
// Make room for 3 pixels high meter, with one pixel between
|
|
|
|
// each. Then 9 pixels below it for ticks and numbers.
|
|
|
|
setMinimumSize(130, displayNrAudioChannels * 4 + 8);
|
|
|
|
|
|
|
|
resetLevels();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool VolumeMeter::detectIdle(uint64_t ts)
|
|
|
|
{
|
|
|
|
float timeSinceLastUpdate = (ts - currentLastUpdateTime) * 0.000000001;
|
|
|
|
if (timeSinceLastUpdate > 0.5) {
|
|
|
|
resetLevels();
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void VolumeMeter::calculateBallisticsForChannel(int channelNr,
|
|
|
|
uint64_t ts, qreal timeSinceLastRedraw)
|
|
|
|
{
|
|
|
|
if (currentPeak[channelNr] >= displayPeak[channelNr] ||
|
|
|
|
isnan(displayPeak[channelNr])) {
|
|
|
|
// Attack of peak is immediate.
|
|
|
|
displayPeak[channelNr] = currentPeak[channelNr];
|
|
|
|
} else {
|
|
|
|
// Decay of peak is 40 dB / 1.7 seconds for Fast Profile
|
|
|
|
// 20 dB / 1.7 seconds for Medium Profile (Type I PPM)
|
|
|
|
// 24 dB / 2.8 seconds for Slow Profile (Type II PPM)
|
|
|
|
qreal decay = peakDecayRate * timeSinceLastRedraw;
|
|
|
|
displayPeak[channelNr] = CLAMP(displayPeak[channelNr] - decay,
|
|
|
|
currentPeak[channelNr], 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentPeak[channelNr] >= displayPeakHold[channelNr] ||
|
|
|
|
!isfinite(displayPeakHold[channelNr])) {
|
|
|
|
// Attack of peak-hold is immediate, but keep track
|
|
|
|
// when it was last updated.
|
|
|
|
displayPeakHold[channelNr] = currentPeak[channelNr];
|
|
|
|
displayPeakHoldLastUpdateTime[channelNr] = ts;
|
|
|
|
} else {
|
|
|
|
// The peak and hold falls back to peak
|
|
|
|
// after 20 seconds.
|
|
|
|
qreal timeSinceLastPeak = (uint64_t)(ts -
|
|
|
|
displayPeakHoldLastUpdateTime[channelNr]) * 0.000000001;
|
|
|
|
if (timeSinceLastPeak > peakHoldDuration) {
|
|
|
|
displayPeakHold[channelNr] = currentPeak[channelNr];
|
|
|
|
displayPeakHoldLastUpdateTime[channelNr] = ts;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentInputPeak[channelNr] >= displayInputPeakHold[channelNr] ||
|
|
|
|
!isfinite(displayInputPeakHold[channelNr])) {
|
|
|
|
// Attack of peak-hold is immediate, but keep track
|
|
|
|
// when it was last updated.
|
|
|
|
displayInputPeakHold[channelNr] = currentInputPeak[channelNr];
|
|
|
|
displayInputPeakHoldLastUpdateTime[channelNr] = ts;
|
|
|
|
} else {
|
|
|
|
// The peak and hold falls back to peak after 1 second.
|
|
|
|
qreal timeSinceLastPeak = (uint64_t)(ts -
|
|
|
|
displayInputPeakHoldLastUpdateTime[channelNr]) *
|
|
|
|
0.000000001;
|
|
|
|
if (timeSinceLastPeak > inputPeakHoldDuration) {
|
|
|
|
displayInputPeakHold[channelNr] =
|
|
|
|
currentInputPeak[channelNr];
|
|
|
|
displayInputPeakHoldLastUpdateTime[channelNr] =
|
|
|
|
ts;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isfinite(displayMagnitude[channelNr])) {
|
|
|
|
// The statements in the else-leg do not work with
|
|
|
|
// NaN and infinite displayMagnitude.
|
|
|
|
displayMagnitude[channelNr] =
|
|
|
|
currentMagnitude[channelNr];
|
|
|
|
} else {
|
|
|
|
// A VU meter will integrate to the new value to 99% in 300 ms.
|
|
|
|
// The calculation here is very simplified and is more accurate
|
|
|
|
// with higher frame-rate.
|
|
|
|
qreal attack = (currentMagnitude[channelNr] -
|
|
|
|
displayMagnitude[channelNr]) *
|
|
|
|
(timeSinceLastRedraw /
|
|
|
|
magnitudeIntegrationTime) * 0.99;
|
|
|
|
displayMagnitude[channelNr] = CLAMP(
|
|
|
|
displayMagnitude[channelNr] + attack,
|
|
|
|
minimumLevel, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void VolumeMeter::calculateBallistics(uint64_t ts,
|
|
|
|
qreal timeSinceLastRedraw)
|
2017-04-19 19:54:15 +00:00
|
|
|
{
|
|
|
|
QMutexLocker locker(&dataMutex);
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
for (int channelNr = 0; channelNr < MAX_AUDIO_CHANNELS; channelNr++) {
|
|
|
|
calculateBallisticsForChannel(channelNr, ts,
|
|
|
|
timeSinceLastRedraw);
|
2017-04-19 19:54:15 +00:00
|
|
|
}
|
2018-02-19 19:54:37 +00:00
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
void VolumeMeter::paintInputMeter(QPainter &painter, int x, int y,
|
|
|
|
int width, int height, float peakHold)
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&dataMutex);
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
if (peakHold < minimumInputLevel) {
|
|
|
|
painter.fillRect(x, y, width, height, backgroundNominalColor);
|
|
|
|
} else if (peakHold < warningLevel) {
|
|
|
|
painter.fillRect(x, y, width, height, foregroundNominalColor);
|
|
|
|
} else if (peakHold < errorLevel) {
|
|
|
|
painter.fillRect(x, y, width, height, foregroundWarningColor);
|
|
|
|
} else if (peakHold <= clipLevel) {
|
|
|
|
painter.fillRect(x, y, width, height, foregroundErrorColor);
|
|
|
|
} else {
|
|
|
|
painter.fillRect(x, y, width, height, clipColor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::paintTicks(QPainter &painter, int x, int y,
|
|
|
|
int width, int height)
|
|
|
|
{
|
|
|
|
qreal scale = width / minimumLevel;
|
|
|
|
|
|
|
|
painter.setFont(tickFont);
|
|
|
|
painter.setPen(majorTickColor);
|
|
|
|
|
|
|
|
// Draw major tick lines and numeric indicators.
|
|
|
|
for (int i = 0; i >= minimumLevel; i-= 5) {
|
|
|
|
int position = x + width - (i * scale) - 1;
|
|
|
|
QString str = QString::number(i);
|
|
|
|
|
|
|
|
if (i == 0 || i == -5) {
|
|
|
|
painter.drawText(position - 3, height, str);
|
|
|
|
} else {
|
|
|
|
painter.drawText(position - 5, height, str);
|
|
|
|
}
|
|
|
|
painter.drawLine(position, y, position, y + 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw minor tick lines.
|
|
|
|
painter.setPen(minorTickColor);
|
|
|
|
for (int i = 0; i >= minimumLevel; i--) {
|
|
|
|
int position = x + width - (i * scale) - 1;
|
|
|
|
|
|
|
|
if (i % 5 != 0) {
|
|
|
|
painter.drawLine(position, y, position, y + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::paintMeter(QPainter &painter, int x, int y,
|
|
|
|
int width, int height, float magnitude, float peak, float peakHold)
|
|
|
|
{
|
|
|
|
qreal scale = width / minimumLevel;
|
|
|
|
|
|
|
|
QMutexLocker locker(&dataMutex);
|
|
|
|
int minimumPosition = x + 0;
|
|
|
|
int maximumPosition = x + width;
|
|
|
|
int magnitudePosition = x + width - (magnitude * scale);
|
|
|
|
int peakPosition = x + width - (peak * scale);
|
|
|
|
int peakHoldPosition = x + width - (peakHold * scale);
|
|
|
|
int warningPosition = x + width - (warningLevel * scale);
|
|
|
|
int errorPosition = x + width - (errorLevel * scale);
|
|
|
|
|
|
|
|
int nominalLength = warningPosition - minimumPosition;
|
|
|
|
int warningLength = errorPosition - warningPosition;
|
|
|
|
int errorLength = maximumPosition - errorPosition;
|
|
|
|
locker.unlock();
|
|
|
|
|
|
|
|
if (peakPosition < minimumPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
minimumPosition, y,
|
|
|
|
nominalLength, height,
|
|
|
|
backgroundNominalColor);
|
|
|
|
painter.fillRect(
|
|
|
|
warningPosition, y,
|
|
|
|
warningLength, height,
|
|
|
|
backgroundWarningColor);
|
|
|
|
painter.fillRect(
|
|
|
|
errorPosition, y,
|
|
|
|
errorLength, height,
|
|
|
|
backgroundErrorColor);
|
|
|
|
|
|
|
|
} else if (peakPosition < warningPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
minimumPosition, y,
|
|
|
|
peakPosition - minimumPosition, height,
|
|
|
|
foregroundNominalColor);
|
|
|
|
painter.fillRect(
|
|
|
|
peakPosition, y,
|
|
|
|
warningPosition - peakPosition, height,
|
|
|
|
backgroundNominalColor);
|
|
|
|
painter.fillRect(
|
|
|
|
warningPosition, y,
|
|
|
|
warningLength, height,
|
|
|
|
backgroundWarningColor);
|
|
|
|
painter.fillRect(errorPosition, y,
|
|
|
|
errorLength, height,
|
|
|
|
backgroundErrorColor);
|
|
|
|
|
|
|
|
} else if (peakPosition < errorPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
minimumPosition, y,
|
|
|
|
nominalLength, height,
|
|
|
|
foregroundNominalColor);
|
|
|
|
painter.fillRect(
|
|
|
|
warningPosition, y,
|
|
|
|
peakPosition - warningPosition, height,
|
|
|
|
foregroundWarningColor);
|
|
|
|
painter.fillRect(
|
|
|
|
peakPosition, y,
|
|
|
|
errorPosition - peakPosition, height,
|
|
|
|
backgroundWarningColor);
|
|
|
|
painter.fillRect(
|
|
|
|
errorPosition, y,
|
|
|
|
errorLength, height,
|
|
|
|
backgroundErrorColor);
|
|
|
|
|
|
|
|
} else if (peakPosition < maximumPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
minimumPosition, y,
|
|
|
|
nominalLength, height,
|
|
|
|
foregroundNominalColor);
|
|
|
|
painter.fillRect(
|
|
|
|
warningPosition, y,
|
|
|
|
warningLength, height,
|
|
|
|
foregroundWarningColor);
|
|
|
|
painter.fillRect(
|
|
|
|
errorPosition, y,
|
|
|
|
peakPosition - errorPosition, height,
|
|
|
|
foregroundErrorColor);
|
|
|
|
painter.fillRect(
|
|
|
|
peakPosition, y,
|
|
|
|
maximumPosition - peakPosition, height,
|
|
|
|
backgroundErrorColor);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
qreal end = errorLength + warningLength + nominalLength;
|
|
|
|
painter.fillRect(
|
|
|
|
minimumPosition, y,
|
|
|
|
end, height,
|
|
|
|
QBrush(foregroundErrorColor));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (peakHoldPosition - 3 < minimumPosition) {
|
|
|
|
// Peak-hold below minimum, no drawing.
|
|
|
|
|
|
|
|
} else if (peakHoldPosition < warningPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
peakHoldPosition - 3, y,
|
|
|
|
3, height,
|
|
|
|
foregroundNominalColor);
|
|
|
|
|
|
|
|
} else if (peakHoldPosition < errorPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
peakHoldPosition - 3, y,
|
|
|
|
3, height,
|
|
|
|
foregroundWarningColor);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
painter.fillRect(
|
|
|
|
peakHoldPosition - 3, y,
|
|
|
|
3, height,
|
|
|
|
foregroundErrorColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (magnitudePosition - 3 < minimumPosition) {
|
|
|
|
// Magnitude below minimum, no drawing.
|
|
|
|
|
|
|
|
} else if (magnitudePosition < warningPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
magnitudePosition - 3, y,
|
|
|
|
3, height,
|
|
|
|
magnitudeColor);
|
|
|
|
|
|
|
|
} else if (magnitudePosition < errorPosition) {
|
|
|
|
painter.fillRect(
|
|
|
|
magnitudePosition - 3, y,
|
|
|
|
3, height,
|
|
|
|
magnitudeColor);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
painter.fillRect(
|
|
|
|
magnitudePosition - 3, y,
|
|
|
|
3, height,
|
|
|
|
magnitudeColor);
|
2017-04-19 19:54:15 +00:00
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::paintEvent(QPaintEvent *event)
|
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(event);
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
uint64_t ts = os_gettime_ns();
|
|
|
|
qreal timeSinceLastRedraw = (ts - lastRedrawTime) * 0.000000001;
|
2016-02-23 23:16:51 +00:00
|
|
|
|
|
|
|
int width = size().width();
|
|
|
|
int height = size().height();
|
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
handleChannelCofigurationChange();
|
|
|
|
calculateBallistics(ts, timeSinceLastRedraw);
|
|
|
|
bool idle = detectIdle(ts);
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
// Draw the ticks in a off-screen buffer when the widget changes size.
|
|
|
|
QSize tickPaintCacheSize = QSize(width, 9);
|
|
|
|
if (tickPaintCache == NULL ||
|
|
|
|
tickPaintCache->size() != tickPaintCacheSize) {
|
|
|
|
delete tickPaintCache;
|
|
|
|
tickPaintCache = new QPixmap(tickPaintCacheSize);
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
QColor clearColor(0, 0, 0, 0);
|
|
|
|
tickPaintCache->fill(clearColor);
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
QPainter tickPainter(tickPaintCache);
|
|
|
|
paintTicks(tickPainter, 6, 0, tickPaintCacheSize.width() - 6,
|
|
|
|
tickPaintCacheSize.height());
|
|
|
|
tickPainter.end();
|
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
// Actual painting of the widget starts here.
|
|
|
|
QPainter painter(this);
|
|
|
|
painter.drawPixmap(0, height - 9, *tickPaintCache);
|
|
|
|
|
|
|
|
for (int channelNr = 0; channelNr < displayNrAudioChannels;
|
|
|
|
channelNr++) {
|
|
|
|
paintMeter(painter,
|
|
|
|
5, channelNr * 4, width - 5, 3,
|
|
|
|
displayMagnitude[channelNr], displayPeak[channelNr],
|
|
|
|
displayPeakHold[channelNr]);
|
|
|
|
|
|
|
|
if (!idle) {
|
|
|
|
// By not drawing the input meter boxes the user can
|
|
|
|
// see that the audio stream has been stopped, without
|
|
|
|
// having too much visual impact.
|
|
|
|
paintInputMeter(painter,
|
|
|
|
0, channelNr * 4, 3, 3,
|
|
|
|
displayInputPeakHold[channelNr]);
|
|
|
|
}
|
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2018-02-19 19:54:37 +00:00
|
|
|
lastRedrawTime = ts;
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
2017-04-19 19:54:15 +00:00
|
|
|
|
|
|
|
void VolumeMeterTimer::AddVolControl(VolumeMeter *meter)
|
|
|
|
{
|
|
|
|
volumeMeters.push_back(meter);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeterTimer::RemoveVolControl(VolumeMeter *meter)
|
|
|
|
{
|
|
|
|
volumeMeters.removeOne(meter);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeterTimer::timerEvent(QTimerEvent*)
|
|
|
|
{
|
|
|
|
for (VolumeMeter *meter : volumeMeters)
|
|
|
|
meter->update();
|
|
|
|
}
|