window-basic-filters.cpp 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822
  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 "window-namedialog.hpp"
  15. #include "window-basic-main.hpp"
  16. #include "window-basic-filters.hpp"
  17. #include "display-helpers.hpp"
  18. #include "qt-wrappers.hpp"
  19. #include "visibility-item-widget.hpp"
  20. #include "item-widget-helpers.hpp"
  21. #include "obs-app.hpp"
  22. #include <QMessageBox>
  23. #include <QCloseEvent>
  24. #include <vector>
  25. #include <string>
  26. #include <QMenu>
  27. #include <QVariant>
  28. using namespace std;
  29. Q_DECLARE_METATYPE(OBSSource);
  30. OBSBasicFilters::OBSBasicFilters(QWidget *parent, OBSSource source_)
  31. : QDialog (parent),
  32. ui (new Ui::OBSBasicFilters),
  33. source (source_),
  34. addSignal (obs_source_get_signal_handler(source),
  35. "filter_add",
  36. OBSBasicFilters::OBSSourceFilterAdded,
  37. this),
  38. removeSignal (obs_source_get_signal_handler(source),
  39. "filter_remove",
  40. OBSBasicFilters::OBSSourceFilterRemoved,
  41. this),
  42. reorderSignal (obs_source_get_signal_handler(source),
  43. "reorder_filters",
  44. OBSBasicFilters::OBSSourceReordered,
  45. this),
  46. removeSourceSignal (obs_source_get_signal_handler(source),
  47. "remove",
  48. OBSBasicFilters::SourceRemoved, this),
  49. renameSourceSignal (obs_source_get_signal_handler(source),
  50. "rename",
  51. OBSBasicFilters::SourceRenamed, this),
  52. noPreviewMargin (13)
  53. {
  54. main = reinterpret_cast<OBSBasic*>(parent);
  55. ui->setupUi(this);
  56. UpdateFilters();
  57. ui->asyncFilters->setItemDelegate(
  58. new VisibilityItemDelegate(ui->asyncFilters));
  59. ui->effectFilters->setItemDelegate(
  60. new VisibilityItemDelegate(ui->effectFilters));
  61. const char *name = obs_source_get_name(source);
  62. setWindowTitle(QTStr("Basic.Filters.Title").arg(QT_UTF8(name)));
  63. #ifndef QT_NO_SHORTCUT
  64. ui->actionRemoveFilter->setShortcut(QApplication::translate("OBSBasicFilters", "Del", nullptr));
  65. #endif // QT_NO_SHORTCUT
  66. addAction(ui->actionRemoveFilter);
  67. installEventFilter(CreateShortcutFilter());
  68. connect(ui->asyncFilters->itemDelegate(),
  69. SIGNAL(closeEditor(QWidget*,
  70. QAbstractItemDelegate::EndEditHint)),
  71. this,
  72. SLOT(AsyncFilterNameEdited(QWidget*,
  73. QAbstractItemDelegate::EndEditHint)));
  74. connect(ui->effectFilters->itemDelegate(),
  75. SIGNAL(closeEditor(QWidget*,
  76. QAbstractItemDelegate::EndEditHint)),
  77. this,
  78. SLOT(EffectFilterNameEdited(QWidget*,
  79. QAbstractItemDelegate::EndEditHint)));
  80. QPushButton *close = ui->buttonBox->button(QDialogButtonBox::Close);
  81. connect(close, SIGNAL(clicked()), this, SLOT(close()));
  82. close->setDefault(true);
  83. ui->buttonBox->button(QDialogButtonBox::Reset)->setText(
  84. QTStr("Defaults"));
  85. connect(ui->buttonBox->button(QDialogButtonBox::Reset),
  86. SIGNAL(clicked()), this, SLOT(ResetFilters()));
  87. uint32_t caps = obs_source_get_output_flags(source);
  88. bool audio = (caps & OBS_SOURCE_AUDIO) != 0;
  89. bool audioOnly = (caps & OBS_SOURCE_VIDEO) == 0;
  90. bool async = (caps & OBS_SOURCE_ASYNC) != 0;
  91. if (!async && !audio) {
  92. ui->asyncWidget->setVisible(false);
  93. ui->separatorLine->setVisible(false);
  94. }
  95. if (audioOnly) {
  96. ui->effectWidget->setVisible(false);
  97. ui->separatorLine->setVisible(false);
  98. }
  99. if (audioOnly || (audio && !async))
  100. ui->asyncLabel->setText(QTStr("Basic.Filters.AudioFilters"));
  101. auto addDrawCallback = [this] ()
  102. {
  103. obs_display_add_draw_callback(ui->preview->GetDisplay(),
  104. OBSBasicFilters::DrawPreview, this);
  105. };
  106. enum obs_source_type type = obs_source_get_type(source);
  107. bool drawable_type = type == OBS_SOURCE_TYPE_INPUT ||
  108. type == OBS_SOURCE_TYPE_SCENE;
  109. if ((caps & OBS_SOURCE_VIDEO) != 0) {
  110. ui->rightLayout->setContentsMargins(0, 0, 0, 0);
  111. ui->preview->show();
  112. if (drawable_type)
  113. connect(ui->preview, &OBSQTDisplay::DisplayCreated,
  114. addDrawCallback);
  115. } else {
  116. ui->rightLayout->setContentsMargins(0, noPreviewMargin, 0, 0);
  117. ui->rightContainerLayout->insertStretch(1);
  118. ui->preview->hide();
  119. }
  120. }
  121. OBSBasicFilters::~OBSBasicFilters()
  122. {
  123. ClearListItems(ui->asyncFilters);
  124. ClearListItems(ui->effectFilters);
  125. }
  126. void OBSBasicFilters::Init()
  127. {
  128. show();
  129. }
  130. inline OBSSource OBSBasicFilters::GetFilter(int row, bool async)
  131. {
  132. if (row == -1)
  133. return OBSSource();
  134. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  135. QListWidgetItem *item = list->item(row);
  136. if (!item)
  137. return OBSSource();
  138. QVariant v = item->data(Qt::UserRole);
  139. return v.value<OBSSource>();
  140. }
  141. void OBSBasicFilters::UpdatePropertiesView(int row, bool async)
  142. {
  143. if (view) {
  144. updatePropertiesSignal.Disconnect();
  145. ui->rightLayout->removeWidget(view);
  146. view->deleteLater();
  147. view = nullptr;
  148. }
  149. OBSSource filter = GetFilter(row, async);
  150. if (!filter)
  151. return;
  152. obs_data_t *settings = obs_source_get_settings(filter);
  153. view = new OBSPropertiesView(settings, filter,
  154. (PropertiesReloadCallback)obs_source_properties,
  155. (PropertiesUpdateCallback)obs_source_update);
  156. updatePropertiesSignal.Connect(obs_source_get_signal_handler(filter),
  157. "update_properties",
  158. OBSBasicFilters::UpdateProperties,
  159. this);
  160. uint32_t caps = obs_source_get_output_flags(filter);
  161. if ((caps & OBS_SOURCE_VIDEO)) {
  162. ui->rightLayout->setContentsMargins(0, 0, 0, 0);
  163. ui->preview->show();
  164. } else {
  165. ui->rightLayout->setContentsMargins(0, noPreviewMargin, 0, 0);
  166. ui->preview->hide();
  167. }
  168. obs_data_release(settings);
  169. view->setMaximumHeight(250);
  170. view->setMinimumHeight(150);
  171. ui->rightLayout->addWidget(view);
  172. view->show();
  173. }
  174. void OBSBasicFilters::UpdateProperties(void *data, calldata_t *)
  175. {
  176. QMetaObject::invokeMethod(static_cast<OBSBasicFilters*>(data)->view,
  177. "ReloadProperties");
  178. }
  179. void OBSBasicFilters::AddFilter(OBSSource filter)
  180. {
  181. uint32_t flags = obs_source_get_output_flags(filter);
  182. bool async = (flags & OBS_SOURCE_ASYNC) != 0;
  183. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  184. QListWidgetItem *item = new QListWidgetItem();
  185. Qt::ItemFlags itemFlags = item->flags();
  186. item->setFlags(itemFlags | Qt::ItemIsEditable);
  187. item->setData(Qt::UserRole, QVariant::fromValue(filter));
  188. list->addItem(item);
  189. list->setCurrentItem(item);
  190. SetupVisibilityItem(list, item, filter);
  191. }
  192. void OBSBasicFilters::RemoveFilter(OBSSource filter)
  193. {
  194. uint32_t flags = obs_source_get_output_flags(filter);
  195. bool async = (flags & OBS_SOURCE_ASYNC) != 0;
  196. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  197. for (int i = 0; i < list->count(); i++) {
  198. QListWidgetItem *item = list->item(i);
  199. QVariant v = item->data(Qt::UserRole);
  200. OBSSource curFilter = v.value<OBSSource>();
  201. if (filter == curFilter) {
  202. DeleteListItem(list, item);
  203. break;
  204. }
  205. }
  206. const char *filterName = obs_source_get_name(filter);
  207. const char *sourceName = obs_source_get_name(source);
  208. if (!sourceName || !filterName)
  209. return;
  210. const char *filterId = obs_source_get_id(filter);
  211. blog(LOG_INFO, "User removed filter '%s' (%s) from source '%s'",
  212. filterName, filterId, sourceName);
  213. main->SaveProject();
  214. }
  215. struct FilterOrderInfo {
  216. int asyncIdx = 0;
  217. int effectIdx = 0;
  218. OBSBasicFilters *window;
  219. inline FilterOrderInfo(OBSBasicFilters *window_) : window(window_) {}
  220. };
  221. void OBSBasicFilters::ReorderFilter(QListWidget *list,
  222. obs_source_t *filter, size_t idx)
  223. {
  224. int count = list->count();
  225. for (int i = 0; i < count; i++) {
  226. QListWidgetItem *listItem = list->item(i);
  227. QVariant v = listItem->data(Qt::UserRole);
  228. OBSSource filterItem = v.value<OBSSource>();
  229. if (filterItem == filter) {
  230. if ((int)idx != i) {
  231. bool sel = (list->currentRow() == i);
  232. listItem = TakeListItem(list, i);
  233. if (listItem) {
  234. list->insertItem((int)idx, listItem);
  235. SetupVisibilityItem(list,
  236. listItem, filterItem);
  237. if (sel)
  238. list->setCurrentRow((int)idx);
  239. }
  240. }
  241. break;
  242. }
  243. }
  244. }
  245. void OBSBasicFilters::ReorderFilters()
  246. {
  247. FilterOrderInfo info(this);
  248. obs_source_enum_filters(source,
  249. [] (obs_source_t*, obs_source_t *filter, void *p)
  250. {
  251. FilterOrderInfo *info =
  252. reinterpret_cast<FilterOrderInfo*>(p);
  253. uint32_t flags;
  254. bool async;
  255. flags = obs_source_get_output_flags(filter);
  256. async = (flags & OBS_SOURCE_ASYNC) != 0;
  257. if (async) {
  258. info->window->ReorderFilter(
  259. info->window->ui->asyncFilters,
  260. filter, info->asyncIdx++);
  261. } else {
  262. info->window->ReorderFilter(
  263. info->window->ui->effectFilters,
  264. filter, info->effectIdx++);
  265. }
  266. }, &info);
  267. }
  268. void OBSBasicFilters::UpdateFilters()
  269. {
  270. if (!source)
  271. return;
  272. ClearListItems(ui->effectFilters);
  273. ClearListItems(ui->asyncFilters);
  274. obs_source_enum_filters(source,
  275. [] (obs_source_t*, obs_source_t *filter, void *p)
  276. {
  277. OBSBasicFilters *window =
  278. reinterpret_cast<OBSBasicFilters*>(p);
  279. window->AddFilter(filter);
  280. }, this);
  281. main->SaveProject();
  282. }
  283. static bool filter_compatible(bool async, uint32_t sourceFlags,
  284. uint32_t filterFlags)
  285. {
  286. bool filterVideo = (filterFlags & OBS_SOURCE_VIDEO) != 0;
  287. bool filterAsync = (filterFlags & OBS_SOURCE_ASYNC) != 0;
  288. bool filterAudio = (filterFlags & OBS_SOURCE_AUDIO) != 0;
  289. bool audio = (sourceFlags & OBS_SOURCE_AUDIO) != 0;
  290. bool audioOnly = (sourceFlags & OBS_SOURCE_VIDEO) == 0;
  291. bool asyncSource = (sourceFlags & OBS_SOURCE_ASYNC) != 0;
  292. if (async && ((audioOnly && filterVideo) || (!audio && !asyncSource)))
  293. return false;
  294. return (async && (filterAudio || filterAsync)) ||
  295. (!async && !filterAudio && !filterAsync);
  296. }
  297. QMenu *OBSBasicFilters::CreateAddFilterPopupMenu(bool async)
  298. {
  299. uint32_t sourceFlags = obs_source_get_output_flags(source);
  300. const char *type_str;
  301. bool foundValues = false;
  302. size_t idx = 0;
  303. struct FilterInfo {
  304. string type;
  305. string name;
  306. inline FilterInfo(const char *type_, const char *name_)
  307. : type(type_), name(name_)
  308. {}
  309. };
  310. vector<FilterInfo> types;
  311. while (obs_enum_filter_types(idx++, &type_str)) {
  312. const char *name = obs_source_get_display_name(type_str);
  313. uint32_t caps = obs_get_source_output_flags(type_str);
  314. if ((caps & OBS_SOURCE_DEPRECATED) != 0)
  315. continue;
  316. if ((caps & OBS_SOURCE_CAP_DISABLED) != 0)
  317. continue;
  318. auto it = types.begin();
  319. for (; it != types.end(); ++it) {
  320. if (it->name >= name)
  321. break;
  322. }
  323. types.emplace(it, type_str, name);
  324. }
  325. QMenu *popup = new QMenu(QTStr("Add"), this);
  326. for (FilterInfo &type : types) {
  327. uint32_t filterFlags = obs_get_source_output_flags(
  328. type.type.c_str());
  329. if (!filter_compatible(async, sourceFlags, filterFlags))
  330. continue;
  331. QAction *popupItem = new QAction(QT_UTF8(type.name.c_str()),
  332. this);
  333. popupItem->setData(QT_UTF8(type.type.c_str()));
  334. connect(popupItem, SIGNAL(triggered(bool)),
  335. this, SLOT(AddFilterFromAction()));
  336. popup->addAction(popupItem);
  337. foundValues = true;
  338. }
  339. if (!foundValues) {
  340. delete popup;
  341. popup = nullptr;
  342. }
  343. return popup;
  344. }
  345. void OBSBasicFilters::AddNewFilter(const char *id)
  346. {
  347. if (id && *id) {
  348. obs_source_t *existing_filter;
  349. string name = obs_source_get_display_name(id);
  350. bool success = NameDialog::AskForName(this,
  351. QTStr("Basic.Filters.AddFilter.Title"),
  352. QTStr("Basic.FIlters.AddFilter.Text"), name,
  353. QT_UTF8(name.c_str()));
  354. if (!success)
  355. return;
  356. if (name.empty()) {
  357. OBSMessageBox::information(this,
  358. QTStr("NoNameEntered.Title"),
  359. QTStr("NoNameEntered.Text"));
  360. AddNewFilter(id);
  361. return;
  362. }
  363. existing_filter = obs_source_get_filter_by_name(source,
  364. name.c_str());
  365. if (existing_filter) {
  366. OBSMessageBox::information(this,
  367. QTStr("NameExists.Title"),
  368. QTStr("NameExists.Text"));
  369. obs_source_release(existing_filter);
  370. AddNewFilter(id);
  371. return;
  372. }
  373. obs_source_t *filter = obs_source_create(id, name.c_str(),
  374. nullptr, nullptr);
  375. if (filter) {
  376. const char *sourceName = obs_source_get_name(source);
  377. blog(LOG_INFO, "User added filter '%s' (%s) "
  378. "to source '%s'",
  379. name.c_str(), id, sourceName);
  380. obs_source_filter_add(source, filter);
  381. obs_source_release(filter);
  382. }
  383. }
  384. }
  385. void OBSBasicFilters::AddFilterFromAction()
  386. {
  387. QAction *action = qobject_cast<QAction*>(sender());
  388. if (!action)
  389. return;
  390. AddNewFilter(QT_TO_UTF8(action->data().toString()));
  391. }
  392. void OBSBasicFilters::closeEvent(QCloseEvent *event)
  393. {
  394. QDialog::closeEvent(event);
  395. if (!event->isAccepted())
  396. return;
  397. obs_display_remove_draw_callback (ui->preview->GetDisplay(),
  398. OBSBasicFilters::DrawPreview, this);
  399. main->SaveProject();
  400. }
  401. /* OBS Signals */
  402. void OBSBasicFilters::OBSSourceFilterAdded(void *param, calldata_t *data)
  403. {
  404. OBSBasicFilters *window = reinterpret_cast<OBSBasicFilters*>(param);
  405. obs_source_t *filter = (obs_source_t*)calldata_ptr(data, "filter");
  406. QMetaObject::invokeMethod(window, "AddFilter",
  407. Q_ARG(OBSSource, OBSSource(filter)));
  408. }
  409. void OBSBasicFilters::OBSSourceFilterRemoved(void *param, calldata_t *data)
  410. {
  411. OBSBasicFilters *window = reinterpret_cast<OBSBasicFilters*>(param);
  412. obs_source_t *filter = (obs_source_t*)calldata_ptr(data, "filter");
  413. QMetaObject::invokeMethod(window, "RemoveFilter",
  414. Q_ARG(OBSSource, OBSSource(filter)));
  415. }
  416. void OBSBasicFilters::OBSSourceReordered(void *param, calldata_t *data)
  417. {
  418. QMetaObject::invokeMethod(reinterpret_cast<OBSBasicFilters*>(param),
  419. "ReorderFilters");
  420. UNUSED_PARAMETER(data);
  421. }
  422. void OBSBasicFilters::SourceRemoved(void *data, calldata_t *params)
  423. {
  424. UNUSED_PARAMETER(params);
  425. QMetaObject::invokeMethod(static_cast<OBSBasicFilters*>(data),
  426. "close");
  427. }
  428. void OBSBasicFilters::SourceRenamed(void *data, calldata_t *params)
  429. {
  430. const char *name = calldata_string(params, "new_name");
  431. QString title = QTStr("Basic.Filters.Title").arg(QT_UTF8(name));
  432. QMetaObject::invokeMethod(static_cast<OBSBasicFilters*>(data),
  433. "setWindowTitle", Q_ARG(QString, title));
  434. }
  435. void OBSBasicFilters::DrawPreview(void *data, uint32_t cx, uint32_t cy)
  436. {
  437. OBSBasicFilters *window = static_cast<OBSBasicFilters*>(data);
  438. if (!window->source)
  439. return;
  440. uint32_t sourceCX = max(obs_source_get_width(window->source), 1u);
  441. uint32_t sourceCY = max(obs_source_get_height(window->source), 1u);
  442. int x, y;
  443. int newCX, newCY;
  444. float scale;
  445. GetScaleAndCenterPos(sourceCX, sourceCY, cx, cy, x, y, scale);
  446. newCX = int(scale * float(sourceCX));
  447. newCY = int(scale * float(sourceCY));
  448. gs_viewport_push();
  449. gs_projection_push();
  450. gs_ortho(0.0f, float(sourceCX), 0.0f, float(sourceCY), -100.0f, 100.0f);
  451. gs_set_viewport(x, y, newCX, newCY);
  452. obs_source_video_render(window->source);
  453. gs_projection_pop();
  454. gs_viewport_pop();
  455. }
  456. /* Qt Slots */
  457. static bool QueryRemove(QWidget *parent, obs_source_t *source)
  458. {
  459. const char *name = obs_source_get_name(source);
  460. QString text = QTStr("ConfirmRemove.Text");
  461. text.replace("$1", QT_UTF8(name));
  462. QMessageBox remove_source(parent);
  463. remove_source.setText(text);
  464. QAbstractButton *Yes = remove_source.addButton(QTStr("Yes"),
  465. QMessageBox::YesRole);
  466. remove_source.addButton(QTStr("No"), QMessageBox::NoRole);
  467. remove_source.setIcon(QMessageBox::Question);
  468. remove_source.setWindowTitle(QTStr("ConfirmRemove.Title"));
  469. remove_source.exec();
  470. return Yes == remove_source.clickedButton();
  471. }
  472. void OBSBasicFilters::on_addAsyncFilter_clicked()
  473. {
  474. QScopedPointer<QMenu> popup(CreateAddFilterPopupMenu(true));
  475. if (popup)
  476. popup->exec(QCursor::pos());
  477. }
  478. void OBSBasicFilters::on_removeAsyncFilter_clicked()
  479. {
  480. OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
  481. if (filter) {
  482. if (QueryRemove(this, filter))
  483. obs_source_filter_remove(source, filter);
  484. }
  485. }
  486. void OBSBasicFilters::on_moveAsyncFilterUp_clicked()
  487. {
  488. OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
  489. if (filter)
  490. obs_source_filter_set_order(source, filter, OBS_ORDER_MOVE_UP);
  491. }
  492. void OBSBasicFilters::on_moveAsyncFilterDown_clicked()
  493. {
  494. OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
  495. if (filter)
  496. obs_source_filter_set_order(source, filter,
  497. OBS_ORDER_MOVE_DOWN);
  498. }
  499. void OBSBasicFilters::on_asyncFilters_GotFocus()
  500. {
  501. UpdatePropertiesView(ui->asyncFilters->currentRow(), true);
  502. isAsync = true;
  503. }
  504. void OBSBasicFilters::on_asyncFilters_currentRowChanged(int row)
  505. {
  506. UpdatePropertiesView(row, true);
  507. }
  508. void OBSBasicFilters::on_addEffectFilter_clicked()
  509. {
  510. QScopedPointer<QMenu> popup(CreateAddFilterPopupMenu(false));
  511. if (popup)
  512. popup->exec(QCursor::pos());
  513. }
  514. void OBSBasicFilters::on_removeEffectFilter_clicked()
  515. {
  516. OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
  517. if (filter) {
  518. if (QueryRemove(this, filter))
  519. obs_source_filter_remove(source, filter);
  520. }
  521. }
  522. void OBSBasicFilters::on_moveEffectFilterUp_clicked()
  523. {
  524. OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
  525. if (filter)
  526. obs_source_filter_set_order(source, filter, OBS_ORDER_MOVE_UP);
  527. }
  528. void OBSBasicFilters::on_moveEffectFilterDown_clicked()
  529. {
  530. OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
  531. if (filter)
  532. obs_source_filter_set_order(source, filter,
  533. OBS_ORDER_MOVE_DOWN);
  534. }
  535. void OBSBasicFilters::on_effectFilters_GotFocus()
  536. {
  537. UpdatePropertiesView(ui->effectFilters->currentRow(), false);
  538. isAsync = false;
  539. }
  540. void OBSBasicFilters::on_effectFilters_currentRowChanged(int row)
  541. {
  542. UpdatePropertiesView(row, false);
  543. }
  544. void OBSBasicFilters::on_actionRemoveFilter_triggered()
  545. {
  546. if (ui->asyncFilters->hasFocus())
  547. on_removeAsyncFilter_clicked();
  548. else if (ui->effectFilters->hasFocus())
  549. on_removeEffectFilter_clicked();
  550. }
  551. void OBSBasicFilters::CustomContextMenu(const QPoint &pos, bool async)
  552. {
  553. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  554. QListWidgetItem *item = list->itemAt(pos);
  555. QMenu popup(window());
  556. QPointer<QMenu> addMenu = CreateAddFilterPopupMenu(async);
  557. if (addMenu)
  558. popup.addMenu(addMenu);
  559. if (item) {
  560. const char *renameSlot = async ?
  561. SLOT(RenameAsyncFilter()) : SLOT(RenameEffectFilter());
  562. const char *removeSlot = async ?
  563. SLOT(on_removeAsyncFilter_clicked()) :
  564. SLOT(on_removeEffectFilter_clicked());
  565. popup.addSeparator();
  566. popup.addAction(QTStr("Rename"), this, renameSlot);
  567. popup.addAction(QTStr("Remove"), this, removeSlot);
  568. }
  569. popup.exec(QCursor::pos());
  570. }
  571. void OBSBasicFilters::EditItem(QListWidgetItem *item, bool async)
  572. {
  573. Qt::ItemFlags flags = item->flags();
  574. OBSSource filter = item->data(Qt::UserRole).value<OBSSource>();
  575. const char *name = obs_source_get_name(filter);
  576. QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
  577. item->setText(QT_UTF8(name));
  578. item->setFlags(flags | Qt::ItemIsEditable);
  579. list->removeItemWidget(item);
  580. list->editItem(item);
  581. item->setFlags(flags);
  582. }
  583. void OBSBasicFilters::on_asyncFilters_customContextMenuRequested(
  584. const QPoint &pos)
  585. {
  586. CustomContextMenu(pos, true);
  587. }
  588. void OBSBasicFilters::on_effectFilters_customContextMenuRequested(
  589. const QPoint &pos)
  590. {
  591. CustomContextMenu(pos, false);
  592. }
  593. void OBSBasicFilters::RenameAsyncFilter()
  594. {
  595. EditItem(ui->asyncFilters->currentItem(), true);
  596. }
  597. void OBSBasicFilters::RenameEffectFilter()
  598. {
  599. EditItem(ui->effectFilters->currentItem(), false);
  600. }
  601. void OBSBasicFilters::FilterNameEdited(QWidget *editor, QListWidget *list)
  602. {
  603. QListWidgetItem *listItem = list->currentItem();
  604. OBSSource filter = listItem->data(Qt::UserRole).value<OBSSource>();
  605. QLineEdit *edit = qobject_cast<QLineEdit*>(editor);
  606. string name = QT_TO_UTF8(edit->text().trimmed());
  607. const char *prevName = obs_source_get_name(filter);
  608. bool sameName = (name == prevName);
  609. obs_source_t *foundFilter = nullptr;
  610. if (!sameName)
  611. foundFilter = obs_source_get_filter_by_name(source,
  612. name.c_str());
  613. if (foundFilter || name.empty() || sameName) {
  614. listItem->setText(QT_UTF8(prevName));
  615. if (foundFilter) {
  616. OBSMessageBox::information(window(),
  617. QTStr("NameExists.Title"),
  618. QTStr("NameExists.Text"));
  619. obs_source_release(foundFilter);
  620. } else if (name.empty()) {
  621. OBSMessageBox::information(window(),
  622. QTStr("NoNameEntered.Title"),
  623. QTStr("NoNameEntered.Text"));
  624. }
  625. } else {
  626. const char *sourceName = obs_source_get_name(source);
  627. blog(LOG_INFO, "User renamed filter '%s' on source '%s' to '%s'",
  628. prevName, sourceName, name.c_str());
  629. listItem->setText(QT_UTF8(name.c_str()));
  630. obs_source_set_name(filter, name.c_str());
  631. }
  632. listItem->setText(QString());
  633. SetupVisibilityItem(list, listItem, filter);
  634. }
  635. void OBSBasicFilters::AsyncFilterNameEdited(QWidget *editor,
  636. QAbstractItemDelegate::EndEditHint endHint)
  637. {
  638. FilterNameEdited(editor, ui->asyncFilters);
  639. UNUSED_PARAMETER(endHint);
  640. }
  641. void OBSBasicFilters::EffectFilterNameEdited(QWidget *editor,
  642. QAbstractItemDelegate::EndEditHint endHint)
  643. {
  644. FilterNameEdited(editor, ui->effectFilters);
  645. UNUSED_PARAMETER(endHint);
  646. }
  647. void OBSBasicFilters::ResetFilters()
  648. {
  649. QListWidget *list = isAsync ? ui->asyncFilters : ui->effectFilters;
  650. int row = list->currentRow();
  651. OBSSource filter = GetFilter(row, isAsync);
  652. if (!filter)
  653. return;
  654. obs_data_t *settings = obs_source_get_settings(filter);
  655. obs_data_clear(settings);
  656. obs_data_release(settings);
  657. if (!view->DeferUpdate())
  658. obs_source_update(filter, nullptr);
  659. view->RefreshProperties();
  660. }