window-basic-filters.cpp 21 KB

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