volume-control.cpp 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. #include "volume-control.hpp"
  2. #include "qt-wrappers.hpp"
  3. #include <util/platform.h>
  4. #include <QHBoxLayout>
  5. #include <QVBoxLayout>
  6. #include <QSlider>
  7. #include <QLabel>
  8. #include <QPainter>
  9. #include <QTimer>
  10. #include <string>
  11. #include <math.h>
  12. using namespace std;
  13. #define VOL_MIN -96.0f
  14. #define VOL_MAX 0.0f
  15. /*
  16. VOL_MIN_LOG = DBToLog(VOL_MIN)
  17. VOL_MAX_LOG = DBToLog(VOL_MAX)
  18. ... just in case someone wants to use a smaller scale
  19. */
  20. #define VOL_MIN_LOG -2.0086001717619175
  21. #define VOL_MAX_LOG -0.77815125038364363
  22. #define UPDATE_INTERVAL_MS 50
  23. static inline float DBToLog(float db)
  24. {
  25. return -log10f(0.0f - (db - 6.0f));
  26. }
  27. static inline float DBToLinear(float db_full)
  28. {
  29. float db = fmaxf(fminf(db_full, VOL_MAX), VOL_MIN);
  30. return (DBToLog(db) - VOL_MIN_LOG) / (VOL_MAX_LOG - VOL_MIN_LOG);
  31. }
  32. void VolControl::OBSVolumeChanged(void *data, calldata_t *calldata)
  33. {
  34. VolControl *volControl = static_cast<VolControl*>(data);
  35. int vol = (int)(calldata_float(calldata, "volume") * 100.0f + 0.5f);
  36. QMetaObject::invokeMethod(volControl, "VolumeChanged", Q_ARG(int, vol));
  37. }
  38. void VolControl::OBSVolumeLevel(void *data, calldata_t *calldata)
  39. {
  40. VolControl *volControl = static_cast<VolControl*>(data);
  41. float peak = calldata_float(calldata, "level");
  42. float mag = calldata_float(calldata, "magnitude");
  43. float peakHold = calldata_float(calldata, "peak");
  44. QMetaObject::invokeMethod(volControl, "VolumeLevel",
  45. Q_ARG(float, mag),
  46. Q_ARG(float, peak),
  47. Q_ARG(float, peakHold));
  48. }
  49. void VolControl::VolumeChanged(int vol)
  50. {
  51. signalChanged = false;
  52. slider->setValue(vol);
  53. signalChanged = true;
  54. }
  55. void VolControl::VolumeLevel(float mag, float peak, float peakHold)
  56. {
  57. uint64_t curMeterTime = os_gettime_ns() / 1000000;
  58. /*
  59. Add again peak averaging?
  60. */
  61. /* only update after a certain amount of time */
  62. if ((curMeterTime - lastMeterTime) > UPDATE_INTERVAL_MS) {
  63. float vol = (float)slider->value() * 0.01f;
  64. lastMeterTime = curMeterTime;
  65. volMeter->setLevels(DBToLinear(mag) * vol,
  66. DBToLinear(peak) * vol,
  67. DBToLinear(peakHold) * vol);
  68. }
  69. }
  70. void VolControl::SliderChanged(int vol)
  71. {
  72. if (signalChanged) {
  73. signal_handler_disconnect(obs_source_get_signal_handler(source),
  74. "volume", OBSVolumeChanged, this);
  75. obs_source_set_volume(source, float(vol)*0.01f);
  76. signal_handler_connect(obs_source_get_signal_handler(source),
  77. "volume", OBSVolumeChanged, this);
  78. }
  79. volLabel->setText(QString::number(vol));
  80. }
  81. QString VolControl::GetName() const
  82. {
  83. return nameLabel->text();
  84. }
  85. void VolControl::SetName(const QString &newName)
  86. {
  87. nameLabel->setText(newName);
  88. }
  89. VolControl::VolControl(OBSSource source_)
  90. : source (source_),
  91. signalChanged (true),
  92. lastMeterTime (0),
  93. levelTotal (0.0f),
  94. levelCount (0.0f)
  95. {
  96. QVBoxLayout *mainLayout = new QVBoxLayout();
  97. QHBoxLayout *textLayout = new QHBoxLayout();
  98. int vol = int(obs_source_get_volume(source) * 100.0f);
  99. nameLabel = new QLabel();
  100. volLabel = new QLabel();
  101. volMeter = new VolumeMeter();
  102. slider = new QSlider(Qt::Horizontal);
  103. QFont font = nameLabel->font();
  104. font.setPointSize(font.pointSize()-1);
  105. nameLabel->setText(obs_source_get_name(source));
  106. nameLabel->setFont(font);
  107. volLabel->setText(QString::number(vol));
  108. volLabel->setFont(font);
  109. slider->setMinimum(0);
  110. slider->setMaximum(100);
  111. slider->setValue(vol);
  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. mainLayout->setContentsMargins(4, 4, 4, 4);
  119. mainLayout->setSpacing(2);
  120. mainLayout->addItem(textLayout);
  121. mainLayout->addWidget(volMeter);
  122. mainLayout->addWidget(slider);
  123. setLayout(mainLayout);
  124. signal_handler_connect(obs_source_get_signal_handler(source),
  125. "volume", OBSVolumeChanged, this);
  126. signal_handler_connect(obs_source_get_signal_handler(source),
  127. "volume_level", OBSVolumeLevel, this);
  128. QWidget::connect(slider, SIGNAL(valueChanged(int)),
  129. this, SLOT(SliderChanged(int)));
  130. }
  131. VolControl::~VolControl()
  132. {
  133. signal_handler_disconnect(obs_source_get_signal_handler(source),
  134. "volume", OBSVolumeChanged, this);
  135. signal_handler_disconnect(obs_source_get_signal_handler(source),
  136. "volume_level", OBSVolumeLevel, this);
  137. }
  138. VolumeMeter::VolumeMeter(QWidget *parent)
  139. : QWidget(parent)
  140. {
  141. setMinimumSize(1, 3);
  142. bkColor.setRgb(0xDD, 0xDD, 0xDD);
  143. magColor.setRgb(0x20, 0x7D, 0x17);
  144. peakColor.setRgb(0x3E, 0xF1, 0x2B);
  145. peakHoldColor.setRgb(0x00, 0x00, 0x00);
  146. resetTimer = new QTimer(this);
  147. connect(resetTimer, SIGNAL(timeout()), this, SLOT(resetState()));
  148. resetState();
  149. }
  150. void VolumeMeter::resetState(void)
  151. {
  152. setLevels(0.0f, 0.0f, 0.0f);
  153. if (resetTimer->isActive())
  154. resetTimer->stop();
  155. }
  156. void VolumeMeter::setLevels(float nmag, float npeak, float npeakHold)
  157. {
  158. mag = nmag;
  159. peak = npeak;
  160. peakHold = npeakHold;
  161. update();
  162. if (resetTimer->isActive())
  163. resetTimer->stop();
  164. resetTimer->start(1000);
  165. }
  166. void VolumeMeter::paintEvent(QPaintEvent *event)
  167. {
  168. UNUSED_PARAMETER(event);
  169. QPainter painter(this);
  170. QLinearGradient gradient;
  171. int width = size().width();
  172. int height = size().height();
  173. int scaledMag = int((float)width * mag);
  174. int scaledPeak = int((float)width * peak);
  175. int scaledPeakHold = int((float)width * peakHold);
  176. gradient.setStart(qreal(scaledMag), 0);
  177. gradient.setFinalStop(qreal(scaledPeak), 0);
  178. gradient.setColorAt(0, magColor);
  179. gradient.setColorAt(1, peakColor);
  180. // RMS
  181. painter.fillRect(0, 0,
  182. scaledMag, height,
  183. magColor);
  184. // RMS - Peak gradient
  185. painter.fillRect(scaledMag, 0,
  186. scaledPeak - scaledMag + 1, height,
  187. QBrush(gradient));
  188. // Background
  189. painter.fillRect(scaledPeak, 0,
  190. width - scaledPeak, height,
  191. bkColor);
  192. // Peak hold
  193. if (peakHold == 1.0f)
  194. scaledPeakHold--;
  195. painter.setPen(peakHoldColor);
  196. painter.drawLine(scaledPeakHold, 0,
  197. scaledPeakHold, height);
  198. }