window-basic-filters.cpp 35 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244
  1. /******************************************************************************
  2. Copyright (C) 2015 by Hugh Bailey <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. ******************************************************************************/
  14. #include "properties-view.hpp"
  15. #include "window-namedialog.hpp"
  16. #include "window-basic-main.hpp"
  17. #include "window-basic-filters.hpp"
  18. #include "display-helpers.hpp"
  19. #include "qt-wrappers.hpp"
  20. #include "visibility-item-widget.hpp"
  21. #include "item-widget-helpers.hpp"
  22. #include "obs-app.hpp"
  23. #include "undo-stack-obs.hpp"
  24. #include <QMessageBox>
  25. #include <QCloseEvent>
  26. #include <obs-data.h>
  27. #include <obs.h>
  28. #include <util/base.h>
  29. #include <vector>
  30. #include <string>
  31. #include <QMenu>
  32. #include <QVariant>
  33. using namespace std;
  34. Q_DECLARE_METATYPE(OBSSource);
  35. OBSBasicFilters::OBSBasicFilters(QWidget *parent, OBSSource source_)
  36. : QDialog(parent),
  37. ui(new Ui::OBSBasicFilters),
  38. source(source_),
  39. addSignal(obs_source_get_signal_handler(source), "filter_add",
  40. OBSBasicFilters::OBSSourceFilterAdded, this),
  41. removeSignal(obs_source_get_signal_handler(source), "filter_remove",
  42. OBSBasicFilters::OBSSourceFilterRemoved, this),
  43. reorderSignal(obs_source_get_signal_handler(source),
  44. "reorder_filters", OBSBasicFilters::OBSSourceReordered,
  45. this),
  46. removeSourceSignal(obs_source_get_signal_handler(source), "remove",
  47. OBSBasicFilters::SourceRemoved, this),
  48. renameSourceSignal(obs_source_get_signal_handler(source), "rename",
  49. OBSBasicFilters::SourceRenamed, this),
  50. noPreviewMargin(13)
  51. {
  52. main = reinterpret_cast<OBSBasic *>(parent);
  53. setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
  54. ui->setupUi(this);
  55. ui->asyncFilters->setItemDelegate(
  56. new VisibilityItemDelegate(ui->asyncFilters));
  57. ui->effectFilters->setItemDelegate(
  58. new VisibilityItemDelegate(ui->effectFilters));
  59. const char *name = obs_source_get_name(source);
  60. setWindowTitle(QTStr("Basic.Filters.Title").arg(QT_UTF8(name)));
  61. #ifndef QT_NO_SHORTCUT
  62. ui->actionRemoveFilter->setShortcut(
  63. QApplication::translate("OBSBasicFilters", "Del", nullptr));
  64. #endif // QT_NO_SHORTCUT
  65. addAction(ui->actionRemoveFilter);
  66. addAction(ui->actionMoveUp);
  67. addAction(ui->actionMoveDown);
  68. installEventFilter(CreateShortcutFilter());
  69. connect(ui->asyncFilters->itemDelegate(),
  70. SIGNAL(closeEditor(QWidget *,
  71. QAbstractItemDelegate::EndEditHint)),
  72. this,
  73. SLOT(AsyncFilterNameEdited(
  74. QWidget *, QAbstractItemDelegate::EndEditHint)));
  75. connect(ui->effectFilters->itemDelegate(),
  76. SIGNAL(closeEditor(QWidget *,
  77. QAbstractItemDelegate::EndEditHint)),
  78. this,
  79. SLOT(EffectFilterNameEdited(
  80. QWidget *, QAbstractItemDelegate::EndEditHint)));
  81. QPushButton *close = ui->buttonBox->button(QDialogButtonBox::Close);
  82. connect(close, SIGNAL(clicked()), this, SLOT(close()));
  83. close->setDefault(true);
  84. ui->buttonBox->button(QDialogButtonBox::Reset)
  85. ->setText(QTStr("Defaults"));
  86. connect(ui->buttonBox->button(QDialogButtonBox::Reset),
  87. SIGNAL(clicked()), this, SLOT(ResetFilters()));
  88. uint32_t caps = obs_source_get_output_flags(source);
  89. bool audio = (caps & OBS_SOURCE_AUDIO) != 0;
  90. bool audioOnly = (caps & OBS_SOURCE_VIDEO) == 0;
  91. bool async = (caps & OBS_SOURCE_ASYNC) != 0;
  92. if (!async && !audio) {
  93. ui->asyncWidget->setVisible(false);
  94. ui->separatorLine->setVisible(false);
  95. }
  96. if (audioOnly) {
  97. ui->effectWidget->setVisible(false);
  98. ui->separatorLine->setVisible(false);
  99. UpdateSplitter(false);
  100. }
  101. if (async && !audioOnly && ui->asyncFilters->count() == 0 &&
  102. ui->effectFilters->count() != 0) {
  103. ui->effectFilters->setFocus();
  104. }
  105. if (audioOnly || (audio && !async))
  106. ui->asyncLabel->setText(QTStr("Basic.Filters.AudioFilters"));
  107. if (async && audio && ui->asyncFilters->count() == 0) {
  108. UpdateSplitter(false);
  109. } else if (!audioOnly) {
  110. UpdateSplitter();
  111. }
  112. obs_source_inc_showing(source);
  113. auto addDrawCallback = [this]() {
  114. obs_display_add_draw_callback(ui->preview->GetDisplay(),
  115. OBSBasicFilters::DrawPreview,
  116. this);
  117. };
  118. enum obs_source_type type = obs_source_get_type(source);
  119. bool drawable_type = type == OBS_SOURCE_TYPE_INPUT ||
  120. type == OBS_SOURCE_TYPE_SCENE;
  121. if ((caps & OBS_SOURCE_VIDEO) != 0) {
  122. ui->rightLayout->setContentsMargins(0, 0, 0, 0);
  123. ui->preview->show();
  124. if (drawable_type)
  125. connect(ui->preview, &OBSQTDisplay::DisplayCreated,
  126. addDrawCallback);
  127. } else {
  128. ui->rightLayout->setContentsMargins(0, noPreviewMargin, 0, 0);
  129. ui->preview->hide();
  130. }
  131. QAction *renameAsync = new QAction(ui->asyncWidget);
  132. renameAsync->setShortcutContext(Qt::WidgetWithChildrenShortcut);
  133. connect(renameAsync, SIGNAL(triggered()), this,
  134. SLOT(RenameAsyncFilter()));
  135. ui->asyncWidget->addAction(renameAsync);
  136. QAction *renameEffect = new QAction(ui->effectWidget);
  137. renameEffect->setShortcutContext(Qt::WidgetWithChildrenShortcut);
  138. connect(renameEffect, SIGNAL(triggered()), this,
  139. SLOT(RenameEffectFilter()));
  140. ui->effectWidget->addAction(renameEffect);
  141. #ifdef __APPLE__
  142. renameAsync->setShortcut({Qt::Key_Return});
  143. renameEffect->setShortcut({Qt::Key_Return});
  144. #else
  145. renameAsync->setShortcut({Qt::Key_F2});
  146. renameEffect->setShortcut({Qt::Key_F2});
  147. #endif
  148. UpdateFilters();
  149. }
  150. OBSBasicFilters::~OBSBasicFilters()
  151. {
  152. obs_source_dec_showing(source);
  153. ClearListItems(ui->asyncFilters);
  154. ClearListItems(ui->effectFilters);
  155. }
  156. void OBSBasicFilters::Init()
  157. {
  158. show();
  159. }
  160. inline OBSSource OBSBasicFilters::GetFilter(int row, bool async)
  161. {
  162. if (row == -1)
  163. return OBSSource();
  164. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  165. QListWidgetItem *item = list->item(row);
  166. if (!item)
  167. return OBSSource();
  168. QVariant v = item->data(Qt::UserRole);
  169. return v.value<OBSSource>();
  170. }
  171. void FilterChangeUndoRedo(void *vp, obs_data_t *nd_old_settings,
  172. obs_data_t *new_settings)
  173. {
  174. obs_source_t *source = reinterpret_cast<obs_source_t *>(vp);
  175. obs_source_t *parent = obs_filter_get_parent(source);
  176. const char *source_name = obs_source_get_name(source);
  177. OBSBasic *main = OBSBasic::Get();
  178. OBSDataAutoRelease redo_wrapper = obs_data_create();
  179. obs_data_set_string(redo_wrapper, "name", source_name);
  180. obs_data_set_string(redo_wrapper, "settings",
  181. obs_data_get_json(new_settings));
  182. obs_data_set_string(redo_wrapper, "parent",
  183. obs_source_get_name(parent));
  184. OBSDataAutoRelease filter_settings = obs_source_get_settings(source);
  185. OBSDataAutoRelease undo_wrapper = obs_data_create();
  186. obs_data_set_string(undo_wrapper, "name", source_name);
  187. obs_data_set_string(undo_wrapper, "settings",
  188. obs_data_get_json(nd_old_settings));
  189. obs_data_set_string(undo_wrapper, "parent",
  190. obs_source_get_name(parent));
  191. auto undo_redo = [](const std::string &data) {
  192. OBSDataAutoRelease dat =
  193. obs_data_create_from_json(data.c_str());
  194. OBSSourceAutoRelease parent_source = obs_get_source_by_name(
  195. obs_data_get_string(dat, "parent"));
  196. const char *filter_name = obs_data_get_string(dat, "name");
  197. OBSSourceAutoRelease filter = obs_source_get_filter_by_name(
  198. parent_source, filter_name);
  199. OBSDataAutoRelease new_settings = obs_data_create_from_json(
  200. obs_data_get_string(dat, "settings"));
  201. OBSDataAutoRelease current_settings =
  202. obs_source_get_settings(filter);
  203. obs_data_clear(current_settings);
  204. obs_source_update(filter, new_settings);
  205. obs_source_update_properties(filter);
  206. };
  207. main->undo_s.enable();
  208. std::string name = std::string(obs_source_get_name(source));
  209. std::string undo_data = obs_data_get_json(undo_wrapper);
  210. std::string redo_data = obs_data_get_json(redo_wrapper);
  211. main->undo_s.add_action(QTStr("Undo.Filters").arg(name.c_str()),
  212. undo_redo, undo_redo, undo_data, redo_data);
  213. obs_source_update(source, new_settings);
  214. }
  215. void OBSBasicFilters::UpdatePropertiesView(int row, bool async)
  216. {
  217. OBSSource filter = GetFilter(row, async);
  218. if (filter && view && view->IsObject(filter)) {
  219. /* do not recreate properties view if already using a view
  220. * with the same object */
  221. return;
  222. }
  223. ui->propertiesFrame->setVisible(false);
  224. if (view) {
  225. updatePropertiesSignal.Disconnect();
  226. ui->propertiesFrame->setVisible(false);
  227. view->hide();
  228. view->deleteLater();
  229. view = nullptr;
  230. }
  231. if (!filter)
  232. return;
  233. OBSDataAutoRelease settings = obs_source_get_settings(filter);
  234. auto disabled_undo = [](void *vp, obs_data_t *settings) {
  235. OBSBasic *main =
  236. reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
  237. main->undo_s.disable();
  238. obs_source_t *source = reinterpret_cast<obs_source_t *>(vp);
  239. obs_source_update(source, settings);
  240. };
  241. view = new OBSPropertiesView(
  242. settings.Get(), filter,
  243. (PropertiesReloadCallback)obs_source_properties,
  244. (PropertiesUpdateCallback)FilterChangeUndoRedo,
  245. (PropertiesVisualUpdateCb)disabled_undo);
  246. updatePropertiesSignal.Connect(obs_source_get_signal_handler(filter),
  247. "update_properties",
  248. OBSBasicFilters::UpdateProperties, this);
  249. view->setMinimumHeight(150);
  250. UpdateSplitter();
  251. ui->propertiesLayout->addWidget(view);
  252. view->show();
  253. }
  254. void OBSBasicFilters::UpdateProperties(void *data, calldata_t *)
  255. {
  256. QMetaObject::invokeMethod(static_cast<OBSBasicFilters *>(data)->view,
  257. "ReloadProperties");
  258. }
  259. void OBSBasicFilters::AddFilter(OBSSource filter, bool focus)
  260. {
  261. uint32_t flags = obs_source_get_output_flags(filter);
  262. bool async = (flags & OBS_SOURCE_ASYNC) != 0;
  263. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  264. QListWidgetItem *item = new QListWidgetItem();
  265. Qt::ItemFlags itemFlags = item->flags();
  266. item->setFlags(itemFlags | Qt::ItemIsEditable);
  267. item->setData(Qt::UserRole, QVariant::fromValue(filter));
  268. list->addItem(item);
  269. if (focus)
  270. list->setCurrentItem(item);
  271. SetupVisibilityItem(list, item, filter);
  272. }
  273. void OBSBasicFilters::RemoveFilter(OBSSource filter)
  274. {
  275. uint32_t flags = obs_source_get_output_flags(filter);
  276. bool async = (flags & OBS_SOURCE_ASYNC) != 0;
  277. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  278. for (int i = 0; i < list->count(); i++) {
  279. QListWidgetItem *item = list->item(i);
  280. QVariant v = item->data(Qt::UserRole);
  281. OBSSource curFilter = v.value<OBSSource>();
  282. if (filter == curFilter) {
  283. DeleteListItem(list, item);
  284. break;
  285. }
  286. }
  287. const char *filterName = obs_source_get_name(filter);
  288. const char *sourceName = obs_source_get_name(source);
  289. if (!sourceName || !filterName)
  290. return;
  291. const char *filterId = obs_source_get_id(filter);
  292. blog(LOG_INFO, "User removed filter '%s' (%s) from source '%s'",
  293. filterName, filterId, sourceName);
  294. main->SaveProject();
  295. }
  296. struct FilterOrderInfo {
  297. int asyncIdx = 0;
  298. int effectIdx = 0;
  299. OBSBasicFilters *window;
  300. inline FilterOrderInfo(OBSBasicFilters *window_) : window(window_) {}
  301. };
  302. void OBSBasicFilters::ReorderFilter(QListWidget *list, obs_source_t *filter,
  303. size_t idx)
  304. {
  305. int count = list->count();
  306. for (int i = 0; i < count; i++) {
  307. QListWidgetItem *listItem = list->item(i);
  308. QVariant v = listItem->data(Qt::UserRole);
  309. OBSSource filterItem = v.value<OBSSource>();
  310. if (filterItem == filter) {
  311. if ((int)idx != i) {
  312. bool sel = (list->currentRow() == i);
  313. listItem = TakeListItem(list, i);
  314. if (listItem) {
  315. list->insertItem((int)idx, listItem);
  316. SetupVisibilityItem(list, listItem,
  317. filterItem);
  318. if (sel)
  319. list->setCurrentRow((int)idx);
  320. }
  321. }
  322. break;
  323. }
  324. }
  325. }
  326. void OBSBasicFilters::ReorderFilters()
  327. {
  328. FilterOrderInfo info(this);
  329. obs_source_enum_filters(
  330. source,
  331. [](obs_source_t *, obs_source_t *filter, void *p) {
  332. FilterOrderInfo *info =
  333. reinterpret_cast<FilterOrderInfo *>(p);
  334. uint32_t flags;
  335. bool async;
  336. flags = obs_source_get_output_flags(filter);
  337. async = (flags & OBS_SOURCE_ASYNC) != 0;
  338. if (async) {
  339. info->window->ReorderFilter(
  340. info->window->ui->asyncFilters, filter,
  341. info->asyncIdx++);
  342. } else {
  343. info->window->ReorderFilter(
  344. info->window->ui->effectFilters, filter,
  345. info->effectIdx++);
  346. }
  347. },
  348. &info);
  349. }
  350. void OBSBasicFilters::UpdateFilters()
  351. {
  352. if (!source)
  353. return;
  354. ClearListItems(ui->effectFilters);
  355. ClearListItems(ui->asyncFilters);
  356. obs_source_enum_filters(
  357. source,
  358. [](obs_source_t *, obs_source_t *filter, void *p) {
  359. OBSBasicFilters *window =
  360. reinterpret_cast<OBSBasicFilters *>(p);
  361. window->AddFilter(filter, false);
  362. },
  363. this);
  364. if (ui->asyncFilters->count() > 0) {
  365. ui->asyncFilters->setCurrentItem(ui->asyncFilters->item(0));
  366. } else if (ui->effectFilters->count() > 0) {
  367. ui->effectFilters->setCurrentItem(ui->effectFilters->item(0));
  368. }
  369. main->SaveProject();
  370. }
  371. void OBSBasicFilters::UpdateSplitter()
  372. {
  373. bool show_splitter_frame =
  374. ui->asyncFilters->count() + ui->effectFilters->count() > 0;
  375. UpdateSplitter(show_splitter_frame);
  376. }
  377. void OBSBasicFilters::UpdateSplitter(bool show_splitter_frame)
  378. {
  379. bool show_splitter_handle = show_splitter_frame;
  380. uint32_t caps = obs_source_get_output_flags(source);
  381. if ((caps & OBS_SOURCE_VIDEO) == 0)
  382. show_splitter_handle = false;
  383. for (int i = 0; i < ui->rightLayout->count(); i++) {
  384. QSplitterHandle *hndl = ui->rightLayout->handle(i);
  385. hndl->setEnabled(show_splitter_handle);
  386. }
  387. ui->propertiesFrame->setVisible(show_splitter_frame);
  388. }
  389. static bool filter_compatible(bool async, uint32_t sourceFlags,
  390. uint32_t filterFlags)
  391. {
  392. bool filterVideo = (filterFlags & OBS_SOURCE_VIDEO) != 0;
  393. bool filterAsync = (filterFlags & OBS_SOURCE_ASYNC) != 0;
  394. bool filterAudio = (filterFlags & OBS_SOURCE_AUDIO) != 0;
  395. bool audio = (sourceFlags & OBS_SOURCE_AUDIO) != 0;
  396. bool audioOnly = (sourceFlags & OBS_SOURCE_VIDEO) == 0;
  397. bool asyncSource = (sourceFlags & OBS_SOURCE_ASYNC) != 0;
  398. if (async && ((audioOnly && filterVideo) || (!audio && !asyncSource) ||
  399. (filterAudio && !audio)))
  400. return false;
  401. return (async && (filterAudio || filterAsync)) ||
  402. (!async && !filterAudio && !filterAsync);
  403. }
  404. QMenu *OBSBasicFilters::CreateAddFilterPopupMenu(bool async)
  405. {
  406. uint32_t sourceFlags = obs_source_get_output_flags(source);
  407. const char *type_str;
  408. bool foundValues = false;
  409. size_t idx = 0;
  410. struct FilterInfo {
  411. string type;
  412. string name;
  413. inline FilterInfo(const char *type_, const char *name_)
  414. : type(type_), name(name_)
  415. {
  416. }
  417. };
  418. vector<FilterInfo> types;
  419. while (obs_enum_filter_types(idx++, &type_str)) {
  420. const char *name = obs_source_get_display_name(type_str);
  421. uint32_t caps = obs_get_source_output_flags(type_str);
  422. if ((caps & OBS_SOURCE_DEPRECATED) != 0)
  423. continue;
  424. if ((caps & OBS_SOURCE_CAP_DISABLED) != 0)
  425. continue;
  426. if ((caps & OBS_SOURCE_CAP_OBSOLETE) != 0)
  427. continue;
  428. auto it = types.begin();
  429. for (; it != types.end(); ++it) {
  430. if (it->name >= name)
  431. break;
  432. }
  433. types.emplace(it, type_str, name);
  434. }
  435. QMenu *popup = new QMenu(QTStr("Add"), this);
  436. for (FilterInfo &type : types) {
  437. uint32_t filterFlags =
  438. obs_get_source_output_flags(type.type.c_str());
  439. if (!filter_compatible(async, sourceFlags, filterFlags))
  440. continue;
  441. QAction *popupItem =
  442. new QAction(QT_UTF8(type.name.c_str()), this);
  443. popupItem->setData(QT_UTF8(type.type.c_str()));
  444. connect(popupItem, SIGNAL(triggered(bool)), this,
  445. SLOT(AddFilterFromAction()));
  446. popup->addAction(popupItem);
  447. foundValues = true;
  448. }
  449. if (!foundValues) {
  450. delete popup;
  451. popup = nullptr;
  452. }
  453. return popup;
  454. }
  455. void OBSBasicFilters::AddNewFilter(const char *id)
  456. {
  457. if (id && *id) {
  458. OBSSourceAutoRelease existing_filter;
  459. string name = obs_source_get_display_name(id);
  460. QString placeholder = QString::fromStdString(name);
  461. QString text{placeholder};
  462. int i = 2;
  463. while ((existing_filter = obs_source_get_filter_by_name(
  464. source, QT_TO_UTF8(text)))) {
  465. text = QString("%1 %2").arg(placeholder).arg(i++);
  466. }
  467. bool success = NameDialog::AskForName(
  468. this, QTStr("Basic.Filters.AddFilter.Title"),
  469. QTStr("Basic.Filters.AddFilter.Text"), name, text);
  470. if (!success)
  471. return;
  472. if (name.empty()) {
  473. OBSMessageBox::warning(this,
  474. QTStr("NoNameEntered.Title"),
  475. QTStr("NoNameEntered.Text"));
  476. AddNewFilter(id);
  477. return;
  478. }
  479. existing_filter =
  480. obs_source_get_filter_by_name(source, name.c_str());
  481. if (existing_filter) {
  482. OBSMessageBox::warning(this, QTStr("NameExists.Title"),
  483. QTStr("NameExists.Text"));
  484. AddNewFilter(id);
  485. return;
  486. }
  487. OBSDataAutoRelease wrapper = obs_data_create();
  488. obs_data_set_string(wrapper, "sname",
  489. obs_source_get_name(source));
  490. obs_data_set_string(wrapper, "fname", name.c_str());
  491. std::string scene_name = obs_source_get_name(
  492. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  493. ->GetCurrentSceneSource());
  494. auto undo = [scene_name](const std::string &data) {
  495. obs_source_t *ssource =
  496. obs_get_source_by_name(scene_name.c_str());
  497. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  498. ->SetCurrentScene(ssource, true);
  499. obs_source_release(ssource);
  500. obs_data_t *dat =
  501. obs_data_create_from_json(data.c_str());
  502. obs_source_t *source = obs_get_source_by_name(
  503. obs_data_get_string(dat, "sname"));
  504. obs_source_t *filter = obs_source_get_filter_by_name(
  505. source, obs_data_get_string(dat, "fname"));
  506. obs_source_filter_remove(source, filter);
  507. obs_data_release(dat);
  508. obs_source_release(source);
  509. obs_source_release(filter);
  510. };
  511. OBSDataAutoRelease rwrapper = obs_data_create();
  512. obs_data_set_string(rwrapper, "sname",
  513. obs_source_get_name(source));
  514. auto redo = [scene_name, id = std::string(id),
  515. name](const std::string &data) {
  516. OBSSourceAutoRelease ssource =
  517. obs_get_source_by_name(scene_name.c_str());
  518. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  519. ->SetCurrentScene(ssource.Get(), true);
  520. OBSDataAutoRelease dat =
  521. obs_data_create_from_json(data.c_str());
  522. OBSSourceAutoRelease source = obs_get_source_by_name(
  523. obs_data_get_string(dat, "sname"));
  524. OBSSourceAutoRelease filter = obs_source_create(
  525. id.c_str(), name.c_str(), nullptr, nullptr);
  526. if (filter) {
  527. obs_source_filter_add(source, filter);
  528. }
  529. };
  530. std::string undo_data(obs_data_get_json(wrapper));
  531. std::string redo_data(obs_data_get_json(rwrapper));
  532. main->undo_s.add_action(QTStr("Undo.Add").arg(name.c_str()),
  533. undo, redo, undo_data, redo_data);
  534. OBSSourceAutoRelease filter =
  535. obs_source_create(id, name.c_str(), nullptr, nullptr);
  536. if (filter) {
  537. const char *sourceName = obs_source_get_name(source);
  538. blog(LOG_INFO,
  539. "User added filter '%s' (%s) "
  540. "to source '%s'",
  541. name.c_str(), id, sourceName);
  542. obs_source_filter_add(source, filter);
  543. }
  544. }
  545. }
  546. void OBSBasicFilters::AddFilterFromAction()
  547. {
  548. QAction *action = qobject_cast<QAction *>(sender());
  549. if (!action)
  550. return;
  551. AddNewFilter(QT_TO_UTF8(action->data().toString()));
  552. }
  553. void OBSBasicFilters::closeEvent(QCloseEvent *event)
  554. {
  555. QDialog::closeEvent(event);
  556. if (!event->isAccepted())
  557. return;
  558. obs_display_remove_draw_callback(ui->preview->GetDisplay(),
  559. OBSBasicFilters::DrawPreview, this);
  560. main->SaveProject();
  561. }
  562. /* OBS Signals */
  563. void OBSBasicFilters::OBSSourceFilterAdded(void *param, calldata_t *data)
  564. {
  565. OBSBasicFilters *window = reinterpret_cast<OBSBasicFilters *>(param);
  566. obs_source_t *filter = (obs_source_t *)calldata_ptr(data, "filter");
  567. QMetaObject::invokeMethod(window, "AddFilter",
  568. Q_ARG(OBSSource, OBSSource(filter)));
  569. }
  570. void OBSBasicFilters::OBSSourceFilterRemoved(void *param, calldata_t *data)
  571. {
  572. OBSBasicFilters *window = reinterpret_cast<OBSBasicFilters *>(param);
  573. obs_source_t *filter = (obs_source_t *)calldata_ptr(data, "filter");
  574. QMetaObject::invokeMethod(window, "RemoveFilter",
  575. Q_ARG(OBSSource, OBSSource(filter)));
  576. }
  577. void OBSBasicFilters::OBSSourceReordered(void *param, calldata_t *data)
  578. {
  579. QMetaObject::invokeMethod(reinterpret_cast<OBSBasicFilters *>(param),
  580. "ReorderFilters");
  581. UNUSED_PARAMETER(data);
  582. }
  583. void OBSBasicFilters::SourceRemoved(void *param, calldata_t *data)
  584. {
  585. UNUSED_PARAMETER(data);
  586. QMetaObject::invokeMethod(static_cast<OBSBasicFilters *>(param),
  587. "close");
  588. }
  589. void OBSBasicFilters::SourceRenamed(void *param, calldata_t *data)
  590. {
  591. const char *name = calldata_string(data, "new_name");
  592. QString title = QTStr("Basic.Filters.Title").arg(QT_UTF8(name));
  593. QMetaObject::invokeMethod(static_cast<OBSBasicFilters *>(param),
  594. "setWindowTitle", Q_ARG(QString, title));
  595. }
  596. void OBSBasicFilters::DrawPreview(void *data, uint32_t cx, uint32_t cy)
  597. {
  598. OBSBasicFilters *window = static_cast<OBSBasicFilters *>(data);
  599. if (!window->source)
  600. return;
  601. uint32_t sourceCX = max(obs_source_get_width(window->source), 1u);
  602. uint32_t sourceCY = max(obs_source_get_height(window->source), 1u);
  603. int x, y;
  604. int newCX, newCY;
  605. float scale;
  606. GetScaleAndCenterPos(sourceCX, sourceCY, cx, cy, x, y, scale);
  607. newCX = int(scale * float(sourceCX));
  608. newCY = int(scale * float(sourceCY));
  609. gs_viewport_push();
  610. gs_projection_push();
  611. const bool previous = gs_set_linear_srgb(true);
  612. gs_ortho(0.0f, float(sourceCX), 0.0f, float(sourceCY), -100.0f, 100.0f);
  613. gs_set_viewport(x, y, newCX, newCY);
  614. obs_source_video_render(window->source);
  615. gs_set_linear_srgb(previous);
  616. gs_projection_pop();
  617. gs_viewport_pop();
  618. }
  619. /* Qt Slots */
  620. static bool QueryRemove(QWidget *parent, obs_source_t *source)
  621. {
  622. const char *name = obs_source_get_name(source);
  623. QString text = QTStr("ConfirmRemove.Text");
  624. text.replace("$1", QT_UTF8(name));
  625. QMessageBox remove_source(parent);
  626. remove_source.setText(text);
  627. QAbstractButton *Yes =
  628. remove_source.addButton(QTStr("Yes"), QMessageBox::YesRole);
  629. remove_source.addButton(QTStr("No"), QMessageBox::NoRole);
  630. remove_source.setIcon(QMessageBox::Question);
  631. remove_source.setWindowTitle(QTStr("ConfirmRemove.Title"));
  632. remove_source.exec();
  633. return Yes == remove_source.clickedButton();
  634. }
  635. void OBSBasicFilters::on_addAsyncFilter_clicked()
  636. {
  637. ui->asyncFilters->setFocus();
  638. QScopedPointer<QMenu> popup(CreateAddFilterPopupMenu(true));
  639. if (popup)
  640. popup->exec(QCursor::pos());
  641. }
  642. void OBSBasicFilters::on_removeAsyncFilter_clicked()
  643. {
  644. OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
  645. if (filter) {
  646. if (QueryRemove(this, filter))
  647. delete_filter(filter);
  648. }
  649. }
  650. void OBSBasicFilters::on_moveAsyncFilterUp_clicked()
  651. {
  652. OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
  653. if (filter)
  654. obs_source_filter_set_order(source, filter, OBS_ORDER_MOVE_UP);
  655. }
  656. void OBSBasicFilters::on_moveAsyncFilterDown_clicked()
  657. {
  658. OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
  659. if (filter)
  660. obs_source_filter_set_order(source, filter,
  661. OBS_ORDER_MOVE_DOWN);
  662. }
  663. void OBSBasicFilters::on_asyncFilters_GotFocus()
  664. {
  665. UpdatePropertiesView(ui->asyncFilters->currentRow(), true);
  666. isAsync = true;
  667. }
  668. void OBSBasicFilters::on_asyncFilters_currentRowChanged(int row)
  669. {
  670. UpdatePropertiesView(row, true);
  671. }
  672. void OBSBasicFilters::on_addEffectFilter_clicked()
  673. {
  674. ui->effectFilters->setFocus();
  675. QScopedPointer<QMenu> popup(CreateAddFilterPopupMenu(false));
  676. if (popup)
  677. popup->exec(QCursor::pos());
  678. }
  679. void OBSBasicFilters::on_removeEffectFilter_clicked()
  680. {
  681. OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
  682. if (filter) {
  683. if (QueryRemove(this, filter)) {
  684. delete_filter(filter);
  685. }
  686. }
  687. }
  688. void OBSBasicFilters::on_moveEffectFilterUp_clicked()
  689. {
  690. OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
  691. if (filter)
  692. obs_source_filter_set_order(source, filter, OBS_ORDER_MOVE_UP);
  693. }
  694. void OBSBasicFilters::on_moveEffectFilterDown_clicked()
  695. {
  696. OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
  697. if (filter)
  698. obs_source_filter_set_order(source, filter,
  699. OBS_ORDER_MOVE_DOWN);
  700. }
  701. void OBSBasicFilters::on_effectFilters_GotFocus()
  702. {
  703. UpdatePropertiesView(ui->effectFilters->currentRow(), false);
  704. isAsync = false;
  705. }
  706. void OBSBasicFilters::on_effectFilters_currentRowChanged(int row)
  707. {
  708. UpdatePropertiesView(row, false);
  709. }
  710. void OBSBasicFilters::on_actionRemoveFilter_triggered()
  711. {
  712. if (ui->asyncFilters->hasFocus())
  713. on_removeAsyncFilter_clicked();
  714. else if (ui->effectFilters->hasFocus())
  715. on_removeEffectFilter_clicked();
  716. }
  717. void OBSBasicFilters::on_actionMoveUp_triggered()
  718. {
  719. if (ui->asyncFilters->hasFocus())
  720. on_moveAsyncFilterUp_clicked();
  721. else if (ui->effectFilters->hasFocus())
  722. on_moveEffectFilterUp_clicked();
  723. }
  724. void OBSBasicFilters::on_actionMoveDown_triggered()
  725. {
  726. if (ui->asyncFilters->hasFocus())
  727. on_moveAsyncFilterDown_clicked();
  728. else if (ui->effectFilters->hasFocus())
  729. on_moveEffectFilterDown_clicked();
  730. }
  731. void OBSBasicFilters::CustomContextMenu(const QPoint &pos, bool async)
  732. {
  733. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  734. QListWidgetItem *item = list->itemAt(pos);
  735. QMenu popup(window());
  736. QPointer<QMenu> addMenu = CreateAddFilterPopupMenu(async);
  737. if (addMenu)
  738. popup.addMenu(addMenu);
  739. if (item) {
  740. const char *dulpicateSlot =
  741. async ? SLOT(DuplicateAsyncFilter())
  742. : SLOT(DuplicateEffectFilter());
  743. const char *renameSlot = async ? SLOT(RenameAsyncFilter())
  744. : SLOT(RenameEffectFilter());
  745. const char *removeSlot =
  746. async ? SLOT(on_removeAsyncFilter_clicked())
  747. : SLOT(on_removeEffectFilter_clicked());
  748. popup.addSeparator();
  749. popup.addAction(QTStr("Duplicate"), this, dulpicateSlot);
  750. popup.addSeparator();
  751. popup.addAction(QTStr("Rename"), this, renameSlot);
  752. popup.addAction(QTStr("Remove"), this, removeSlot);
  753. popup.addSeparator();
  754. QAction *copyAction = new QAction(QTStr("Copy"));
  755. connect(copyAction, SIGNAL(triggered()), this,
  756. SLOT(CopyFilter()));
  757. copyAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_C));
  758. ui->effectWidget->addAction(copyAction);
  759. ui->asyncWidget->addAction(copyAction);
  760. popup.addAction(copyAction);
  761. }
  762. QAction *pasteAction = new QAction(QTStr("Paste"));
  763. pasteAction->setEnabled(main->copyFilter);
  764. connect(pasteAction, SIGNAL(triggered()), this, SLOT(PasteFilter()));
  765. pasteAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_V));
  766. ui->effectWidget->addAction(pasteAction);
  767. ui->asyncWidget->addAction(pasteAction);
  768. popup.addAction(pasteAction);
  769. popup.exec(QCursor::pos());
  770. }
  771. void OBSBasicFilters::EditItem(QListWidgetItem *item, bool async)
  772. {
  773. if (editActive)
  774. return;
  775. Qt::ItemFlags flags = item->flags();
  776. OBSSource filter = item->data(Qt::UserRole).value<OBSSource>();
  777. const char *name = obs_source_get_name(filter);
  778. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  779. item->setText(QT_UTF8(name));
  780. item->setFlags(flags | Qt::ItemIsEditable);
  781. list->removeItemWidget(item);
  782. list->editItem(item);
  783. item->setFlags(flags);
  784. editActive = true;
  785. }
  786. void OBSBasicFilters::DuplicateItem(QListWidgetItem *item)
  787. {
  788. OBSSource filter = item->data(Qt::UserRole).value<OBSSource>();
  789. string name = obs_source_get_name(filter);
  790. OBSSourceAutoRelease existing_filter;
  791. QString placeholder = QString::fromStdString(name);
  792. QString text{placeholder};
  793. int i = 2;
  794. while ((existing_filter = obs_source_get_filter_by_name(
  795. source, QT_TO_UTF8(text)))) {
  796. text = QString("%1 %2").arg(placeholder).arg(i++);
  797. }
  798. bool success = NameDialog::AskForName(
  799. this, QTStr("Basic.Filters.AddFilter.Title"),
  800. QTStr("Basic.Filters.AddFilter.Text"), name, text);
  801. if (!success)
  802. return;
  803. if (name.empty()) {
  804. OBSMessageBox::warning(this, QTStr("NoNameEntered.Title"),
  805. QTStr("NoNameEntered.Text"));
  806. DuplicateItem(item);
  807. return;
  808. }
  809. existing_filter = obs_source_get_filter_by_name(source, name.c_str());
  810. if (existing_filter) {
  811. OBSMessageBox::warning(this, QTStr("NameExists.Title"),
  812. QTStr("NameExists.Text"));
  813. DuplicateItem(item);
  814. return;
  815. }
  816. bool enabled = obs_source_enabled(filter);
  817. OBSSourceAutoRelease new_filter =
  818. obs_source_duplicate(filter, name.c_str(), false);
  819. if (new_filter) {
  820. const char *sourceName = obs_source_get_name(source);
  821. const char *id = obs_source_get_id(new_filter);
  822. blog(LOG_INFO,
  823. "User duplicated filter '%s' (%s) from '%s' "
  824. "to source '%s'",
  825. name.c_str(), id, name.c_str(), sourceName);
  826. obs_source_set_enabled(new_filter, enabled);
  827. obs_source_filter_add(source, new_filter);
  828. }
  829. }
  830. void OBSBasicFilters::on_asyncFilters_customContextMenuRequested(
  831. const QPoint &pos)
  832. {
  833. CustomContextMenu(pos, true);
  834. }
  835. void OBSBasicFilters::on_effectFilters_customContextMenuRequested(
  836. const QPoint &pos)
  837. {
  838. CustomContextMenu(pos, false);
  839. }
  840. void OBSBasicFilters::RenameAsyncFilter()
  841. {
  842. EditItem(ui->asyncFilters->currentItem(), true);
  843. }
  844. void OBSBasicFilters::RenameEffectFilter()
  845. {
  846. EditItem(ui->effectFilters->currentItem(), false);
  847. }
  848. void OBSBasicFilters::DuplicateAsyncFilter()
  849. {
  850. DuplicateItem(ui->asyncFilters->currentItem());
  851. }
  852. void OBSBasicFilters::DuplicateEffectFilter()
  853. {
  854. DuplicateItem(ui->effectFilters->currentItem());
  855. }
  856. void OBSBasicFilters::FilterNameEdited(QWidget *editor, QListWidget *list)
  857. {
  858. QListWidgetItem *listItem = list->currentItem();
  859. OBSSource filter = listItem->data(Qt::UserRole).value<OBSSource>();
  860. QLineEdit *edit = qobject_cast<QLineEdit *>(editor);
  861. string name = QT_TO_UTF8(edit->text().trimmed());
  862. const char *prevName = obs_source_get_name(filter);
  863. bool sameName = (name == prevName);
  864. OBSSourceAutoRelease foundFilter = nullptr;
  865. if (!sameName)
  866. foundFilter =
  867. obs_source_get_filter_by_name(source, name.c_str());
  868. if (foundFilter || name.empty() || sameName) {
  869. listItem->setText(QT_UTF8(prevName));
  870. if (foundFilter) {
  871. OBSMessageBox::information(window(),
  872. QTStr("NameExists.Title"),
  873. QTStr("NameExists.Text"));
  874. } else if (name.empty()) {
  875. OBSMessageBox::information(window(),
  876. QTStr("NoNameEntered.Title"),
  877. QTStr("NoNameEntered.Text"));
  878. }
  879. } else {
  880. const char *sourceName = obs_source_get_name(source);
  881. blog(LOG_INFO,
  882. "User renamed filter '%s' on source '%s' to '%s'",
  883. prevName, sourceName, name.c_str());
  884. listItem->setText(QT_UTF8(name.c_str()));
  885. obs_source_set_name(filter, name.c_str());
  886. std::string scene_name = obs_source_get_name(
  887. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  888. ->GetCurrentSceneSource());
  889. auto undo = [scene_name, prev = std::string(prevName),
  890. name](const std::string &data) {
  891. OBSSourceAutoRelease ssource =
  892. obs_get_source_by_name(scene_name.c_str());
  893. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  894. ->SetCurrentScene(ssource.Get(), true);
  895. OBSSourceAutoRelease source =
  896. obs_get_source_by_name(data.c_str());
  897. OBSSourceAutoRelease filter =
  898. obs_source_get_filter_by_name(source,
  899. name.c_str());
  900. obs_source_set_name(filter, prev.c_str());
  901. };
  902. auto redo = [scene_name, prev = std::string(prevName),
  903. name](const std::string &data) {
  904. OBSSourceAutoRelease ssource =
  905. obs_get_source_by_name(scene_name.c_str());
  906. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  907. ->SetCurrentScene(ssource.Get(), true);
  908. OBSSourceAutoRelease source =
  909. obs_get_source_by_name(data.c_str());
  910. OBSSourceAutoRelease filter =
  911. obs_source_get_filter_by_name(source,
  912. prev.c_str());
  913. obs_source_set_name(filter, name.c_str());
  914. };
  915. std::string undo_data(sourceName);
  916. std::string redo_data(sourceName);
  917. main->undo_s.add_action(QTStr("Undo.Rename").arg(name.c_str()),
  918. undo, redo, undo_data, redo_data);
  919. }
  920. listItem->setText(QString());
  921. SetupVisibilityItem(list, listItem, filter);
  922. editActive = false;
  923. }
  924. void OBSBasicFilters::AsyncFilterNameEdited(
  925. QWidget *editor, QAbstractItemDelegate::EndEditHint endHint)
  926. {
  927. FilterNameEdited(editor, ui->asyncFilters);
  928. UNUSED_PARAMETER(endHint);
  929. }
  930. void OBSBasicFilters::EffectFilterNameEdited(
  931. QWidget *editor, QAbstractItemDelegate::EndEditHint endHint)
  932. {
  933. FilterNameEdited(editor, ui->effectFilters);
  934. UNUSED_PARAMETER(endHint);
  935. }
  936. void OBSBasicFilters::ResetFilters()
  937. {
  938. QListWidget *list = isAsync ? ui->asyncFilters : ui->effectFilters;
  939. int row = list->currentRow();
  940. OBSSource filter = GetFilter(row, isAsync);
  941. if (!filter)
  942. return;
  943. OBSDataAutoRelease settings = obs_source_get_settings(filter);
  944. OBSDataAutoRelease empty_settings = obs_data_create();
  945. FilterChangeUndoRedo((void *)filter, settings, empty_settings);
  946. obs_data_clear(settings);
  947. if (!view->DeferUpdate())
  948. obs_source_update(filter, nullptr);
  949. view->RefreshProperties();
  950. }
  951. void OBSBasicFilters::CopyFilter()
  952. {
  953. OBSSource filter = nullptr;
  954. if (isAsync)
  955. filter = GetFilter(ui->asyncFilters->currentRow(), true);
  956. else
  957. filter = GetFilter(ui->effectFilters->currentRow(), false);
  958. main->copyFilter = OBSGetWeakRef(filter);
  959. }
  960. void OBSBasicFilters::PasteFilter()
  961. {
  962. OBSSource filter = OBSGetStrongRef(main->copyFilter);
  963. if (!filter)
  964. return;
  965. OBSDataArrayAutoRelease undo_array = obs_source_backup_filters(source);
  966. obs_source_copy_single_filter(source, filter);
  967. OBSDataArrayAutoRelease redo_array = obs_source_backup_filters(source);
  968. const char *filterName = obs_source_get_name(filter);
  969. const char *sourceName = obs_source_get_name(source);
  970. QString text =
  971. QTStr("Undo.Filters.Paste.Single").arg(filterName, sourceName);
  972. main->CreateFilterPasteUndoRedoAction(text, source, undo_array,
  973. redo_array);
  974. }
  975. void OBSBasicFilters::delete_filter(OBSSource filter)
  976. {
  977. OBSDataAutoRelease wrapper = obs_save_source(filter);
  978. std::string parent_name(obs_source_get_name(source));
  979. obs_data_set_string(wrapper, "undo_name", parent_name.c_str());
  980. std::string scene_name = obs_source_get_name(
  981. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  982. ->GetCurrentSceneSource());
  983. auto undo = [scene_name](const std::string &data) {
  984. OBSSourceAutoRelease ssource =
  985. obs_get_source_by_name(scene_name.c_str());
  986. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  987. ->SetCurrentScene(ssource.Get(), true);
  988. OBSDataAutoRelease dat =
  989. obs_data_create_from_json(data.c_str());
  990. OBSSourceAutoRelease source = obs_get_source_by_name(
  991. obs_data_get_string(dat, "undo_name"));
  992. OBSSourceAutoRelease filter = obs_load_source(dat);
  993. obs_source_filter_add(source, filter);
  994. };
  995. OBSDataAutoRelease rwrapper = obs_data_create();
  996. obs_data_set_string(rwrapper, "fname", obs_source_get_name(filter));
  997. obs_data_set_string(rwrapper, "sname", parent_name.c_str());
  998. auto redo = [scene_name](const std::string &data) {
  999. OBSSourceAutoRelease ssource =
  1000. obs_get_source_by_name(scene_name.c_str());
  1001. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  1002. ->SetCurrentScene(ssource.Get(), true);
  1003. OBSDataAutoRelease dat =
  1004. obs_data_create_from_json(data.c_str());
  1005. OBSSourceAutoRelease source = obs_get_source_by_name(
  1006. obs_data_get_string(dat, "sname"));
  1007. OBSSourceAutoRelease filter = obs_source_get_filter_by_name(
  1008. source, obs_data_get_string(dat, "fname"));
  1009. obs_source_filter_remove(source, filter);
  1010. };
  1011. std::string undo_data(obs_data_get_json(wrapper));
  1012. std::string redo_data(obs_data_get_json(rwrapper));
  1013. main->undo_s.add_action(
  1014. QTStr("Undo.Delete").arg(obs_source_get_name(filter)), undo,
  1015. redo, undo_data, redo_data, false);
  1016. obs_source_filter_remove(source, filter);
  1017. }