adv-audio-control.cpp 20 KB

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