context-bar-controls.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735
  1. #include "window-basic-main.hpp"
  2. #include "context-bar-controls.hpp"
  3. #include "qt-wrappers.hpp"
  4. #include "obs-app.hpp"
  5. #include <QStandardItemModel>
  6. #include <QColorDialog>
  7. #include <QFontDialog>
  8. #include "ui_browser-source-toolbar.h"
  9. #include "ui_device-select-toolbar.h"
  10. #include "ui_game-capture-toolbar.h"
  11. #include "ui_image-source-toolbar.h"
  12. #include "ui_color-source-toolbar.h"
  13. #include "ui_text-source-toolbar.h"
  14. #ifdef _WIN32
  15. #define get_os_module(win, mac, linux) obs_get_module(win)
  16. #define get_os_text(mod, win, mac, linux) obs_module_get_locale_text(mod, win)
  17. #elif __APPLE__
  18. #define get_os_module(win, mac, linux) obs_get_module(mac)
  19. #define get_os_text(mod, win, mac, linux) obs_module_get_locale_text(mod, mac)
  20. #else
  21. #define get_os_module(win, mac, linux) obs_get_module(linux)
  22. #define get_os_text(mod, win, mac, linux) obs_module_get_locale_text(mod, linux)
  23. #endif
  24. /* ========================================================================= */
  25. SourceToolbar::SourceToolbar(QWidget *parent, OBSSource source)
  26. : QWidget(parent),
  27. weakSource(OBSGetWeakRef(source)),
  28. props(obs_source_properties(source), obs_properties_destroy)
  29. {
  30. }
  31. void SourceToolbar::SaveOldProperties(obs_source_t *source)
  32. {
  33. oldData = obs_data_create();
  34. OBSDataAutoRelease oldSettings = obs_source_get_settings(source);
  35. obs_data_apply(oldData, oldSettings);
  36. obs_data_set_string(oldData, "undo_sname", obs_source_get_name(source));
  37. }
  38. void SourceToolbar::SetUndoProperties(obs_source_t *source, bool repeatable)
  39. {
  40. if (!oldData) {
  41. blog(LOG_ERROR, "%s: somehow oldData was null.", __FUNCTION__);
  42. return;
  43. }
  44. OBSBasic *main = reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
  45. OBSSource currentSceneSource = main->GetCurrentSceneSource();
  46. if (!currentSceneSource)
  47. return;
  48. std::string scene_name = obs_source_get_name(currentSceneSource);
  49. auto undo_redo = [scene_name,
  50. main = std::move(main)](const std::string &data) {
  51. OBSDataAutoRelease settings =
  52. obs_data_create_from_json(data.c_str());
  53. OBSSourceAutoRelease source = obs_get_source_by_name(
  54. obs_data_get_string(settings, "undo_sname"));
  55. obs_source_reset_settings(source, settings);
  56. OBSSourceAutoRelease scene_source =
  57. obs_get_source_by_name(scene_name.c_str());
  58. main->SetCurrentScene(scene_source.Get(), true);
  59. main->UpdateContextBar();
  60. };
  61. OBSDataAutoRelease new_settings = obs_data_create();
  62. OBSDataAutoRelease curr_settings = obs_source_get_settings(source);
  63. obs_data_apply(new_settings, curr_settings);
  64. obs_data_set_string(new_settings, "undo_sname",
  65. obs_source_get_name(source));
  66. std::string undo_data(obs_data_get_json(oldData));
  67. std::string redo_data(obs_data_get_json(new_settings));
  68. if (undo_data.compare(redo_data) != 0)
  69. main->undo_s.add_action(
  70. QTStr("Undo.Properties")
  71. .arg(obs_source_get_name(source)),
  72. undo_redo, undo_redo, undo_data, redo_data, repeatable);
  73. oldData = nullptr;
  74. }
  75. /* ========================================================================= */
  76. BrowserToolbar::BrowserToolbar(QWidget *parent, OBSSource source)
  77. : SourceToolbar(parent, source), ui(new Ui_BrowserSourceToolbar)
  78. {
  79. ui->setupUi(this);
  80. }
  81. BrowserToolbar::~BrowserToolbar() {}
  82. void BrowserToolbar::on_refresh_clicked()
  83. {
  84. OBSSource source = GetSource();
  85. if (!source) {
  86. return;
  87. }
  88. obs_property_t *p = obs_properties_get(props.get(), "refreshnocache");
  89. obs_property_button_clicked(p, source.Get());
  90. }
  91. /* ========================================================================= */
  92. ComboSelectToolbar::ComboSelectToolbar(QWidget *parent, OBSSource source)
  93. : SourceToolbar(parent, source), ui(new Ui_DeviceSelectToolbar)
  94. {
  95. ui->setupUi(this);
  96. }
  97. ComboSelectToolbar::~ComboSelectToolbar() {}
  98. static int FillPropertyCombo(QComboBox *c, obs_property_t *p,
  99. const std::string &cur_id, bool is_int = false)
  100. {
  101. size_t count = obs_property_list_item_count(p);
  102. int cur_idx = -1;
  103. for (size_t i = 0; i < count; i++) {
  104. const char *name = obs_property_list_item_name(p, i);
  105. std::string id;
  106. if (is_int) {
  107. id = std::to_string(obs_property_list_item_int(p, i));
  108. } else {
  109. const char *val = obs_property_list_item_string(p, i);
  110. id = val ? val : "";
  111. }
  112. if (cur_id == id)
  113. cur_idx = (int)i;
  114. c->addItem(name, id.c_str());
  115. }
  116. return cur_idx;
  117. }
  118. void UpdateSourceComboToolbarProperties(QComboBox *combo, OBSSource source,
  119. obs_properties_t *props,
  120. const char *prop_name, bool is_int)
  121. {
  122. std::string cur_id;
  123. OBSDataAutoRelease settings = obs_source_get_settings(source);
  124. if (is_int) {
  125. cur_id = std::to_string(obs_data_get_int(settings, prop_name));
  126. } else {
  127. cur_id = obs_data_get_string(settings, prop_name);
  128. }
  129. combo->blockSignals(true);
  130. obs_property_t *p = obs_properties_get(props, prop_name);
  131. int cur_idx = FillPropertyCombo(combo, p, cur_id, is_int);
  132. if (cur_idx == -1 || obs_property_list_item_disabled(p, cur_idx)) {
  133. if (cur_idx == -1) {
  134. combo->insertItem(
  135. 0,
  136. QTStr("Basic.Settings.Audio.UnknownAudioDevice"));
  137. cur_idx = 0;
  138. }
  139. SetComboItemEnabled(combo, cur_idx, false);
  140. }
  141. combo->setCurrentIndex(cur_idx);
  142. combo->blockSignals(false);
  143. }
  144. void ComboSelectToolbar::Init()
  145. {
  146. OBSSource source = GetSource();
  147. if (!source) {
  148. return;
  149. }
  150. UpdateSourceComboToolbarProperties(ui->device, source, props.get(),
  151. prop_name, is_int);
  152. }
  153. void UpdateSourceComboToolbarValue(QComboBox *combo, OBSSource source, int idx,
  154. const char *prop_name, bool is_int)
  155. {
  156. QString id = combo->itemData(idx).toString();
  157. OBSDataAutoRelease settings = obs_data_create();
  158. if (is_int) {
  159. obs_data_set_int(settings, prop_name, id.toInt());
  160. } else {
  161. obs_data_set_string(settings, prop_name, QT_TO_UTF8(id));
  162. }
  163. obs_source_update(source, settings);
  164. }
  165. void ComboSelectToolbar::on_device_currentIndexChanged(int idx)
  166. {
  167. OBSSource source = GetSource();
  168. if (idx == -1 || !source) {
  169. return;
  170. }
  171. SaveOldProperties(source);
  172. UpdateSourceComboToolbarValue(ui->device, source, idx, prop_name,
  173. is_int);
  174. SetUndoProperties(source);
  175. }
  176. AudioCaptureToolbar::AudioCaptureToolbar(QWidget *parent, OBSSource source)
  177. : ComboSelectToolbar(parent, source)
  178. {
  179. }
  180. void AudioCaptureToolbar::Init()
  181. {
  182. delete ui->activateButton;
  183. ui->activateButton = nullptr;
  184. obs_module_t *mod =
  185. get_os_module("win-wasapi", "mac-capture", "linux-pulseaudio");
  186. if (!mod)
  187. return;
  188. const char *device_str =
  189. get_os_text(mod, "Device", "CoreAudio.Device", "Device");
  190. ui->deviceLabel->setText(device_str);
  191. prop_name = "device_id";
  192. ComboSelectToolbar::Init();
  193. }
  194. WindowCaptureToolbar::WindowCaptureToolbar(QWidget *parent, OBSSource source)
  195. : ComboSelectToolbar(parent, source)
  196. {
  197. }
  198. void WindowCaptureToolbar::Init()
  199. {
  200. delete ui->activateButton;
  201. ui->activateButton = nullptr;
  202. obs_module_t *mod =
  203. get_os_module("win-capture", "mac-capture", "linux-capture");
  204. if (!mod)
  205. return;
  206. const char *device_str = get_os_text(mod, "WindowCapture.Window",
  207. "WindowUtils.Window", "Window");
  208. ui->deviceLabel->setText(device_str);
  209. #if !defined(_WIN32) && !defined(__APPLE__) //linux
  210. prop_name = "capture_window";
  211. #else
  212. prop_name = "window";
  213. #endif
  214. #ifdef __APPLE__
  215. is_int = true;
  216. #endif
  217. ComboSelectToolbar::Init();
  218. }
  219. ApplicationAudioCaptureToolbar::ApplicationAudioCaptureToolbar(QWidget *parent,
  220. OBSSource source)
  221. : ComboSelectToolbar(parent, source)
  222. {
  223. }
  224. void ApplicationAudioCaptureToolbar::Init()
  225. {
  226. delete ui->activateButton;
  227. ui->activateButton = nullptr;
  228. obs_module_t *mod = obs_get_module("win-wasapi");
  229. const char *device_str = obs_module_get_locale_text(mod, "Window");
  230. ui->deviceLabel->setText(device_str);
  231. prop_name = "window";
  232. ComboSelectToolbar::Init();
  233. }
  234. DisplayCaptureToolbar::DisplayCaptureToolbar(QWidget *parent, OBSSource source)
  235. : ComboSelectToolbar(parent, source)
  236. {
  237. }
  238. void DisplayCaptureToolbar::Init()
  239. {
  240. delete ui->activateButton;
  241. ui->activateButton = nullptr;
  242. obs_module_t *mod =
  243. get_os_module("win-capture", "mac-capture", "linux-capture");
  244. if (!mod)
  245. return;
  246. const char *device_str =
  247. get_os_text(mod, "Monitor", "DisplayCapture.Display", "Screen");
  248. ui->deviceLabel->setText(device_str);
  249. is_int = true;
  250. #ifdef _WIN32
  251. prop_name = "monitor";
  252. #elif __APPLE__
  253. prop_name = "display";
  254. #else
  255. prop_name = "screen";
  256. #endif
  257. ComboSelectToolbar::Init();
  258. }
  259. /* ========================================================================= */
  260. DeviceCaptureToolbar::DeviceCaptureToolbar(QWidget *parent, OBSSource source)
  261. : QWidget(parent),
  262. weakSource(OBSGetWeakRef(source)),
  263. ui(new Ui_DeviceSelectToolbar)
  264. {
  265. ui->setupUi(this);
  266. delete ui->deviceLabel;
  267. delete ui->device;
  268. ui->deviceLabel = nullptr;
  269. ui->device = nullptr;
  270. OBSDataAutoRelease settings = obs_source_get_settings(source);
  271. active = obs_data_get_bool(settings, "active");
  272. obs_module_t *mod = obs_get_module("win-dshow");
  273. if (!mod)
  274. return;
  275. activateText = obs_module_get_locale_text(mod, "Activate");
  276. deactivateText = obs_module_get_locale_text(mod, "Deactivate");
  277. ui->activateButton->setText(active ? deactivateText : activateText);
  278. }
  279. DeviceCaptureToolbar::~DeviceCaptureToolbar() {}
  280. void DeviceCaptureToolbar::on_activateButton_clicked()
  281. {
  282. OBSSource source = OBSGetStrongRef(weakSource);
  283. if (!source) {
  284. return;
  285. }
  286. OBSDataAutoRelease settings = obs_source_get_settings(source);
  287. bool now_active = obs_data_get_bool(settings, "active");
  288. bool desyncedSetting = now_active != active;
  289. active = !active;
  290. const char *text = active ? deactivateText : activateText;
  291. ui->activateButton->setText(text);
  292. if (desyncedSetting) {
  293. return;
  294. }
  295. calldata_t cd = {};
  296. calldata_set_bool(&cd, "active", active);
  297. proc_handler_t *ph = obs_source_get_proc_handler(source);
  298. proc_handler_call(ph, "activate", &cd);
  299. calldata_free(&cd);
  300. }
  301. /* ========================================================================= */
  302. GameCaptureToolbar::GameCaptureToolbar(QWidget *parent, OBSSource source)
  303. : SourceToolbar(parent, source), ui(new Ui_GameCaptureToolbar)
  304. {
  305. obs_property_t *p;
  306. int cur_idx;
  307. ui->setupUi(this);
  308. obs_module_t *mod = obs_get_module("win-capture");
  309. if (!mod)
  310. return;
  311. ui->modeLabel->setText(obs_module_get_locale_text(mod, "Mode"));
  312. ui->windowLabel->setText(
  313. obs_module_get_locale_text(mod, "WindowCapture.Window"));
  314. OBSDataAutoRelease settings = obs_source_get_settings(source);
  315. std::string cur_mode = obs_data_get_string(settings, "capture_mode");
  316. std::string cur_window = obs_data_get_string(settings, "window");
  317. ui->mode->blockSignals(true);
  318. p = obs_properties_get(props.get(), "capture_mode");
  319. cur_idx = FillPropertyCombo(ui->mode, p, cur_mode);
  320. ui->mode->setCurrentIndex(cur_idx);
  321. ui->mode->blockSignals(false);
  322. ui->window->blockSignals(true);
  323. p = obs_properties_get(props.get(), "window");
  324. cur_idx = FillPropertyCombo(ui->window, p, cur_window);
  325. ui->window->setCurrentIndex(cur_idx);
  326. ui->window->blockSignals(false);
  327. if (cur_idx != -1 && obs_property_list_item_disabled(p, cur_idx)) {
  328. SetComboItemEnabled(ui->window, cur_idx, false);
  329. }
  330. UpdateWindowVisibility();
  331. }
  332. GameCaptureToolbar::~GameCaptureToolbar() {}
  333. void GameCaptureToolbar::UpdateWindowVisibility()
  334. {
  335. QString mode = ui->mode->currentData().toString();
  336. bool is_window = (mode == "window");
  337. ui->windowLabel->setVisible(is_window);
  338. ui->window->setVisible(is_window);
  339. }
  340. void GameCaptureToolbar::on_mode_currentIndexChanged(int idx)
  341. {
  342. OBSSource source = GetSource();
  343. if (idx == -1 || !source) {
  344. return;
  345. }
  346. QString id = ui->mode->itemData(idx).toString();
  347. SaveOldProperties(source);
  348. OBSDataAutoRelease settings = obs_data_create();
  349. obs_data_set_string(settings, "capture_mode", QT_TO_UTF8(id));
  350. obs_source_update(source, settings);
  351. SetUndoProperties(source);
  352. UpdateWindowVisibility();
  353. }
  354. void GameCaptureToolbar::on_window_currentIndexChanged(int idx)
  355. {
  356. OBSSource source = GetSource();
  357. if (idx == -1 || !source) {
  358. return;
  359. }
  360. QString id = ui->window->itemData(idx).toString();
  361. SaveOldProperties(source);
  362. OBSDataAutoRelease settings = obs_data_create();
  363. obs_data_set_string(settings, "window", QT_TO_UTF8(id));
  364. obs_source_update(source, settings);
  365. SetUndoProperties(source);
  366. }
  367. /* ========================================================================= */
  368. ImageSourceToolbar::ImageSourceToolbar(QWidget *parent, OBSSource source)
  369. : SourceToolbar(parent, source), ui(new Ui_ImageSourceToolbar)
  370. {
  371. ui->setupUi(this);
  372. obs_module_t *mod = obs_get_module("image-source");
  373. ui->pathLabel->setText(obs_module_get_locale_text(mod, "File"));
  374. OBSDataAutoRelease settings = obs_source_get_settings(source);
  375. std::string file = obs_data_get_string(settings, "file");
  376. ui->path->setText(file.c_str());
  377. }
  378. ImageSourceToolbar::~ImageSourceToolbar() {}
  379. void ImageSourceToolbar::on_browse_clicked()
  380. {
  381. OBSSource source = GetSource();
  382. if (!source) {
  383. return;
  384. }
  385. obs_property_t *p = obs_properties_get(props.get(), "file");
  386. const char *desc = obs_property_description(p);
  387. const char *filter = obs_property_path_filter(p);
  388. const char *default_path = obs_property_path_default_path(p);
  389. QString path = OpenFile(this, desc, default_path, filter);
  390. if (path.isEmpty()) {
  391. return;
  392. }
  393. ui->path->setText(path);
  394. SaveOldProperties(source);
  395. OBSDataAutoRelease settings = obs_data_create();
  396. obs_data_set_string(settings, "file", QT_TO_UTF8(path));
  397. obs_source_update(source, settings);
  398. SetUndoProperties(source);
  399. }
  400. /* ========================================================================= */
  401. static inline QColor color_from_int(long long val)
  402. {
  403. return QColor(val & 0xff, (val >> 8) & 0xff, (val >> 16) & 0xff,
  404. (val >> 24) & 0xff);
  405. }
  406. static inline long long color_to_int(QColor color)
  407. {
  408. auto shift = [&](unsigned val, int shift) {
  409. return ((val & 0xff) << shift);
  410. };
  411. return shift(color.red(), 0) | shift(color.green(), 8) |
  412. shift(color.blue(), 16) | shift(color.alpha(), 24);
  413. }
  414. ColorSourceToolbar::ColorSourceToolbar(QWidget *parent, OBSSource source)
  415. : SourceToolbar(parent, source), ui(new Ui_ColorSourceToolbar)
  416. {
  417. ui->setupUi(this);
  418. OBSDataAutoRelease settings = obs_source_get_settings(source);
  419. unsigned int val = (unsigned int)obs_data_get_int(settings, "color");
  420. color = color_from_int(val);
  421. UpdateColor();
  422. }
  423. ColorSourceToolbar::~ColorSourceToolbar() {}
  424. void ColorSourceToolbar::UpdateColor()
  425. {
  426. QPalette palette = QPalette(color);
  427. ui->color->setFrameStyle(QFrame::Sunken | QFrame::Panel);
  428. ui->color->setText(color.name(QColor::HexRgb));
  429. ui->color->setPalette(palette);
  430. ui->color->setStyleSheet(
  431. QString("background-color :%1; color: %2;")
  432. .arg(palette.color(QPalette::Window)
  433. .name(QColor::HexRgb))
  434. .arg(palette.color(QPalette::WindowText)
  435. .name(QColor::HexRgb)));
  436. ui->color->setAutoFillBackground(true);
  437. ui->color->setAlignment(Qt::AlignCenter);
  438. }
  439. void ColorSourceToolbar::on_choose_clicked()
  440. {
  441. OBSSource source = GetSource();
  442. if (!source) {
  443. return;
  444. }
  445. obs_property_t *p = obs_properties_get(props.get(), "color");
  446. const char *desc = obs_property_description(p);
  447. QColorDialog::ColorDialogOptions options;
  448. options |= QColorDialog::ShowAlphaChannel;
  449. #ifndef _WIN32
  450. options |= QColorDialog::DontUseNativeDialog;
  451. #endif
  452. QColor newColor = QColorDialog::getColor(color, this, desc, options);
  453. if (!newColor.isValid()) {
  454. return;
  455. }
  456. color = newColor;
  457. UpdateColor();
  458. SaveOldProperties(source);
  459. OBSDataAutoRelease settings = obs_data_create();
  460. obs_data_set_int(settings, "color", color_to_int(color));
  461. obs_source_update(source, settings);
  462. SetUndoProperties(source);
  463. }
  464. /* ========================================================================= */
  465. extern void MakeQFont(obs_data_t *font_obj, QFont &font, bool limit = false);
  466. TextSourceToolbar::TextSourceToolbar(QWidget *parent, OBSSource source)
  467. : SourceToolbar(parent, source), ui(new Ui_TextSourceToolbar)
  468. {
  469. ui->setupUi(this);
  470. OBSDataAutoRelease settings = obs_source_get_settings(source);
  471. const char *id = obs_source_get_unversioned_id(source);
  472. bool ft2 = strcmp(id, "text_ft2_source") == 0;
  473. bool read_from_file = obs_data_get_bool(
  474. settings, ft2 ? "from_file" : "read_from_file");
  475. OBSDataAutoRelease font_obj = obs_data_get_obj(settings, "font");
  476. MakeQFont(font_obj, font);
  477. // Use "color1" if it's a freetype source and "color" elsewise
  478. unsigned int val = (unsigned int)obs_data_get_int(
  479. settings,
  480. (strncmp(obs_source_get_id(source), "text_ft2_source", 15) == 0)
  481. ? "color1"
  482. : "color");
  483. color = color_from_int(val);
  484. const char *text = obs_data_get_string(settings, "text");
  485. bool single_line = !read_from_file &&
  486. (!text || (strchr(text, '\n') == nullptr));
  487. ui->emptySpace->setVisible(!single_line);
  488. ui->text->setVisible(single_line);
  489. if (single_line)
  490. ui->text->setText(text);
  491. }
  492. TextSourceToolbar::~TextSourceToolbar() {}
  493. void TextSourceToolbar::on_selectFont_clicked()
  494. {
  495. OBSSource source = GetSource();
  496. if (!source) {
  497. return;
  498. }
  499. QFontDialog::FontDialogOptions options;
  500. uint32_t flags;
  501. bool success;
  502. #ifndef _WIN32
  503. options = QFontDialog::DontUseNativeDialog;
  504. #endif
  505. font = QFontDialog::getFont(&success, font, this, "Pick a Font",
  506. options);
  507. if (!success) {
  508. return;
  509. }
  510. OBSDataAutoRelease font_obj = obs_data_create();
  511. obs_data_set_string(font_obj, "face", QT_TO_UTF8(font.family()));
  512. obs_data_set_string(font_obj, "style", QT_TO_UTF8(font.styleName()));
  513. obs_data_set_int(font_obj, "size", font.pointSize());
  514. flags = font.bold() ? OBS_FONT_BOLD : 0;
  515. flags |= font.italic() ? OBS_FONT_ITALIC : 0;
  516. flags |= font.underline() ? OBS_FONT_UNDERLINE : 0;
  517. flags |= font.strikeOut() ? OBS_FONT_STRIKEOUT : 0;
  518. obs_data_set_int(font_obj, "flags", flags);
  519. SaveOldProperties(source);
  520. OBSDataAutoRelease settings = obs_data_create();
  521. obs_data_set_obj(settings, "font", font_obj);
  522. obs_source_update(source, settings);
  523. SetUndoProperties(source);
  524. }
  525. void TextSourceToolbar::on_selectColor_clicked()
  526. {
  527. OBSSource source = GetSource();
  528. if (!source) {
  529. return;
  530. }
  531. bool freetype =
  532. strncmp(obs_source_get_id(source), "text_ft2_source", 15) == 0;
  533. obs_property_t *p =
  534. obs_properties_get(props.get(), freetype ? "color1" : "color");
  535. const char *desc = obs_property_description(p);
  536. QColorDialog::ColorDialogOptions options;
  537. options |= QColorDialog::ShowAlphaChannel;
  538. #ifndef _WIN32
  539. options |= QColorDialog::DontUseNativeDialog;
  540. #endif
  541. QColor newColor = QColorDialog::getColor(color, this, desc, options);
  542. if (!newColor.isValid()) {
  543. return;
  544. }
  545. color = newColor;
  546. SaveOldProperties(source);
  547. OBSDataAutoRelease settings = obs_data_create();
  548. if (freetype) {
  549. obs_data_set_int(settings, "color1", color_to_int(color));
  550. obs_data_set_int(settings, "color2", color_to_int(color));
  551. } else {
  552. obs_data_set_int(settings, "color", color_to_int(color));
  553. }
  554. obs_source_update(source, settings);
  555. SetUndoProperties(source);
  556. }
  557. void TextSourceToolbar::on_text_textChanged()
  558. {
  559. OBSSource source = GetSource();
  560. if (!source) {
  561. return;
  562. }
  563. SaveOldProperties(source);
  564. OBSDataAutoRelease settings = obs_data_create();
  565. obs_data_set_string(settings, "text", QT_TO_UTF8(ui->text->text()));
  566. obs_source_update(source, settings);
  567. SetUndoProperties(source, true);
  568. }