adv-audio-control.cpp 12 KB

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