adv-audio-control.cpp 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707
  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. QString sourceName = QT_UTF8(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. iconLabel = new QLabel();
  33. nameLabel = new QLabel();
  34. active = new QLabel();
  35. stackedWidget = new QStackedWidget();
  36. volume = new QDoubleSpinBox();
  37. percent = new QSpinBox();
  38. forceMono = new QCheckBox();
  39. balance = new BalanceSlider();
  40. if (obs_audio_monitoring_available())
  41. monitoringType = new QComboBox();
  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. if (obs_audio_monitoring_available())
  59. monitoringTypeSignal.Connect(handler, "audio_monitoring",
  60. OBSSourceMonitoringTypeChanged,
  61. this);
  62. mixersSignal.Connect(handler, "audio_mixers", OBSSourceMixersChanged,
  63. this);
  64. balChangedSignal.Connect(handler, "audio_balance",
  65. OBSSourceBalanceChanged, this);
  66. renameSignal.Connect(handler, "rename", OBSSourceRenamed, this);
  67. hlayout = new QHBoxLayout();
  68. hlayout->setContentsMargins(0, 0, 0, 0);
  69. activeContainer->setLayout(hlayout);
  70. hlayout = new QHBoxLayout();
  71. hlayout->setContentsMargins(0, 0, 0, 0);
  72. forceMonoContainer->setLayout(hlayout);
  73. hlayout = new QHBoxLayout();
  74. hlayout->setContentsMargins(0, 0, 0, 0);
  75. mixerContainer->setLayout(hlayout);
  76. hlayout = new QHBoxLayout();
  77. hlayout->setContentsMargins(0, 0, 0, 0);
  78. balanceContainer->setLayout(hlayout);
  79. balanceContainer->setFixedWidth(150);
  80. labelL->setText("L");
  81. labelR->setText("R");
  82. OBSBasic *main = reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
  83. QIcon sourceIcon = main->GetSourceIcon(obs_source_get_id(source));
  84. QPixmap pixmap = sourceIcon.pixmap(QSize(16, 16));
  85. iconLabel->setPixmap(pixmap);
  86. iconLabel->setFixedSize(16, 16);
  87. iconLabel->setStyleSheet("background: none");
  88. SetSourceName(sourceName);
  89. nameLabel->setAlignment(Qt::AlignVCenter);
  90. bool isActive = obs_source_active(source);
  91. active->setText(isActive ? QTStr("Basic.Stats.Status.Active")
  92. : QTStr("Basic.Stats.Status.Inactive"));
  93. if (isActive)
  94. setThemeID(active, "error");
  95. activeContainer->layout()->addWidget(active);
  96. activeContainer->layout()->setAlignment(active, Qt::AlignVCenter);
  97. activeContainer->setFixedWidth(120);
  98. volume->setMinimum(MIN_DB - 0.1);
  99. volume->setMaximum(MAX_DB);
  100. volume->setSingleStep(0.1);
  101. volume->setDecimals(1);
  102. volume->setSuffix(" dB");
  103. volume->setValue(obs_mul_to_db(vol));
  104. volume->setFixedWidth(100);
  105. volume->setAccessibleName(
  106. QTStr("Basic.AdvAudio.VolumeSource").arg(sourceName));
  107. if (volume->value() < MIN_DB) {
  108. volume->setSpecialValueText("-inf dB");
  109. volume->setAccessibleDescription("-inf dB");
  110. }
  111. percent->setMinimum(0);
  112. percent->setMaximum(2000);
  113. percent->setSuffix("%");
  114. percent->setValue((int)(obs_source_get_volume(source) * 100.0f));
  115. percent->setFixedWidth(100);
  116. percent->setAccessibleName(
  117. QTStr("Basic.AdvAudio.VolumeSource").arg(sourceName));
  118. stackedWidget->addWidget(volume);
  119. stackedWidget->addWidget(percent);
  120. VolumeType volType = (VolumeType)config_get_int(
  121. GetGlobalConfig(), "BasicWindow", "AdvAudioVolumeType");
  122. SetVolumeWidget(volType);
  123. forceMono->setChecked((flags & OBS_SOURCE_FLAG_FORCE_MONO) != 0);
  124. forceMono->setAccessibleName(
  125. QTStr("Basic.AdvAudio.MonoSource").arg(sourceName));
  126. forceMonoContainer->layout()->addWidget(forceMono);
  127. forceMonoContainer->layout()->setAlignment(forceMono, Qt::AlignVCenter);
  128. forceMonoContainer->setFixedWidth(50);
  129. balance->setOrientation(Qt::Horizontal);
  130. balance->setMinimum(0);
  131. balance->setMaximum(100);
  132. balance->setTickPosition(QSlider::TicksAbove);
  133. balance->setTickInterval(50);
  134. balance->setAccessibleName(
  135. QTStr("Basic.AdvAudio.BalanceSource").arg(sourceName));
  136. const char *speakers =
  137. config_get_string(main->Config(), "Audio", "ChannelSetup");
  138. if (strcmp(speakers, "Mono") == 0)
  139. balance->setEnabled(false);
  140. else
  141. balance->setEnabled(true);
  142. float bal = obs_source_get_balance_value(source) * 100.0f;
  143. balance->setValue((int)bal);
  144. int64_t cur_sync = obs_source_get_sync_offset(source);
  145. syncOffset->setMinimum(-950);
  146. syncOffset->setMaximum(20000);
  147. syncOffset->setSuffix(" ms");
  148. syncOffset->setValue(int(cur_sync / NSEC_PER_MSEC));
  149. syncOffset->setFixedWidth(100);
  150. syncOffset->setAccessibleName(
  151. QTStr("Basic.AdvAudio.SyncOffsetSource").arg(sourceName));
  152. int idx;
  153. if (obs_audio_monitoring_available()) {
  154. monitoringType->addItem(QTStr("Basic.AdvAudio.Monitoring.None"),
  155. (int)OBS_MONITORING_TYPE_NONE);
  156. monitoringType->addItem(
  157. QTStr("Basic.AdvAudio.Monitoring.MonitorOnly"),
  158. (int)OBS_MONITORING_TYPE_MONITOR_ONLY);
  159. monitoringType->addItem(
  160. QTStr("Basic.AdvAudio.Monitoring.Both"),
  161. (int)OBS_MONITORING_TYPE_MONITOR_AND_OUTPUT);
  162. int mt = (int)obs_source_get_monitoring_type(source);
  163. idx = monitoringType->findData(mt);
  164. monitoringType->setCurrentIndex(idx);
  165. monitoringType->setAccessibleName(
  166. QTStr("Basic.AdvAudio.MonitoringSource")
  167. .arg(sourceName));
  168. }
  169. mixer1->setText("1");
  170. mixer1->setChecked(mixers & (1 << 0));
  171. mixer1->setAccessibleName(
  172. QTStr("Basic.Settings.Output.Adv.Audio.Track1"));
  173. mixer2->setText("2");
  174. mixer2->setChecked(mixers & (1 << 1));
  175. mixer2->setAccessibleName(
  176. QTStr("Basic.Settings.Output.Adv.Audio.Track2"));
  177. mixer3->setText("3");
  178. mixer3->setChecked(mixers & (1 << 2));
  179. mixer3->setAccessibleName(
  180. QTStr("Basic.Settings.Output.Adv.Audio.Track3"));
  181. mixer4->setText("4");
  182. mixer4->setChecked(mixers & (1 << 3));
  183. mixer4->setAccessibleName(
  184. QTStr("Basic.Settings.Output.Adv.Audio.Track4"));
  185. mixer5->setText("5");
  186. mixer5->setChecked(mixers & (1 << 4));
  187. mixer5->setAccessibleName(
  188. QTStr("Basic.Settings.Output.Adv.Audio.Track5"));
  189. mixer6->setText("6");
  190. mixer6->setChecked(mixers & (1 << 5));
  191. mixer6->setAccessibleName(
  192. QTStr("Basic.Settings.Output.Adv.Audio.Track6"));
  193. speaker_layout sl = obs_source_get_speaker_layout(source);
  194. if (sl == SPEAKERS_STEREO) {
  195. balanceContainer->layout()->addWidget(labelL);
  196. balanceContainer->layout()->addWidget(balance);
  197. balanceContainer->layout()->addWidget(labelR);
  198. balanceContainer->setMaximumWidth(170);
  199. }
  200. mixerContainer->layout()->addWidget(mixer1);
  201. mixerContainer->layout()->addWidget(mixer2);
  202. mixerContainer->layout()->addWidget(mixer3);
  203. mixerContainer->layout()->addWidget(mixer4);
  204. mixerContainer->layout()->addWidget(mixer5);
  205. mixerContainer->layout()->addWidget(mixer6);
  206. QWidget::connect(volume, SIGNAL(valueChanged(double)), this,
  207. SLOT(volumeChanged(double)));
  208. QWidget::connect(percent, SIGNAL(valueChanged(int)), this,
  209. SLOT(percentChanged(int)));
  210. QWidget::connect(forceMono, SIGNAL(clicked(bool)), this,
  211. SLOT(downmixMonoChanged(bool)));
  212. QWidget::connect(balance, SIGNAL(valueChanged(int)), this,
  213. SLOT(balanceChanged(int)));
  214. QWidget::connect(balance, SIGNAL(doubleClicked()), this,
  215. SLOT(ResetBalance()));
  216. QWidget::connect(syncOffset, SIGNAL(valueChanged(int)), this,
  217. SLOT(syncOffsetChanged(int)));
  218. if (obs_audio_monitoring_available())
  219. QWidget::connect(monitoringType,
  220. SIGNAL(currentIndexChanged(int)), this,
  221. SLOT(monitoringTypeChanged(int)));
  222. QWidget::connect(mixer1, SIGNAL(clicked(bool)), this,
  223. SLOT(mixer1Changed(bool)));
  224. QWidget::connect(mixer2, SIGNAL(clicked(bool)), this,
  225. SLOT(mixer2Changed(bool)));
  226. QWidget::connect(mixer3, SIGNAL(clicked(bool)), this,
  227. SLOT(mixer3Changed(bool)));
  228. QWidget::connect(mixer4, SIGNAL(clicked(bool)), this,
  229. SLOT(mixer4Changed(bool)));
  230. QWidget::connect(mixer5, SIGNAL(clicked(bool)), this,
  231. SLOT(mixer5Changed(bool)));
  232. QWidget::connect(mixer6, SIGNAL(clicked(bool)), this,
  233. SLOT(mixer6Changed(bool)));
  234. setObjectName(sourceName);
  235. }
  236. OBSAdvAudioCtrl::~OBSAdvAudioCtrl()
  237. {
  238. iconLabel->deleteLater();
  239. nameLabel->deleteLater();
  240. activeContainer->deleteLater();
  241. stackedWidget->deleteLater();
  242. forceMonoContainer->deleteLater();
  243. balanceContainer->deleteLater();
  244. syncOffset->deleteLater();
  245. if (obs_audio_monitoring_available())
  246. monitoringType->deleteLater();
  247. mixerContainer->deleteLater();
  248. }
  249. void OBSAdvAudioCtrl::ShowAudioControl(QGridLayout *layout)
  250. {
  251. int lastRow = layout->rowCount();
  252. int idx = 0;
  253. layout->addWidget(iconLabel, lastRow, idx++);
  254. layout->addWidget(nameLabel, lastRow, idx++);
  255. layout->addWidget(activeContainer, lastRow, idx++);
  256. layout->addWidget(stackedWidget, lastRow, idx++);
  257. layout->addWidget(forceMonoContainer, lastRow, idx++);
  258. layout->addWidget(balanceContainer, lastRow, idx++);
  259. layout->addWidget(syncOffset, lastRow, idx++);
  260. if (obs_audio_monitoring_available())
  261. layout->addWidget(monitoringType, lastRow, idx++);
  262. layout->addWidget(mixerContainer, lastRow, idx++);
  263. layout->layout()->setAlignment(mixerContainer, Qt::AlignVCenter);
  264. layout->setHorizontalSpacing(15);
  265. }
  266. /* ------------------------------------------------------------------------- */
  267. /* OBS source callbacks */
  268. void OBSAdvAudioCtrl::OBSSourceActivated(void *param, calldata_t *calldata)
  269. {
  270. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  271. "SourceActiveChanged", Q_ARG(bool, true));
  272. UNUSED_PARAMETER(calldata);
  273. }
  274. void OBSAdvAudioCtrl::OBSSourceDeactivated(void *param, calldata_t *calldata)
  275. {
  276. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  277. "SourceActiveChanged", Q_ARG(bool, false));
  278. UNUSED_PARAMETER(calldata);
  279. }
  280. void OBSAdvAudioCtrl::OBSSourceFlagsChanged(void *param, calldata_t *calldata)
  281. {
  282. uint32_t flags = (uint32_t)calldata_int(calldata, "flags");
  283. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  284. "SourceFlagsChanged", Q_ARG(uint32_t, flags));
  285. }
  286. void OBSAdvAudioCtrl::OBSSourceVolumeChanged(void *param, calldata_t *calldata)
  287. {
  288. float volume = (float)calldata_float(calldata, "volume");
  289. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  290. "SourceVolumeChanged", Q_ARG(float, volume));
  291. }
  292. void OBSAdvAudioCtrl::OBSSourceSyncChanged(void *param, calldata_t *calldata)
  293. {
  294. int64_t offset = calldata_int(calldata, "offset");
  295. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  296. "SourceSyncChanged", Q_ARG(int64_t, offset));
  297. }
  298. void OBSAdvAudioCtrl::OBSSourceMonitoringTypeChanged(void *param,
  299. calldata_t *calldata)
  300. {
  301. int type = calldata_int(calldata, "type");
  302. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  303. "SourceMonitoringTypeChanged",
  304. Q_ARG(int, type));
  305. }
  306. void OBSAdvAudioCtrl::OBSSourceMixersChanged(void *param, calldata_t *calldata)
  307. {
  308. uint32_t mixers = (uint32_t)calldata_int(calldata, "mixers");
  309. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  310. "SourceMixersChanged",
  311. Q_ARG(uint32_t, mixers));
  312. }
  313. void OBSAdvAudioCtrl::OBSSourceBalanceChanged(void *param, calldata_t *calldata)
  314. {
  315. int balance = (float)calldata_float(calldata, "balance") * 100.0f;
  316. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  317. "SourceBalanceChanged", Q_ARG(int, balance));
  318. }
  319. void OBSAdvAudioCtrl::OBSSourceRenamed(void *param, calldata_t *calldata)
  320. {
  321. QString newName = QT_UTF8(calldata_string(calldata, "new_name"));
  322. QMetaObject::invokeMethod(reinterpret_cast<OBSAdvAudioCtrl *>(param),
  323. "SetSourceName",
  324. Q_ARG(const QString &, newName));
  325. }
  326. /* ------------------------------------------------------------------------- */
  327. /* Qt event queue source callbacks */
  328. static inline void setCheckboxState(QCheckBox *checkbox, bool checked)
  329. {
  330. checkbox->blockSignals(true);
  331. checkbox->setChecked(checked);
  332. checkbox->blockSignals(false);
  333. }
  334. void OBSAdvAudioCtrl::SourceActiveChanged(bool isActive)
  335. {
  336. if (isActive) {
  337. active->setText(QTStr("Basic.Stats.Status.Active"));
  338. setThemeID(active, "error");
  339. } else {
  340. active->setText(QTStr("Basic.Stats.Status.Inactive"));
  341. setThemeID(active, "");
  342. }
  343. }
  344. void OBSAdvAudioCtrl::SourceFlagsChanged(uint32_t flags)
  345. {
  346. bool forceMonoVal = (flags & OBS_SOURCE_FLAG_FORCE_MONO) != 0;
  347. setCheckboxState(forceMono, forceMonoVal);
  348. }
  349. void OBSAdvAudioCtrl::SourceVolumeChanged(float value)
  350. {
  351. volume->blockSignals(true);
  352. percent->blockSignals(true);
  353. volume->setValue(obs_mul_to_db(value));
  354. percent->setValue((int)std::round(value * 100.0f));
  355. percent->blockSignals(false);
  356. volume->blockSignals(false);
  357. }
  358. void OBSAdvAudioCtrl::SourceBalanceChanged(int value)
  359. {
  360. balance->blockSignals(true);
  361. balance->setValue(value);
  362. balance->blockSignals(false);
  363. }
  364. void OBSAdvAudioCtrl::SourceSyncChanged(int64_t offset)
  365. {
  366. syncOffset->blockSignals(true);
  367. syncOffset->setValue(offset / NSEC_PER_MSEC);
  368. syncOffset->blockSignals(false);
  369. }
  370. void OBSAdvAudioCtrl::SourceMonitoringTypeChanged(int type)
  371. {
  372. int idx = monitoringType->findData(type);
  373. monitoringType->blockSignals(true);
  374. monitoringType->setCurrentIndex(idx);
  375. monitoringType->blockSignals(false);
  376. }
  377. void OBSAdvAudioCtrl::SourceMixersChanged(uint32_t mixers)
  378. {
  379. setCheckboxState(mixer1, mixers & (1 << 0));
  380. setCheckboxState(mixer2, mixers & (1 << 1));
  381. setCheckboxState(mixer3, mixers & (1 << 2));
  382. setCheckboxState(mixer4, mixers & (1 << 3));
  383. setCheckboxState(mixer5, mixers & (1 << 4));
  384. setCheckboxState(mixer6, mixers & (1 << 5));
  385. }
  386. /* ------------------------------------------------------------------------- */
  387. /* Qt control callbacks */
  388. void OBSAdvAudioCtrl::volumeChanged(double db)
  389. {
  390. float prev = obs_source_get_volume(source);
  391. if (db < MIN_DB) {
  392. volume->setSpecialValueText("-inf dB");
  393. db = -INFINITY;
  394. }
  395. float val = obs_db_to_mul(db);
  396. obs_source_set_volume(source, val);
  397. auto undo_redo = [](const std::string &name, float val) {
  398. OBSSourceAutoRelease source =
  399. obs_get_source_by_name(name.c_str());
  400. obs_source_set_volume(source, val);
  401. };
  402. const char *name = obs_source_get_name(source);
  403. OBSBasic *main = OBSBasic::Get();
  404. main->undo_s.add_action(
  405. QTStr("Undo.Volume.Change").arg(name),
  406. std::bind(undo_redo, std::placeholders::_1, prev),
  407. std::bind(undo_redo, std::placeholders::_1, val), name, name,
  408. true);
  409. }
  410. void OBSAdvAudioCtrl::percentChanged(int percent)
  411. {
  412. float prev = obs_source_get_volume(source);
  413. float val = (float)percent / 100.0f;
  414. obs_source_set_volume(source, val);
  415. auto undo_redo = [](const std::string &name, float val) {
  416. OBSSourceAutoRelease source =
  417. obs_get_source_by_name(name.c_str());
  418. obs_source_set_volume(source, val);
  419. };
  420. const char *name = obs_source_get_name(source);
  421. OBSBasic::Get()->undo_s.add_action(
  422. QTStr("Undo.Volume.Change").arg(name),
  423. std::bind(undo_redo, std::placeholders::_1, prev),
  424. std::bind(undo_redo, std::placeholders::_1, val), name, name,
  425. true);
  426. }
  427. static inline void set_mono(obs_source_t *source, bool mono)
  428. {
  429. uint32_t flags = obs_source_get_flags(source);
  430. if (mono)
  431. flags |= OBS_SOURCE_FLAG_FORCE_MONO;
  432. else
  433. flags &= ~OBS_SOURCE_FLAG_FORCE_MONO;
  434. obs_source_set_flags(source, flags);
  435. }
  436. void OBSAdvAudioCtrl::downmixMonoChanged(bool val)
  437. {
  438. uint32_t flags = obs_source_get_flags(source);
  439. bool forceMonoActive = (flags & OBS_SOURCE_FLAG_FORCE_MONO) != 0;
  440. if (forceMonoActive == val)
  441. return;
  442. if (val)
  443. flags |= OBS_SOURCE_FLAG_FORCE_MONO;
  444. else
  445. flags &= ~OBS_SOURCE_FLAG_FORCE_MONO;
  446. obs_source_set_flags(source, flags);
  447. auto undo_redo = [](const std::string &name, bool val) {
  448. OBSSourceAutoRelease source =
  449. obs_get_source_by_name(name.c_str());
  450. set_mono(source, val);
  451. };
  452. QString text = QTStr(val ? "Undo.ForceMono.On" : "Undo.ForceMono.Off");
  453. const char *name = obs_source_get_name(source);
  454. OBSBasic::Get()->undo_s.add_action(
  455. text.arg(name),
  456. std::bind(undo_redo, std::placeholders::_1, !val),
  457. std::bind(undo_redo, std::placeholders::_1, val), name, name);
  458. }
  459. void OBSAdvAudioCtrl::balanceChanged(int val)
  460. {
  461. float prev = obs_source_get_balance_value(source);
  462. float bal = (float)val / 100.0f;
  463. if (abs(50 - val) < 10) {
  464. balance->blockSignals(true);
  465. balance->setValue(50);
  466. bal = 0.5f;
  467. balance->blockSignals(false);
  468. }
  469. obs_source_set_balance_value(source, bal);
  470. auto undo_redo = [](const std::string &name, float val) {
  471. OBSSourceAutoRelease source =
  472. obs_get_source_by_name(name.c_str());
  473. obs_source_set_balance_value(source, val);
  474. };
  475. const char *name = obs_source_get_name(source);
  476. OBSBasic::Get()->undo_s.add_action(
  477. QTStr("Undo.Balance.Change").arg(name),
  478. std::bind(undo_redo, std::placeholders::_1, prev),
  479. std::bind(undo_redo, std::placeholders::_1, bal), name, name,
  480. true);
  481. }
  482. void OBSAdvAudioCtrl::ResetBalance()
  483. {
  484. balance->setValue(50);
  485. }
  486. void OBSAdvAudioCtrl::syncOffsetChanged(int milliseconds)
  487. {
  488. int64_t prev = obs_source_get_sync_offset(source);
  489. int64_t val = int64_t(milliseconds) * NSEC_PER_MSEC;
  490. if (prev / NSEC_PER_MSEC == milliseconds)
  491. return;
  492. obs_source_set_sync_offset(source, val);
  493. auto undo_redo = [](const std::string &name, int64_t val) {
  494. OBSSourceAutoRelease source =
  495. obs_get_source_by_name(name.c_str());
  496. obs_source_set_sync_offset(source, val);
  497. };
  498. const char *name = obs_source_get_name(source);
  499. OBSBasic::Get()->undo_s.add_action(
  500. QTStr("Undo.SyncOffset.Change").arg(name),
  501. std::bind(undo_redo, std::placeholders::_1, prev),
  502. std::bind(undo_redo, std::placeholders::_1, val), name, name,
  503. true);
  504. }
  505. void OBSAdvAudioCtrl::monitoringTypeChanged(int index)
  506. {
  507. obs_monitoring_type prev = obs_source_get_monitoring_type(source);
  508. obs_monitoring_type mt =
  509. (obs_monitoring_type)monitoringType->itemData(index).toInt();
  510. obs_source_set_monitoring_type(source, mt);
  511. const char *type = nullptr;
  512. switch (mt) {
  513. case OBS_MONITORING_TYPE_NONE:
  514. type = "none";
  515. break;
  516. case OBS_MONITORING_TYPE_MONITOR_ONLY:
  517. type = "monitor only";
  518. break;
  519. case OBS_MONITORING_TYPE_MONITOR_AND_OUTPUT:
  520. type = "monitor and output";
  521. break;
  522. }
  523. const char *name = obs_source_get_name(source);
  524. blog(LOG_INFO, "User changed audio monitoring for source '%s' to: %s",
  525. name ? name : "(null)", type);
  526. auto undo_redo = [](const std::string &name, obs_monitoring_type val) {
  527. OBSSourceAutoRelease source =
  528. obs_get_source_by_name(name.c_str());
  529. obs_source_set_monitoring_type(source, val);
  530. };
  531. OBSBasic::Get()->undo_s.add_action(
  532. QTStr("Undo.MonitoringType.Change").arg(name),
  533. std::bind(undo_redo, std::placeholders::_1, prev),
  534. std::bind(undo_redo, std::placeholders::_1, mt), name, name);
  535. }
  536. static inline void setMixer(obs_source_t *source, const int mixerIdx,
  537. const bool checked)
  538. {
  539. uint32_t mixers = obs_source_get_audio_mixers(source);
  540. uint32_t new_mixers = mixers;
  541. if (checked)
  542. new_mixers |= (1 << mixerIdx);
  543. else
  544. new_mixers &= ~(1 << mixerIdx);
  545. obs_source_set_audio_mixers(source, new_mixers);
  546. auto undo_redo = [](const std::string &name, uint32_t mixers) {
  547. OBSSourceAutoRelease source =
  548. obs_get_source_by_name(name.c_str());
  549. obs_source_set_audio_mixers(source, mixers);
  550. };
  551. const char *name = obs_source_get_name(source);
  552. OBSBasic::Get()->undo_s.add_action(
  553. QTStr("Undo.Mixers.Change").arg(name),
  554. std::bind(undo_redo, std::placeholders::_1, mixers),
  555. std::bind(undo_redo, std::placeholders::_1, new_mixers), name,
  556. name);
  557. }
  558. void OBSAdvAudioCtrl::mixer1Changed(bool checked)
  559. {
  560. setMixer(source, 0, checked);
  561. }
  562. void OBSAdvAudioCtrl::mixer2Changed(bool checked)
  563. {
  564. setMixer(source, 1, checked);
  565. }
  566. void OBSAdvAudioCtrl::mixer3Changed(bool checked)
  567. {
  568. setMixer(source, 2, checked);
  569. }
  570. void OBSAdvAudioCtrl::mixer4Changed(bool checked)
  571. {
  572. setMixer(source, 3, checked);
  573. }
  574. void OBSAdvAudioCtrl::mixer5Changed(bool checked)
  575. {
  576. setMixer(source, 4, checked);
  577. }
  578. void OBSAdvAudioCtrl::mixer6Changed(bool checked)
  579. {
  580. setMixer(source, 5, checked);
  581. }
  582. void OBSAdvAudioCtrl::SetVolumeWidget(VolumeType type)
  583. {
  584. switch (type) {
  585. case VolumeType::Percent:
  586. stackedWidget->setCurrentWidget(percent);
  587. break;
  588. case VolumeType::dB:
  589. stackedWidget->setCurrentWidget(volume);
  590. break;
  591. }
  592. }
  593. void OBSAdvAudioCtrl::SetIconVisible(bool visible)
  594. {
  595. visible ? iconLabel->show() : iconLabel->hide();
  596. }
  597. void OBSAdvAudioCtrl::SetSourceName(const QString &newName)
  598. {
  599. if (nameLabel->text() != newName)
  600. nameLabel->setText(newName);
  601. }