adv-audio-control.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497
  1. #include <QHBoxLayout>
  2. #include <QGridLayout>
  3. #include <QLabel>
  4. #include <QSpinBox>
  5. #include <QComboBox>
  6. #include <QCheckBox>
  7. #include <cmath>
  8. #include "qt-wrappers.hpp"
  9. #include "obs-app.hpp"
  10. #include "adv-audio-control.hpp"
  11. #include "window-basic-main.hpp"
  12. #ifndef NSEC_PER_MSEC
  13. #define NSEC_PER_MSEC 1000000
  14. #endif
  15. #define MIN_DB -96.0
  16. #define MAX_DB 26.0
  17. OBSAdvAudioCtrl::OBSAdvAudioCtrl(QGridLayout *, obs_source_t *source_)
  18. : source(source_)
  19. {
  20. QHBoxLayout *hlayout;
  21. signal_handler_t *handler = obs_source_get_signal_handler(source);
  22. const char *sourceName = obs_source_get_name(source);
  23. float vol = obs_source_get_volume(source);
  24. uint32_t flags = obs_source_get_flags(source);
  25. uint32_t mixers = obs_source_get_audio_mixers(source);
  26. activeContainer = new QWidget();
  27. forceMonoContainer = new QWidget();
  28. mixerContainer = new QWidget();
  29. balanceContainer = new QWidget();
  30. labelL = new QLabel();
  31. labelR = new QLabel();
  32. nameLabel = new QLabel();
  33. active = new QLabel();
  34. stackedWidget = new QStackedWidget();
  35. volume = new QDoubleSpinBox();
  36. percent = new QSpinBox();
  37. forceMono = new QCheckBox();
  38. balance = new BalanceSlider();
  39. #if defined(_WIN32) || defined(__APPLE__) || HAVE_PULSEAUDIO
  40. monitoringType = new QComboBox();
  41. #endif
  42. syncOffset = new QSpinBox();
  43. mixer1 = new QCheckBox();
  44. mixer2 = new QCheckBox();
  45. mixer3 = new QCheckBox();
  46. mixer4 = new QCheckBox();
  47. mixer5 = new QCheckBox();
  48. mixer6 = new QCheckBox();
  49. activateSignal.Connect(handler, "activate", OBSSourceActivated, this);
  50. deactivateSignal.Connect(handler, "deactivate", OBSSourceDeactivated,
  51. this);
  52. volChangedSignal.Connect(handler, "volume", OBSSourceVolumeChanged,
  53. this);
  54. syncOffsetSignal.Connect(handler, "audio_sync", OBSSourceSyncChanged,
  55. this);
  56. flagsSignal.Connect(handler, "update_flags", OBSSourceFlagsChanged,
  57. this);
  58. mixersSignal.Connect(handler, "audio_mixers", OBSSourceMixersChanged,
  59. this);
  60. hlayout = new QHBoxLayout();
  61. hlayout->setContentsMargins(0, 0, 0, 0);
  62. activeContainer->setLayout(hlayout);
  63. hlayout = new QHBoxLayout();
  64. hlayout->setContentsMargins(0, 0, 0, 0);
  65. forceMonoContainer->setLayout(hlayout);
  66. hlayout = new QHBoxLayout();
  67. hlayout->setContentsMargins(0, 0, 0, 0);
  68. mixerContainer->setLayout(hlayout);
  69. hlayout = new QHBoxLayout();
  70. hlayout->setContentsMargins(0, 0, 0, 0);
  71. balanceContainer->setLayout(hlayout);
  72. balanceContainer->setFixedWidth(150);
  73. labelL->setText("L");
  74. labelR->setText("R");
  75. nameLabel->setText(QT_UTF8(sourceName));
  76. nameLabel->setAlignment(Qt::AlignVCenter);
  77. bool isActive = obs_source_active(source);
  78. active->setText(isActive ? QTStr("Basic.Stats.Status.Active")
  79. : QTStr("Basic.Stats.Status.Inactive"));
  80. if (isActive)
  81. setThemeID(active, "error");
  82. activeContainer->layout()->addWidget(active);
  83. activeContainer->layout()->setAlignment(active, Qt::AlignVCenter);
  84. activeContainer->setFixedWidth(50);
  85. volume->setMinimum(MIN_DB - 0.1);
  86. volume->setMaximum(MAX_DB);
  87. volume->setSingleStep(0.1);
  88. volume->setDecimals(1);
  89. volume->setSuffix(" dB");
  90. volume->setValue(obs_mul_to_db(vol));
  91. volume->setFixedWidth(100);
  92. if (volume->value() < MIN_DB)
  93. volume->setSpecialValueText("-inf dB");
  94. percent->setMinimum(0);
  95. percent->setMaximum(2000);
  96. percent->setSuffix("%");
  97. percent->setValue((int)(obs_source_get_volume(source) * 100.0f));
  98. percent->setFixedWidth(100);
  99. stackedWidget->addWidget(volume);
  100. stackedWidget->addWidget(percent);
  101. VolumeType volType = (VolumeType)config_get_int(
  102. GetGlobalConfig(), "BasicWindow", "AdvAudioVolumeType");
  103. SetVolumeWidget(volType);
  104. forceMono->setChecked((flags & OBS_SOURCE_FLAG_FORCE_MONO) != 0);
  105. forceMonoContainer->layout()->addWidget(forceMono);
  106. forceMonoContainer->layout()->setAlignment(forceMono, Qt::AlignVCenter);
  107. forceMonoContainer->setFixedWidth(50);
  108. balance->setOrientation(Qt::Horizontal);
  109. balance->setMinimum(0);
  110. balance->setMaximum(100);
  111. balance->setTickPosition(QSlider::TicksAbove);
  112. balance->setTickInterval(50);
  113. OBSBasic *main = reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
  114. const char *speakers =
  115. config_get_string(main->Config(), "Audio", "ChannelSetup");
  116. if (strcmp(speakers, "Mono") == 0)
  117. balance->setEnabled(false);
  118. else
  119. balance->setEnabled(true);
  120. float bal = obs_source_get_balance_value(source) * 100.0f;
  121. balance->setValue((int)bal);
  122. int64_t cur_sync = obs_source_get_sync_offset(source);
  123. syncOffset->setMinimum(-950);
  124. syncOffset->setMaximum(20000);
  125. syncOffset->setSuffix(" ms");
  126. syncOffset->setValue(int(cur_sync / NSEC_PER_MSEC));
  127. syncOffset->setFixedWidth(100);
  128. int idx;
  129. #if defined(_WIN32) || defined(__APPLE__) || HAVE_PULSEAUDIO
  130. monitoringType->addItem(QTStr("Basic.AdvAudio.Monitoring.None"),
  131. (int)OBS_MONITORING_TYPE_NONE);
  132. monitoringType->addItem(QTStr("Basic.AdvAudio.Monitoring.MonitorOnly"),
  133. (int)OBS_MONITORING_TYPE_MONITOR_ONLY);
  134. monitoringType->addItem(QTStr("Basic.AdvAudio.Monitoring.Both"),
  135. (int)OBS_MONITORING_TYPE_MONITOR_AND_OUTPUT);
  136. int mt = (int)obs_source_get_monitoring_type(source);
  137. idx = monitoringType->findData(mt);
  138. monitoringType->setCurrentIndex(idx);
  139. #endif
  140. mixer1->setText("1");
  141. mixer1->setChecked(mixers & (1 << 0));
  142. mixer2->setText("2");
  143. mixer2->setChecked(mixers & (1 << 1));
  144. mixer3->setText("3");
  145. mixer3->setChecked(mixers & (1 << 2));
  146. mixer4->setText("4");
  147. mixer4->setChecked(mixers & (1 << 3));
  148. mixer5->setText("5");
  149. mixer5->setChecked(mixers & (1 << 4));
  150. mixer6->setText("6");
  151. mixer6->setChecked(mixers & (1 << 5));
  152. speaker_layout sl = obs_source_get_speaker_layout(source);
  153. if (sl == SPEAKERS_STEREO) {
  154. balanceContainer->layout()->addWidget(labelL);
  155. balanceContainer->layout()->addWidget(balance);
  156. balanceContainer->layout()->addWidget(labelR);
  157. balanceContainer->setMaximumWidth(170);
  158. }
  159. mixerContainer->layout()->addWidget(mixer1);
  160. mixerContainer->layout()->addWidget(mixer2);
  161. mixerContainer->layout()->addWidget(mixer3);
  162. mixerContainer->layout()->addWidget(mixer4);
  163. mixerContainer->layout()->addWidget(mixer5);
  164. mixerContainer->layout()->addWidget(mixer6);
  165. QWidget::connect(volume, SIGNAL(valueChanged(double)), this,
  166. SLOT(volumeChanged(double)));
  167. QWidget::connect(percent, SIGNAL(valueChanged(int)), this,
  168. SLOT(percentChanged(int)));
  169. QWidget::connect(forceMono, SIGNAL(clicked(bool)), this,
  170. SLOT(downmixMonoChanged(bool)));
  171. QWidget::connect(balance, SIGNAL(valueChanged(int)), this,
  172. SLOT(balanceChanged(int)));
  173. QWidget::connect(balance, SIGNAL(doubleClicked()), this,
  174. SLOT(ResetBalance()));
  175. QWidget::connect(syncOffset, SIGNAL(valueChanged(int)), this,
  176. SLOT(syncOffsetChanged(int)));
  177. #if defined(_WIN32) || defined(__APPLE__) || HAVE_PULSEAUDIO
  178. QWidget::connect(monitoringType, SIGNAL(currentIndexChanged(int)), this,
  179. SLOT(monitoringTypeChanged(int)));
  180. #endif
  181. QWidget::connect(mixer1, SIGNAL(clicked(bool)), this,
  182. SLOT(mixer1Changed(bool)));
  183. QWidget::connect(mixer2, SIGNAL(clicked(bool)), this,
  184. SLOT(mixer2Changed(bool)));
  185. QWidget::connect(mixer3, SIGNAL(clicked(bool)), this,
  186. SLOT(mixer3Changed(bool)));
  187. QWidget::connect(mixer4, SIGNAL(clicked(bool)), this,
  188. SLOT(mixer4Changed(bool)));
  189. QWidget::connect(mixer5, SIGNAL(clicked(bool)), this,
  190. SLOT(mixer5Changed(bool)));
  191. QWidget::connect(mixer6, SIGNAL(clicked(bool)), this,
  192. SLOT(mixer6Changed(bool)));
  193. setObjectName(sourceName);
  194. }
  195. OBSAdvAudioCtrl::~OBSAdvAudioCtrl()
  196. {
  197. nameLabel->deleteLater();
  198. activeContainer->deleteLater();
  199. stackedWidget->deleteLater();
  200. forceMonoContainer->deleteLater();
  201. balanceContainer->deleteLater();
  202. syncOffset->deleteLater();
  203. #if defined(_WIN32) || defined(__APPLE__) || HAVE_PULSEAUDIO
  204. monitoringType->deleteLater();
  205. #endif
  206. mixerContainer->deleteLater();
  207. }
  208. void OBSAdvAudioCtrl::ShowAudioControl(QGridLayout *layout)
  209. {
  210. int lastRow = layout->rowCount();
  211. int idx = 0;
  212. layout->addWidget(nameLabel, lastRow, idx++);
  213. layout->addWidget(activeContainer, lastRow, idx++);
  214. layout->addWidget(stackedWidget, lastRow, idx++);
  215. layout->addWidget(forceMonoContainer, lastRow, idx++);
  216. layout->addWidget(balanceContainer, lastRow, idx++);
  217. layout->addWidget(syncOffset, lastRow, idx++);
  218. #if defined(_WIN32) || defined(__APPLE__) || HAVE_PULSEAUDIO
  219. layout->addWidget(monitoringType, lastRow, idx++);
  220. #endif
  221. layout->addWidget(mixerContainer, lastRow, idx++);
  222. layout->layout()->setAlignment(mixerContainer, Qt::AlignVCenter);
  223. layout->setHorizontalSpacing(15);
  224. }
  225. /* ------------------------------------------------------------------------- */
  226. /* OBS source callbacks */
  227. void OBSAdvAudioCtrl::OBSSourceActivated(void *param, calldata_t *calldata)
  228. {
  229. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  230. "SourceActiveChanged", Q_ARG(bool, true));
  231. UNUSED_PARAMETER(calldata);
  232. }
  233. void OBSAdvAudioCtrl::OBSSourceDeactivated(void *param, calldata_t *calldata)
  234. {
  235. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  236. "SourceActiveChanged", Q_ARG(bool, false));
  237. UNUSED_PARAMETER(calldata);
  238. }
  239. void OBSAdvAudioCtrl::OBSSourceFlagsChanged(void *param, calldata_t *calldata)
  240. {
  241. uint32_t flags = (uint32_t)calldata_int(calldata, "flags");
  242. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  243. "SourceFlagsChanged", Q_ARG(uint32_t, flags));
  244. }
  245. void OBSAdvAudioCtrl::OBSSourceVolumeChanged(void *param, calldata_t *calldata)
  246. {
  247. float volume = (float)calldata_float(calldata, "volume");
  248. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  249. "SourceVolumeChanged", Q_ARG(float, volume));
  250. }
  251. void OBSAdvAudioCtrl::OBSSourceSyncChanged(void *param, calldata_t *calldata)
  252. {
  253. int64_t offset = calldata_int(calldata, "offset");
  254. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  255. "SourceSyncChanged", Q_ARG(int64_t, offset));
  256. }
  257. void OBSAdvAudioCtrl::OBSSourceMixersChanged(void *param, calldata_t *calldata)
  258. {
  259. uint32_t mixers = (uint32_t)calldata_int(calldata, "mixers");
  260. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  261. "SourceMixersChanged",
  262. Q_ARG(uint32_t, mixers));
  263. }
  264. /* ------------------------------------------------------------------------- */
  265. /* Qt event queue source callbacks */
  266. static inline void setCheckboxState(QCheckBox *checkbox, bool checked)
  267. {
  268. checkbox->blockSignals(true);
  269. checkbox->setChecked(checked);
  270. checkbox->blockSignals(false);
  271. }
  272. void OBSAdvAudioCtrl::SourceActiveChanged(bool isActive)
  273. {
  274. if (isActive) {
  275. active->setText(QTStr("Basic.Stats.Status.Active"));
  276. setThemeID(active, "error");
  277. } else {
  278. active->setText(QTStr("Basic.Stats.Status.Inactive"));
  279. setThemeID(active, "");
  280. }
  281. }
  282. void OBSAdvAudioCtrl::SourceFlagsChanged(uint32_t flags)
  283. {
  284. bool forceMonoVal = (flags & OBS_SOURCE_FLAG_FORCE_MONO) != 0;
  285. setCheckboxState(forceMono, forceMonoVal);
  286. }
  287. void OBSAdvAudioCtrl::SourceVolumeChanged(float value)
  288. {
  289. volume->blockSignals(true);
  290. percent->blockSignals(true);
  291. volume->setValue(obs_mul_to_db(value));
  292. percent->setValue((int)std::round(value * 100.0f));
  293. percent->blockSignals(false);
  294. volume->blockSignals(false);
  295. }
  296. void OBSAdvAudioCtrl::SourceSyncChanged(int64_t offset)
  297. {
  298. syncOffset->setValue(offset / NSEC_PER_MSEC);
  299. }
  300. void OBSAdvAudioCtrl::SourceMixersChanged(uint32_t mixers)
  301. {
  302. setCheckboxState(mixer1, mixers & (1 << 0));
  303. setCheckboxState(mixer2, mixers & (1 << 1));
  304. setCheckboxState(mixer3, mixers & (1 << 2));
  305. setCheckboxState(mixer4, mixers & (1 << 3));
  306. setCheckboxState(mixer5, mixers & (1 << 4));
  307. setCheckboxState(mixer6, mixers & (1 << 5));
  308. }
  309. /* ------------------------------------------------------------------------- */
  310. /* Qt control callbacks */
  311. void OBSAdvAudioCtrl::volumeChanged(double db)
  312. {
  313. if (db < MIN_DB) {
  314. volume->setSpecialValueText("-inf dB");
  315. db = -INFINITY;
  316. }
  317. float val = obs_db_to_mul(db);
  318. obs_source_set_volume(source, val);
  319. }
  320. void OBSAdvAudioCtrl::percentChanged(int percent)
  321. {
  322. obs_source_set_volume(source, (float)percent / 100.0f);
  323. }
  324. void OBSAdvAudioCtrl::downmixMonoChanged(bool checked)
  325. {
  326. uint32_t flags = obs_source_get_flags(source);
  327. bool forceMonoActive = (flags & OBS_SOURCE_FLAG_FORCE_MONO) != 0;
  328. if (forceMonoActive != checked) {
  329. if (checked)
  330. flags |= OBS_SOURCE_FLAG_FORCE_MONO;
  331. else
  332. flags &= ~OBS_SOURCE_FLAG_FORCE_MONO;
  333. obs_source_set_flags(source, flags);
  334. }
  335. }
  336. void OBSAdvAudioCtrl::balanceChanged(int val)
  337. {
  338. float bal = (float)val / 100.0f;
  339. if (abs(50 - val) < 10) {
  340. balance->blockSignals(true);
  341. balance->setValue(50);
  342. bal = 0.5f;
  343. balance->blockSignals(false);
  344. }
  345. obs_source_set_balance_value(source, bal);
  346. }
  347. void OBSAdvAudioCtrl::ResetBalance()
  348. {
  349. balance->setValue(50);
  350. }
  351. void OBSAdvAudioCtrl::syncOffsetChanged(int milliseconds)
  352. {
  353. int64_t cur_val = obs_source_get_sync_offset(source);
  354. if (cur_val / NSEC_PER_MSEC != milliseconds)
  355. obs_source_set_sync_offset(source, int64_t(milliseconds) *
  356. NSEC_PER_MSEC);
  357. }
  358. void OBSAdvAudioCtrl::monitoringTypeChanged(int index)
  359. {
  360. int mt = monitoringType->itemData(index).toInt();
  361. obs_source_set_monitoring_type(source, (obs_monitoring_type)mt);
  362. const char *type = nullptr;
  363. switch (mt) {
  364. case OBS_MONITORING_TYPE_NONE:
  365. type = "none";
  366. break;
  367. case OBS_MONITORING_TYPE_MONITOR_ONLY:
  368. type = "monitor only";
  369. break;
  370. case OBS_MONITORING_TYPE_MONITOR_AND_OUTPUT:
  371. type = "monitor and output";
  372. break;
  373. }
  374. blog(LOG_INFO, "User changed audio monitoring for source '%s' to: %s",
  375. obs_source_get_name(source), type);
  376. }
  377. static inline void setMixer(obs_source_t *source, const int mixerIdx,
  378. const bool checked)
  379. {
  380. uint32_t mixers = obs_source_get_audio_mixers(source);
  381. uint32_t new_mixers = mixers;
  382. if (checked)
  383. new_mixers |= (1 << mixerIdx);
  384. else
  385. new_mixers &= ~(1 << mixerIdx);
  386. obs_source_set_audio_mixers(source, new_mixers);
  387. }
  388. void OBSAdvAudioCtrl::mixer1Changed(bool checked)
  389. {
  390. setMixer(source, 0, checked);
  391. }
  392. void OBSAdvAudioCtrl::mixer2Changed(bool checked)
  393. {
  394. setMixer(source, 1, checked);
  395. }
  396. void OBSAdvAudioCtrl::mixer3Changed(bool checked)
  397. {
  398. setMixer(source, 2, checked);
  399. }
  400. void OBSAdvAudioCtrl::mixer4Changed(bool checked)
  401. {
  402. setMixer(source, 3, checked);
  403. }
  404. void OBSAdvAudioCtrl::mixer5Changed(bool checked)
  405. {
  406. setMixer(source, 4, checked);
  407. }
  408. void OBSAdvAudioCtrl::mixer6Changed(bool checked)
  409. {
  410. setMixer(source, 5, checked);
  411. }
  412. void OBSAdvAudioCtrl::SetVolumeWidget(VolumeType type)
  413. {
  414. switch (type) {
  415. case VolumeType::Percent:
  416. stackedWidget->setCurrentWidget(percent);
  417. break;
  418. case VolumeType::dB:
  419. stackedWidget->setCurrentWidget(volume);
  420. break;
  421. }
  422. }