window-basic-filters.cpp 35 KB

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