volume-control.cpp 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. #include "volume-control.hpp"
  2. #include "qt-wrappers.hpp"
  3. #include "obs-app.hpp"
  4. #include "mute-checkbox.hpp"
  5. #include "slider-absoluteset-style.hpp"
  6. #include <obs-audio-controls.h>
  7. #include <util/platform.h>
  8. #include <util/threading.h>
  9. #include <QHBoxLayout>
  10. #include <QVBoxLayout>
  11. #include <QPushButton>
  12. #include <QVariant>
  13. #include <QSlider>
  14. #include <QLabel>
  15. #include <QPainter>
  16. #include <QTimer>
  17. #include <string>
  18. #include <math.h>
  19. using namespace std;
  20. QWeakPointer<VolumeMeterTimer> VolumeMeter::updateTimer;
  21. void VolControl::OBSVolumeChanged(void *data, float db)
  22. {
  23. Q_UNUSED(db);
  24. VolControl *volControl = static_cast<VolControl*>(data);
  25. QMetaObject::invokeMethod(volControl, "VolumeChanged");
  26. }
  27. void VolControl::OBSVolumeLevel(void *data, float level, float mag,
  28. float peak, float muted)
  29. {
  30. VolControl *volControl = static_cast<VolControl*>(data);
  31. if (muted)
  32. level = mag = peak = 0.0f;
  33. volControl->volMeter->setLevels(mag, level, peak);
  34. }
  35. void VolControl::OBSVolumeMuted(void *data, calldata_t *calldata)
  36. {
  37. VolControl *volControl = static_cast<VolControl*>(data);
  38. bool muted = calldata_bool(calldata, "muted");
  39. QMetaObject::invokeMethod(volControl, "VolumeMuted",
  40. Q_ARG(bool, muted));
  41. }
  42. void VolControl::VolumeChanged()
  43. {
  44. slider->blockSignals(true);
  45. slider->setValue((int) (obs_fader_get_deflection(obs_fader) * 100.0f));
  46. slider->blockSignals(false);
  47. updateText();
  48. }
  49. void VolControl::VolumeMuted(bool muted)
  50. {
  51. if (mute->isChecked() != muted)
  52. mute->setChecked(muted);
  53. }
  54. void VolControl::SetMuted(bool checked)
  55. {
  56. obs_source_set_muted(source, checked);
  57. }
  58. void VolControl::SliderChanged(int vol)
  59. {
  60. obs_fader_set_deflection(obs_fader, float(vol) * 0.01f);
  61. updateText();
  62. }
  63. void VolControl::updateText()
  64. {
  65. QString db = QString::number(obs_fader_get_db(obs_fader), 'f', 1)
  66. .append(" dB");
  67. volLabel->setText(db);
  68. bool muted = obs_source_muted(source);
  69. const char *accTextLookup = muted
  70. ? "VolControl.SliderMuted"
  71. : "VolControl.SliderUnmuted";
  72. QString sourceName = obs_source_get_name(source);
  73. QString accText = QTStr(accTextLookup).arg(sourceName, db);
  74. slider->setAccessibleName(accText);
  75. }
  76. QString VolControl::GetName() const
  77. {
  78. return nameLabel->text();
  79. }
  80. void VolControl::SetName(const QString &newName)
  81. {
  82. nameLabel->setText(newName);
  83. }
  84. void VolControl::EmitConfigClicked()
  85. {
  86. emit ConfigClicked();
  87. }
  88. VolControl::VolControl(OBSSource source_, bool showConfig)
  89. : source (source_),
  90. levelTotal (0.0f),
  91. levelCount (0.0f),
  92. obs_fader (obs_fader_create(OBS_FADER_CUBIC)),
  93. obs_volmeter (obs_volmeter_create(OBS_FADER_LOG))
  94. {
  95. QHBoxLayout *volLayout = new QHBoxLayout();
  96. QVBoxLayout *mainLayout = new QVBoxLayout();
  97. QHBoxLayout *textLayout = new QHBoxLayout();
  98. QHBoxLayout *botLayout = new QHBoxLayout();
  99. nameLabel = new QLabel();
  100. volLabel = new QLabel();
  101. volMeter = new VolumeMeter();
  102. mute = new MuteCheckBox();
  103. slider = new QSlider(Qt::Horizontal);
  104. QFont font = nameLabel->font();
  105. font.setPointSize(font.pointSize()-1);
  106. QString sourceName = obs_source_get_name(source);
  107. nameLabel->setText(sourceName);
  108. nameLabel->setFont(font);
  109. volLabel->setFont(font);
  110. slider->setMinimum(0);
  111. slider->setMaximum(100);
  112. // slider->setMaximumHeight(13);
  113. textLayout->setContentsMargins(0, 0, 0, 0);
  114. textLayout->addWidget(nameLabel);
  115. textLayout->addWidget(volLabel);
  116. textLayout->setAlignment(nameLabel, Qt::AlignLeft);
  117. textLayout->setAlignment(volLabel, Qt::AlignRight);
  118. bool muted = obs_source_muted(source);
  119. mute->setChecked(muted);
  120. mute->setAccessibleName(
  121. QTStr("VolControl.Mute").arg(sourceName));
  122. volLayout->addWidget(slider);
  123. volLayout->addWidget(mute);
  124. volLayout->setSpacing(5);
  125. botLayout->setContentsMargins(0, 0, 0, 0);
  126. botLayout->setSpacing(0);
  127. botLayout->addLayout(volLayout);
  128. if (showConfig) {
  129. config = new QPushButton(this);
  130. config->setProperty("themeID", "configIconSmall");
  131. config->setFlat(true);
  132. config->setSizePolicy(QSizePolicy::Maximum,
  133. QSizePolicy::Maximum);
  134. config->setMaximumSize(22, 22);
  135. config->setAutoDefault(false);
  136. config->setAccessibleName(QTStr("VolControl.Properties")
  137. .arg(sourceName));
  138. connect(config, &QAbstractButton::clicked,
  139. this, &VolControl::EmitConfigClicked);
  140. botLayout->addWidget(config);
  141. }
  142. mainLayout->setContentsMargins(4, 4, 4, 4);
  143. mainLayout->setSpacing(2);
  144. mainLayout->addItem(textLayout);
  145. mainLayout->addWidget(volMeter);
  146. mainLayout->addItem(botLayout);
  147. setLayout(mainLayout);
  148. obs_fader_add_callback(obs_fader, OBSVolumeChanged, this);
  149. obs_volmeter_add_callback(obs_volmeter, OBSVolumeLevel, this);
  150. signal_handler_connect(obs_source_get_signal_handler(source),
  151. "mute", OBSVolumeMuted, this);
  152. QWidget::connect(slider, SIGNAL(valueChanged(int)),
  153. this, SLOT(SliderChanged(int)));
  154. QWidget::connect(mute, SIGNAL(clicked(bool)),
  155. this, SLOT(SetMuted(bool)));
  156. obs_fader_attach_source(obs_fader, source);
  157. obs_volmeter_attach_source(obs_volmeter, source);
  158. slider->setStyle(new SliderAbsoluteSetStyle(slider->style()));
  159. /* Call volume changed once to init the slider position and label */
  160. VolumeChanged();
  161. }
  162. VolControl::~VolControl()
  163. {
  164. obs_fader_remove_callback(obs_fader, OBSVolumeChanged, this);
  165. obs_volmeter_remove_callback(obs_volmeter, OBSVolumeLevel, this);
  166. signal_handler_disconnect(obs_source_get_signal_handler(source),
  167. "mute", OBSVolumeMuted, this);
  168. obs_fader_destroy(obs_fader);
  169. obs_volmeter_destroy(obs_volmeter);
  170. }
  171. QColor VolumeMeter::getBkColor() const
  172. {
  173. return bkColor;
  174. }
  175. void VolumeMeter::setBkColor(QColor c)
  176. {
  177. bkColor = c;
  178. }
  179. QColor VolumeMeter::getMagColor() const
  180. {
  181. return magColor;
  182. }
  183. void VolumeMeter::setMagColor(QColor c)
  184. {
  185. magColor = c;
  186. }
  187. QColor VolumeMeter::getPeakColor() const
  188. {
  189. return peakColor;
  190. }
  191. void VolumeMeter::setPeakColor(QColor c)
  192. {
  193. peakColor = c;
  194. }
  195. QColor VolumeMeter::getPeakHoldColor() const
  196. {
  197. return peakHoldColor;
  198. }
  199. void VolumeMeter::setPeakHoldColor(QColor c)
  200. {
  201. peakHoldColor = c;
  202. }
  203. VolumeMeter::VolumeMeter(QWidget *parent)
  204. : QWidget(parent)
  205. {
  206. setMinimumSize(1, 3);
  207. //Default meter color settings, they only show if there is no stylesheet, do not remove.
  208. bkColor.setRgb(0xDD, 0xDD, 0xDD);
  209. magColor.setRgb(0x20, 0x7D, 0x17);
  210. peakColor.setRgb(0x3E, 0xF1, 0x2B);
  211. peakHoldColor.setRgb(0x00, 0x00, 0x00);
  212. clipColor1.setRgb(0x7F, 0x00, 0x00);
  213. clipColor2.setRgb(0xFF, 0x00, 0x00);
  214. updateTimerRef = updateTimer.toStrongRef();
  215. if (!updateTimerRef) {
  216. updateTimerRef = QSharedPointer<VolumeMeterTimer>::create();
  217. updateTimerRef->start(34);
  218. updateTimer = updateTimerRef;
  219. }
  220. updateTimerRef->AddVolControl(this);
  221. }
  222. VolumeMeter::~VolumeMeter()
  223. {
  224. updateTimerRef->RemoveVolControl(this);
  225. }
  226. void VolumeMeter::setLevels(float nmag, float npeak, float npeakHold)
  227. {
  228. uint64_t ts = os_gettime_ns();
  229. QMutexLocker locker(&dataMutex);
  230. mag += nmag;
  231. peak += npeak;
  232. peakHold += npeakHold;
  233. multiple += 1.0f;
  234. lastUpdateTime = ts;
  235. }
  236. inline void VolumeMeter::calcLevels()
  237. {
  238. uint64_t ts = os_gettime_ns();
  239. QMutexLocker locker(&dataMutex);
  240. if (lastUpdateTime && ts - lastUpdateTime > 1000000000) {
  241. mag = peak = peakHold = 0.0f;
  242. multiple = 1.0f;
  243. lastUpdateTime = 0;
  244. }
  245. if (multiple > 0.0f) {
  246. curMag = mag / multiple;
  247. curPeak = peak / multiple;
  248. curPeakHold = peakHold / multiple;
  249. mag = peak = peakHold = multiple = 0.0f;
  250. }
  251. }
  252. void VolumeMeter::paintEvent(QPaintEvent *event)
  253. {
  254. UNUSED_PARAMETER(event);
  255. QPainter painter(this);
  256. QLinearGradient gradient;
  257. int width = size().width();
  258. int height = size().height();
  259. calcLevels();
  260. int scaledMag = int((float)width * curMag);
  261. int scaledPeak = int((float)width * curPeak);
  262. int scaledPeakHold = int((float)width * curPeakHold);
  263. float db = obs_volmeter_get_cur_db(OBS_FADER_LOG, curPeakHold);
  264. gradient.setStart(qreal(scaledMag), 0);
  265. gradient.setFinalStop(qreal(scaledPeak), 0);
  266. gradient.setColorAt(0, db == 0.0f ? clipColor1 : magColor);
  267. gradient.setColorAt(1, db == 0.0f ? clipColor2 : peakColor);
  268. // RMS
  269. painter.fillRect(0, 0,
  270. scaledMag, height,
  271. db == 0.0f ? clipColor1 : magColor);
  272. // RMS - Peak gradient
  273. painter.fillRect(scaledMag, 0,
  274. scaledPeak - scaledMag + 1, height,
  275. QBrush(gradient));
  276. // Background
  277. painter.fillRect(scaledPeak, 0,
  278. width - scaledPeak, height,
  279. bkColor);
  280. // Peak hold
  281. if (peakHold == 1.0f)
  282. scaledPeakHold--;
  283. painter.setPen(peakHoldColor);
  284. painter.drawLine(scaledPeakHold, 0,
  285. scaledPeakHold, height);
  286. }
  287. void VolumeMeterTimer::AddVolControl(VolumeMeter *meter)
  288. {
  289. volumeMeters.push_back(meter);
  290. }
  291. void VolumeMeterTimer::RemoveVolControl(VolumeMeter *meter)
  292. {
  293. volumeMeters.removeOne(meter);
  294. }
  295. void VolumeMeterTimer::timerEvent(QTimerEvent*)
  296. {
  297. for (VolumeMeter *meter : volumeMeters)
  298. meter->update();
  299. }