2016-02-23 23:16:51 +00:00
|
|
|
#include "volume-control.hpp"
|
|
|
|
#include "qt-wrappers.hpp"
|
2017-04-15 19:02:06 +00:00
|
|
|
#include "obs-app.hpp"
|
2016-02-23 23:16:51 +00:00
|
|
|
#include "mute-checkbox.hpp"
|
|
|
|
#include "slider-absoluteset-style.hpp"
|
|
|
|
#include <util/platform.h>
|
2017-04-15 19:02:06 +00:00
|
|
|
#include <util/threading.h>
|
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;
|
|
|
|
|
2017-04-15 19:02:06 +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");
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::OBSVolumeLevel(void *data, float level, float mag,
|
|
|
|
float peak, float muted)
|
|
|
|
{
|
|
|
|
VolControl *volControl = static_cast<VolControl*>(data);
|
|
|
|
|
2017-04-15 19:02:06 +00:00
|
|
|
if (muted)
|
|
|
|
level = mag = peak = 0.0f;
|
|
|
|
|
|
|
|
volControl->volMeter->setLevels(mag, level, peak);
|
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);
|
|
|
|
|
|
|
|
updateText();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolControl::VolumeLevel(float mag, float peak, float peakHold, bool muted)
|
|
|
|
{
|
|
|
|
if (muted) {
|
|
|
|
mag = 0.0f;
|
|
|
|
peak = 0.0f;
|
|
|
|
peakHold = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
volMeter->setLevels(mag, peak, peakHold);
|
|
|
|
}
|
|
|
|
|
|
|
|
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-15 19:02:06 +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();
|
|
|
|
}
|
|
|
|
|
|
|
|
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();
|
|
|
|
volMeter = new VolumeMeter();
|
|
|
|
mute = new MuteCheckBox();
|
|
|
|
slider = new QSlider(Qt::Horizontal);
|
|
|
|
|
|
|
|
QFont font = nameLabel->font();
|
|
|
|
font.setPointSize(font.pointSize()-1);
|
|
|
|
|
2017-04-15 19:02:06 +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-15 19:02:06 +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-15 19:02:06 +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);
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getBkColor() const
|
|
|
|
{
|
|
|
|
return bkColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setBkColor(QColor c)
|
|
|
|
{
|
|
|
|
bkColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getMagColor() const
|
|
|
|
{
|
|
|
|
return magColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setMagColor(QColor c)
|
|
|
|
{
|
|
|
|
magColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getPeakColor() const
|
|
|
|
{
|
|
|
|
return peakColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setPeakColor(QColor c)
|
|
|
|
{
|
|
|
|
peakColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor VolumeMeter::getPeakHoldColor() const
|
|
|
|
{
|
|
|
|
return peakHoldColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setPeakHoldColor(QColor c)
|
|
|
|
{
|
|
|
|
peakHoldColor = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VolumeMeter::VolumeMeter(QWidget *parent)
|
|
|
|
: QWidget(parent)
|
|
|
|
{
|
|
|
|
setMinimumSize(1, 3);
|
|
|
|
|
|
|
|
//Default meter color settings, they only show if there is no stylesheet, do not remove.
|
|
|
|
bkColor.setRgb(0xDD, 0xDD, 0xDD);
|
|
|
|
magColor.setRgb(0x20, 0x7D, 0x17);
|
|
|
|
peakColor.setRgb(0x3E, 0xF1, 0x2B);
|
|
|
|
peakHoldColor.setRgb(0x00, 0x00, 0x00);
|
|
|
|
|
2017-04-15 19:02:06 +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-15 19:02:06 +00:00
|
|
|
updateTimerRef->AddVolControl(this);
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
2017-04-15 19:02:06 +00:00
|
|
|
VolumeMeter::~VolumeMeter()
|
2016-02-23 23:16:51 +00:00
|
|
|
{
|
2017-04-15 19:02:06 +00:00
|
|
|
updateTimerRef->RemoveVolControl(this);
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::setLevels(float nmag, float npeak, float npeakHold)
|
|
|
|
{
|
2017-04-15 19:02:06 +00:00
|
|
|
uint64_t ts = os_gettime_ns();
|
|
|
|
QMutexLocker locker(&dataMutex);
|
|
|
|
|
|
|
|
mag += nmag;
|
|
|
|
peak += npeak;
|
|
|
|
peakHold += npeakHold;
|
|
|
|
multiple += 1.0f;
|
|
|
|
lastUpdateTime = ts;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void VolumeMeter::calcLevels()
|
|
|
|
{
|
|
|
|
uint64_t ts = os_gettime_ns();
|
|
|
|
QMutexLocker locker(&dataMutex);
|
|
|
|
|
|
|
|
if (lastUpdateTime && ts - lastUpdateTime > 1000000000) {
|
|
|
|
mag = peak = peakHold = 0.0f;
|
|
|
|
multiple = 1.0f;
|
|
|
|
lastUpdateTime = 0;
|
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2017-04-15 19:02:06 +00:00
|
|
|
if (multiple > 0.0f) {
|
|
|
|
curMag = mag / multiple;
|
|
|
|
curPeak = peak / multiple;
|
|
|
|
curPeakHold = peakHold / multiple;
|
2016-02-23 23:16:51 +00:00
|
|
|
|
2017-04-15 19:02:06 +00:00
|
|
|
mag = peak = peakHold = multiple = 0.0f;
|
|
|
|
}
|
2016-02-23 23:16:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VolumeMeter::paintEvent(QPaintEvent *event)
|
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(event);
|
|
|
|
|
|
|
|
QPainter painter(this);
|
|
|
|
QLinearGradient gradient;
|
|
|
|
|
|
|
|
int width = size().width();
|
|
|
|
int height = size().height();
|
|
|
|
|
2017-04-15 19:02:06 +00:00
|
|
|
calcLevels();
|
|
|
|
|
|
|
|
int scaledMag = int((float)width * curMag);
|
|
|
|
int scaledPeak = int((float)width * curPeak);
|
|
|
|
int scaledPeakHold = int((float)width * curPeakHold);
|
2016-02-23 23:16:51 +00:00
|
|
|
|
|
|
|
gradient.setStart(qreal(scaledMag), 0);
|
|
|
|
gradient.setFinalStop(qreal(scaledPeak), 0);
|
|
|
|
gradient.setColorAt(0, magColor);
|
|
|
|
gradient.setColorAt(1, peakColor);
|
|
|
|
|
|
|
|
// RMS
|
|
|
|
painter.fillRect(0, 0,
|
|
|
|
scaledMag, height,
|
|
|
|
magColor);
|
|
|
|
|
|
|
|
// RMS - Peak gradient
|
|
|
|
painter.fillRect(scaledMag, 0,
|
|
|
|
scaledPeak - scaledMag + 1, height,
|
|
|
|
QBrush(gradient));
|
|
|
|
|
|
|
|
// Background
|
|
|
|
painter.fillRect(scaledPeak, 0,
|
|
|
|
width - scaledPeak, height,
|
|
|
|
bkColor);
|
|
|
|
|
|
|
|
// Peak hold
|
|
|
|
if (peakHold == 1.0f)
|
|
|
|
scaledPeakHold--;
|
|
|
|
|
|
|
|
painter.setPen(peakHoldColor);
|
|
|
|
painter.drawLine(scaledPeakHold, 0,
|
|
|
|
scaledPeakHold, height);
|
|
|
|
|
|
|
|
}
|
2017-04-15 19:02:06 +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();
|
|
|
|
}
|