| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972 | 
							- /******************************************************************************
 
-     Copyright (C) 2015 by Hugh Bailey <[email protected]>
 
-     This program is free software: you can redistribute it and/or modify
 
-     it under the terms of the GNU General Public License as published by
 
-     the Free Software Foundation, either version 2 of the License, or
 
-     (at your option) any later version.
 
-     This program is distributed in the hope that it will be useful,
 
-     but WITHOUT ANY WARRANTY; without even the implied warranty of
 
-     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
-     GNU General Public License for more details.
 
-     You should have received a copy of the GNU General Public License
 
-     along with this program.  If not, see <http://www.gnu.org/licenses/>.
 
- ******************************************************************************/
 
- #include "window-namedialog.hpp"
 
- #include "window-basic-main.hpp"
 
- #include "window-basic-filters.hpp"
 
- #include "display-helpers.hpp"
 
- #include "qt-wrappers.hpp"
 
- #include "visibility-item-widget.hpp"
 
- #include "item-widget-helpers.hpp"
 
- #include "obs-app.hpp"
 
- #include <QMessageBox>
 
- #include <QCloseEvent>
 
- #include <vector>
 
- #include <string>
 
- #include <QMenu>
 
- #include <QVariant>
 
- using namespace std;
 
- Q_DECLARE_METATYPE(OBSSource);
 
- OBSBasicFilters::OBSBasicFilters(QWidget *parent, OBSSource source_)
 
- 	: QDialog(parent),
 
- 	  ui(new Ui::OBSBasicFilters),
 
- 	  source(source_),
 
- 	  addSignal(obs_source_get_signal_handler(source), "filter_add",
 
- 		    OBSBasicFilters::OBSSourceFilterAdded, this),
 
- 	  removeSignal(obs_source_get_signal_handler(source), "filter_remove",
 
- 		       OBSBasicFilters::OBSSourceFilterRemoved, this),
 
- 	  reorderSignal(obs_source_get_signal_handler(source),
 
- 			"reorder_filters", OBSBasicFilters::OBSSourceReordered,
 
- 			this),
 
- 	  removeSourceSignal(obs_source_get_signal_handler(source), "remove",
 
- 			     OBSBasicFilters::SourceRemoved, this),
 
- 	  renameSourceSignal(obs_source_get_signal_handler(source), "rename",
 
- 			     OBSBasicFilters::SourceRenamed, this),
 
- 	  noPreviewMargin(13)
 
- {
 
- 	main = reinterpret_cast<OBSBasic *>(parent);
 
- 	setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
 
- 	ui->setupUi(this);
 
- 	UpdateFilters();
 
- 	ui->asyncFilters->setItemDelegate(
 
- 		new VisibilityItemDelegate(ui->asyncFilters));
 
- 	ui->effectFilters->setItemDelegate(
 
- 		new VisibilityItemDelegate(ui->effectFilters));
 
- 	const char *name = obs_source_get_name(source);
 
- 	setWindowTitle(QTStr("Basic.Filters.Title").arg(QT_UTF8(name)));
 
- #ifndef QT_NO_SHORTCUT
 
- 	ui->actionRemoveFilter->setShortcut(
 
- 		QApplication::translate("OBSBasicFilters", "Del", nullptr));
 
- #endif // QT_NO_SHORTCUT
 
- 	addAction(ui->actionRemoveFilter);
 
- 	addAction(ui->actionMoveUp);
 
- 	addAction(ui->actionMoveDown);
 
- 	installEventFilter(CreateShortcutFilter());
 
- 	connect(ui->asyncFilters->itemDelegate(),
 
- 		SIGNAL(closeEditor(QWidget *,
 
- 				   QAbstractItemDelegate::EndEditHint)),
 
- 		this,
 
- 		SLOT(AsyncFilterNameEdited(
 
- 			QWidget *, QAbstractItemDelegate::EndEditHint)));
 
- 	connect(ui->effectFilters->itemDelegate(),
 
- 		SIGNAL(closeEditor(QWidget *,
 
- 				   QAbstractItemDelegate::EndEditHint)),
 
- 		this,
 
- 		SLOT(EffectFilterNameEdited(
 
- 			QWidget *, QAbstractItemDelegate::EndEditHint)));
 
- 	QPushButton *close = ui->buttonBox->button(QDialogButtonBox::Close);
 
- 	connect(close, SIGNAL(clicked()), this, SLOT(close()));
 
- 	close->setDefault(true);
 
- 	ui->buttonBox->button(QDialogButtonBox::Reset)
 
- 		->setText(QTStr("Defaults"));
 
- 	connect(ui->buttonBox->button(QDialogButtonBox::Reset),
 
- 		SIGNAL(clicked()), this, SLOT(ResetFilters()));
 
- 	uint32_t caps = obs_source_get_output_flags(source);
 
- 	bool audio = (caps & OBS_SOURCE_AUDIO) != 0;
 
- 	bool audioOnly = (caps & OBS_SOURCE_VIDEO) == 0;
 
- 	bool async = (caps & OBS_SOURCE_ASYNC) != 0;
 
- 	if (!async && !audio) {
 
- 		ui->asyncWidget->setVisible(false);
 
- 		ui->separatorLine->setVisible(false);
 
- 	}
 
- 	if (audioOnly) {
 
- 		ui->effectWidget->setVisible(false);
 
- 		ui->separatorLine->setVisible(false);
 
- 	}
 
- 	if (audioOnly || (audio && !async))
 
- 		ui->asyncLabel->setText(QTStr("Basic.Filters.AudioFilters"));
 
- 	auto addDrawCallback = [this]() {
 
- 		obs_display_add_draw_callback(ui->preview->GetDisplay(),
 
- 					      OBSBasicFilters::DrawPreview,
 
- 					      this);
 
- 	};
 
- 	enum obs_source_type type = obs_source_get_type(source);
 
- 	bool drawable_type = type == OBS_SOURCE_TYPE_INPUT ||
 
- 			     type == OBS_SOURCE_TYPE_SCENE;
 
- 	if ((caps & OBS_SOURCE_VIDEO) != 0) {
 
- 		ui->rightLayout->setContentsMargins(0, 0, 0, 0);
 
- 		ui->preview->show();
 
- 		if (drawable_type)
 
- 			connect(ui->preview, &OBSQTDisplay::DisplayCreated,
 
- 				addDrawCallback);
 
- 	} else {
 
- 		ui->rightLayout->setContentsMargins(0, noPreviewMargin, 0, 0);
 
- 		ui->rightContainerLayout->insertStretch(1);
 
- 		ui->preview->hide();
 
- 	}
 
- 	QAction *renameAsync = new QAction(ui->asyncWidget);
 
- 	renameAsync->setShortcutContext(Qt::WidgetWithChildrenShortcut);
 
- 	connect(renameAsync, SIGNAL(triggered()), this,
 
- 		SLOT(RenameAsyncFilter()));
 
- 	ui->asyncWidget->addAction(renameAsync);
 
- 	QAction *renameEffect = new QAction(ui->effectWidget);
 
- 	renameEffect->setShortcutContext(Qt::WidgetWithChildrenShortcut);
 
- 	connect(renameEffect, SIGNAL(triggered()), this,
 
- 		SLOT(RenameEffectFilter()));
 
- 	ui->effectWidget->addAction(renameEffect);
 
- #ifdef __APPLE__
 
- 	renameAsync->setShortcut({Qt::Key_Return});
 
- 	renameEffect->setShortcut({Qt::Key_Return});
 
- #else
 
- 	renameAsync->setShortcut({Qt::Key_F2});
 
- 	renameEffect->setShortcut({Qt::Key_F2});
 
- #endif
 
- }
 
- OBSBasicFilters::~OBSBasicFilters()
 
- {
 
- 	ClearListItems(ui->asyncFilters);
 
- 	ClearListItems(ui->effectFilters);
 
- }
 
- void OBSBasicFilters::Init()
 
- {
 
- 	show();
 
- }
 
- inline OBSSource OBSBasicFilters::GetFilter(int row, bool async)
 
- {
 
- 	if (row == -1)
 
- 		return OBSSource();
 
- 	QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
 
- 	QListWidgetItem *item = list->item(row);
 
- 	if (!item)
 
- 		return OBSSource();
 
- 	QVariant v = item->data(Qt::UserRole);
 
- 	return v.value<OBSSource>();
 
- }
 
- void OBSBasicFilters::UpdatePropertiesView(int row, bool async)
 
- {
 
- 	if (view) {
 
- 		updatePropertiesSignal.Disconnect();
 
- 		ui->rightLayout->removeWidget(view);
 
- 		view->deleteLater();
 
- 		view = nullptr;
 
- 	}
 
- 	OBSSource filter = GetFilter(row, async);
 
- 	if (!filter)
 
- 		return;
 
- 	obs_data_t *settings = obs_source_get_settings(filter);
 
- 	view = new OBSPropertiesView(
 
- 		settings, filter,
 
- 		(PropertiesReloadCallback)obs_source_properties,
 
- 		(PropertiesUpdateCallback)obs_source_update);
 
- 	updatePropertiesSignal.Connect(obs_source_get_signal_handler(filter),
 
- 				       "update_properties",
 
- 				       OBSBasicFilters::UpdateProperties, this);
 
- 	obs_data_release(settings);
 
- 	view->setMaximumHeight(250);
 
- 	view->setMinimumHeight(150);
 
- 	ui->rightLayout->addWidget(view);
 
- 	view->show();
 
- }
 
- void OBSBasicFilters::UpdateProperties(void *data, calldata_t *)
 
- {
 
- 	QMetaObject::invokeMethod(static_cast<OBSBasicFilters *>(data)->view,
 
- 				  "ReloadProperties");
 
- }
 
- void OBSBasicFilters::AddFilter(OBSSource filter)
 
- {
 
- 	uint32_t flags = obs_source_get_output_flags(filter);
 
- 	bool async = (flags & OBS_SOURCE_ASYNC) != 0;
 
- 	QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
 
- 	QListWidgetItem *item = new QListWidgetItem();
 
- 	Qt::ItemFlags itemFlags = item->flags();
 
- 	item->setFlags(itemFlags | Qt::ItemIsEditable);
 
- 	item->setData(Qt::UserRole, QVariant::fromValue(filter));
 
- 	list->addItem(item);
 
- 	list->setCurrentItem(item);
 
- 	SetupVisibilityItem(list, item, filter);
 
- }
 
- void OBSBasicFilters::RemoveFilter(OBSSource filter)
 
- {
 
- 	uint32_t flags = obs_source_get_output_flags(filter);
 
- 	bool async = (flags & OBS_SOURCE_ASYNC) != 0;
 
- 	QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
 
- 	for (int i = 0; i < list->count(); i++) {
 
- 		QListWidgetItem *item = list->item(i);
 
- 		QVariant v = item->data(Qt::UserRole);
 
- 		OBSSource curFilter = v.value<OBSSource>();
 
- 		if (filter == curFilter) {
 
- 			DeleteListItem(list, item);
 
- 			break;
 
- 		}
 
- 	}
 
- 	const char *filterName = obs_source_get_name(filter);
 
- 	const char *sourceName = obs_source_get_name(source);
 
- 	if (!sourceName || !filterName)
 
- 		return;
 
- 	const char *filterId = obs_source_get_id(filter);
 
- 	blog(LOG_INFO, "User removed filter '%s' (%s) from source '%s'",
 
- 	     filterName, filterId, sourceName);
 
- 	main->SaveProject();
 
- }
 
- struct FilterOrderInfo {
 
- 	int asyncIdx = 0;
 
- 	int effectIdx = 0;
 
- 	OBSBasicFilters *window;
 
- 	inline FilterOrderInfo(OBSBasicFilters *window_) : window(window_) {}
 
- };
 
- void OBSBasicFilters::ReorderFilter(QListWidget *list, obs_source_t *filter,
 
- 				    size_t idx)
 
- {
 
- 	int count = list->count();
 
- 	for (int i = 0; i < count; i++) {
 
- 		QListWidgetItem *listItem = list->item(i);
 
- 		QVariant v = listItem->data(Qt::UserRole);
 
- 		OBSSource filterItem = v.value<OBSSource>();
 
- 		if (filterItem == filter) {
 
- 			if ((int)idx != i) {
 
- 				bool sel = (list->currentRow() == i);
 
- 				listItem = TakeListItem(list, i);
 
- 				if (listItem) {
 
- 					list->insertItem((int)idx, listItem);
 
- 					SetupVisibilityItem(list, listItem,
 
- 							    filterItem);
 
- 					if (sel)
 
- 						list->setCurrentRow((int)idx);
 
- 				}
 
- 			}
 
- 			break;
 
- 		}
 
- 	}
 
- }
 
- void OBSBasicFilters::ReorderFilters()
 
- {
 
- 	FilterOrderInfo info(this);
 
- 	obs_source_enum_filters(
 
- 		source,
 
- 		[](obs_source_t *, obs_source_t *filter, void *p) {
 
- 			FilterOrderInfo *info =
 
- 				reinterpret_cast<FilterOrderInfo *>(p);
 
- 			uint32_t flags;
 
- 			bool async;
 
- 			flags = obs_source_get_output_flags(filter);
 
- 			async = (flags & OBS_SOURCE_ASYNC) != 0;
 
- 			if (async) {
 
- 				info->window->ReorderFilter(
 
- 					info->window->ui->asyncFilters, filter,
 
- 					info->asyncIdx++);
 
- 			} else {
 
- 				info->window->ReorderFilter(
 
- 					info->window->ui->effectFilters, filter,
 
- 					info->effectIdx++);
 
- 			}
 
- 		},
 
- 		&info);
 
- }
 
- void OBSBasicFilters::UpdateFilters()
 
- {
 
- 	if (!source)
 
- 		return;
 
- 	ClearListItems(ui->effectFilters);
 
- 	ClearListItems(ui->asyncFilters);
 
- 	obs_source_enum_filters(
 
- 		source,
 
- 		[](obs_source_t *, obs_source_t *filter, void *p) {
 
- 			OBSBasicFilters *window =
 
- 				reinterpret_cast<OBSBasicFilters *>(p);
 
- 			window->AddFilter(filter);
 
- 		},
 
- 		this);
 
- 	main->SaveProject();
 
- }
 
- static bool filter_compatible(bool async, uint32_t sourceFlags,
 
- 			      uint32_t filterFlags)
 
- {
 
- 	bool filterVideo = (filterFlags & OBS_SOURCE_VIDEO) != 0;
 
- 	bool filterAsync = (filterFlags & OBS_SOURCE_ASYNC) != 0;
 
- 	bool filterAudio = (filterFlags & OBS_SOURCE_AUDIO) != 0;
 
- 	bool audio = (sourceFlags & OBS_SOURCE_AUDIO) != 0;
 
- 	bool audioOnly = (sourceFlags & OBS_SOURCE_VIDEO) == 0;
 
- 	bool asyncSource = (sourceFlags & OBS_SOURCE_ASYNC) != 0;
 
- 	if (async && ((audioOnly && filterVideo) || (!audio && !asyncSource)))
 
- 		return false;
 
- 	return (async && (filterAudio || filterAsync)) ||
 
- 	       (!async && !filterAudio && !filterAsync);
 
- }
 
- QMenu *OBSBasicFilters::CreateAddFilterPopupMenu(bool async)
 
- {
 
- 	uint32_t sourceFlags = obs_source_get_output_flags(source);
 
- 	const char *type_str;
 
- 	bool foundValues = false;
 
- 	size_t idx = 0;
 
- 	struct FilterInfo {
 
- 		string type;
 
- 		string name;
 
- 		inline FilterInfo(const char *type_, const char *name_)
 
- 			: type(type_), name(name_)
 
- 		{
 
- 		}
 
- 	};
 
- 	vector<FilterInfo> types;
 
- 	while (obs_enum_filter_types(idx++, &type_str)) {
 
- 		const char *name = obs_source_get_display_name(type_str);
 
- 		uint32_t caps = obs_get_source_output_flags(type_str);
 
- 		if ((caps & OBS_SOURCE_DEPRECATED) != 0)
 
- 			continue;
 
- 		if ((caps & OBS_SOURCE_CAP_DISABLED) != 0)
 
- 			continue;
 
- 		if ((caps & OBS_SOURCE_CAP_OBSOLETE) != 0)
 
- 			continue;
 
- 		auto it = types.begin();
 
- 		for (; it != types.end(); ++it) {
 
- 			if (it->name >= name)
 
- 				break;
 
- 		}
 
- 		types.emplace(it, type_str, name);
 
- 	}
 
- 	QMenu *popup = new QMenu(QTStr("Add"), this);
 
- 	for (FilterInfo &type : types) {
 
- 		uint32_t filterFlags =
 
- 			obs_get_source_output_flags(type.type.c_str());
 
- 		if (!filter_compatible(async, sourceFlags, filterFlags))
 
- 			continue;
 
- 		QAction *popupItem =
 
- 			new QAction(QT_UTF8(type.name.c_str()), this);
 
- 		popupItem->setData(QT_UTF8(type.type.c_str()));
 
- 		connect(popupItem, SIGNAL(triggered(bool)), this,
 
- 			SLOT(AddFilterFromAction()));
 
- 		popup->addAction(popupItem);
 
- 		foundValues = true;
 
- 	}
 
- 	if (!foundValues) {
 
- 		delete popup;
 
- 		popup = nullptr;
 
- 	}
 
- 	return popup;
 
- }
 
- void OBSBasicFilters::AddNewFilter(const char *id)
 
- {
 
- 	if (id && *id) {
 
- 		obs_source_t *existing_filter;
 
- 		string name = obs_source_get_display_name(id);
 
- 		QString placeholder = QString::fromStdString(name);
 
- 		QString text{placeholder};
 
- 		int i = 2;
 
- 		while ((existing_filter = obs_source_get_filter_by_name(
 
- 				source, QT_TO_UTF8(text)))) {
 
- 			obs_source_release(existing_filter);
 
- 			text = QString("%1 %2").arg(placeholder).arg(i++);
 
- 		}
 
- 		bool success = NameDialog::AskForName(
 
- 			this, QTStr("Basic.Filters.AddFilter.Title"),
 
- 			QTStr("Basic.Filters.AddFilter.Text"), name, text);
 
- 		if (!success)
 
- 			return;
 
- 		if (name.empty()) {
 
- 			OBSMessageBox::warning(this,
 
- 					       QTStr("NoNameEntered.Title"),
 
- 					       QTStr("NoNameEntered.Text"));
 
- 			AddNewFilter(id);
 
- 			return;
 
- 		}
 
- 		existing_filter =
 
- 			obs_source_get_filter_by_name(source, name.c_str());
 
- 		if (existing_filter) {
 
- 			OBSMessageBox::warning(this, QTStr("NameExists.Title"),
 
- 					       QTStr("NameExists.Text"));
 
- 			obs_source_release(existing_filter);
 
- 			AddNewFilter(id);
 
- 			return;
 
- 		}
 
- 		obs_source_t *filter =
 
- 			obs_source_create(id, name.c_str(), nullptr, nullptr);
 
- 		if (filter) {
 
- 			const char *sourceName = obs_source_get_name(source);
 
- 			blog(LOG_INFO,
 
- 			     "User added filter '%s' (%s) "
 
- 			     "to source '%s'",
 
- 			     name.c_str(), id, sourceName);
 
- 			obs_source_filter_add(source, filter);
 
- 			obs_source_release(filter);
 
- 		}
 
- 	}
 
- }
 
- void OBSBasicFilters::AddFilterFromAction()
 
- {
 
- 	QAction *action = qobject_cast<QAction *>(sender());
 
- 	if (!action)
 
- 		return;
 
- 	AddNewFilter(QT_TO_UTF8(action->data().toString()));
 
- }
 
- void OBSBasicFilters::closeEvent(QCloseEvent *event)
 
- {
 
- 	QDialog::closeEvent(event);
 
- 	if (!event->isAccepted())
 
- 		return;
 
- 	obs_display_remove_draw_callback(ui->preview->GetDisplay(),
 
- 					 OBSBasicFilters::DrawPreview, this);
 
- 	main->SaveProject();
 
- }
 
- /* OBS Signals */
 
- void OBSBasicFilters::OBSSourceFilterAdded(void *param, calldata_t *data)
 
- {
 
- 	OBSBasicFilters *window = reinterpret_cast<OBSBasicFilters *>(param);
 
- 	obs_source_t *filter = (obs_source_t *)calldata_ptr(data, "filter");
 
- 	QMetaObject::invokeMethod(window, "AddFilter",
 
- 				  Q_ARG(OBSSource, OBSSource(filter)));
 
- }
 
- void OBSBasicFilters::OBSSourceFilterRemoved(void *param, calldata_t *data)
 
- {
 
- 	OBSBasicFilters *window = reinterpret_cast<OBSBasicFilters *>(param);
 
- 	obs_source_t *filter = (obs_source_t *)calldata_ptr(data, "filter");
 
- 	QMetaObject::invokeMethod(window, "RemoveFilter",
 
- 				  Q_ARG(OBSSource, OBSSource(filter)));
 
- }
 
- void OBSBasicFilters::OBSSourceReordered(void *param, calldata_t *data)
 
- {
 
- 	QMetaObject::invokeMethod(reinterpret_cast<OBSBasicFilters *>(param),
 
- 				  "ReorderFilters");
 
- 	UNUSED_PARAMETER(data);
 
- }
 
- void OBSBasicFilters::SourceRemoved(void *param, calldata_t *data)
 
- {
 
- 	UNUSED_PARAMETER(data);
 
- 	QMetaObject::invokeMethod(static_cast<OBSBasicFilters *>(param),
 
- 				  "close");
 
- }
 
- void OBSBasicFilters::SourceRenamed(void *param, calldata_t *data)
 
- {
 
- 	const char *name = calldata_string(data, "new_name");
 
- 	QString title = QTStr("Basic.Filters.Title").arg(QT_UTF8(name));
 
- 	QMetaObject::invokeMethod(static_cast<OBSBasicFilters *>(param),
 
- 				  "setWindowTitle", Q_ARG(QString, title));
 
- }
 
- void OBSBasicFilters::DrawPreview(void *data, uint32_t cx, uint32_t cy)
 
- {
 
- 	OBSBasicFilters *window = static_cast<OBSBasicFilters *>(data);
 
- 	if (!window->source)
 
- 		return;
 
- 	uint32_t sourceCX = max(obs_source_get_width(window->source), 1u);
 
- 	uint32_t sourceCY = max(obs_source_get_height(window->source), 1u);
 
- 	int x, y;
 
- 	int newCX, newCY;
 
- 	float scale;
 
- 	GetScaleAndCenterPos(sourceCX, sourceCY, cx, cy, x, y, scale);
 
- 	newCX = int(scale * float(sourceCX));
 
- 	newCY = int(scale * float(sourceCY));
 
- 	gs_viewport_push();
 
- 	gs_projection_push();
 
- 	gs_ortho(0.0f, float(sourceCX), 0.0f, float(sourceCY), -100.0f, 100.0f);
 
- 	gs_set_viewport(x, y, newCX, newCY);
 
- 	obs_source_video_render(window->source);
 
- 	gs_projection_pop();
 
- 	gs_viewport_pop();
 
- }
 
- /* Qt Slots */
 
- static bool QueryRemove(QWidget *parent, obs_source_t *source)
 
- {
 
- 	const char *name = obs_source_get_name(source);
 
- 	QString text = QTStr("ConfirmRemove.Text");
 
- 	text.replace("$1", QT_UTF8(name));
 
- 	QMessageBox remove_source(parent);
 
- 	remove_source.setText(text);
 
- 	QAbstractButton *Yes =
 
- 		remove_source.addButton(QTStr("Yes"), QMessageBox::YesRole);
 
- 	remove_source.addButton(QTStr("No"), QMessageBox::NoRole);
 
- 	remove_source.setIcon(QMessageBox::Question);
 
- 	remove_source.setWindowTitle(QTStr("ConfirmRemove.Title"));
 
- 	remove_source.exec();
 
- 	return Yes == remove_source.clickedButton();
 
- }
 
- void OBSBasicFilters::on_addAsyncFilter_clicked()
 
- {
 
- 	QScopedPointer<QMenu> popup(CreateAddFilterPopupMenu(true));
 
- 	if (popup)
 
- 		popup->exec(QCursor::pos());
 
- }
 
- void OBSBasicFilters::on_removeAsyncFilter_clicked()
 
- {
 
- 	OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
 
- 	if (filter) {
 
- 		if (QueryRemove(this, filter))
 
- 			obs_source_filter_remove(source, filter);
 
- 	}
 
- }
 
- void OBSBasicFilters::on_moveAsyncFilterUp_clicked()
 
- {
 
- 	OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
 
- 	if (filter)
 
- 		obs_source_filter_set_order(source, filter, OBS_ORDER_MOVE_UP);
 
- }
 
- void OBSBasicFilters::on_moveAsyncFilterDown_clicked()
 
- {
 
- 	OBSSource filter = GetFilter(ui->asyncFilters->currentRow(), true);
 
- 	if (filter)
 
- 		obs_source_filter_set_order(source, filter,
 
- 					    OBS_ORDER_MOVE_DOWN);
 
- }
 
- void OBSBasicFilters::on_asyncFilters_GotFocus()
 
- {
 
- 	UpdatePropertiesView(ui->asyncFilters->currentRow(), true);
 
- 	isAsync = true;
 
- }
 
- void OBSBasicFilters::on_asyncFilters_currentRowChanged(int row)
 
- {
 
- 	UpdatePropertiesView(row, true);
 
- }
 
- void OBSBasicFilters::on_addEffectFilter_clicked()
 
- {
 
- 	QScopedPointer<QMenu> popup(CreateAddFilterPopupMenu(false));
 
- 	if (popup)
 
- 		popup->exec(QCursor::pos());
 
- }
 
- void OBSBasicFilters::on_removeEffectFilter_clicked()
 
- {
 
- 	OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
 
- 	if (filter) {
 
- 		if (QueryRemove(this, filter))
 
- 			obs_source_filter_remove(source, filter);
 
- 	}
 
- }
 
- void OBSBasicFilters::on_moveEffectFilterUp_clicked()
 
- {
 
- 	OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
 
- 	if (filter)
 
- 		obs_source_filter_set_order(source, filter, OBS_ORDER_MOVE_UP);
 
- }
 
- void OBSBasicFilters::on_moveEffectFilterDown_clicked()
 
- {
 
- 	OBSSource filter = GetFilter(ui->effectFilters->currentRow(), false);
 
- 	if (filter)
 
- 		obs_source_filter_set_order(source, filter,
 
- 					    OBS_ORDER_MOVE_DOWN);
 
- }
 
- void OBSBasicFilters::on_effectFilters_GotFocus()
 
- {
 
- 	UpdatePropertiesView(ui->effectFilters->currentRow(), false);
 
- 	isAsync = false;
 
- }
 
- void OBSBasicFilters::on_effectFilters_currentRowChanged(int row)
 
- {
 
- 	UpdatePropertiesView(row, false);
 
- }
 
- void OBSBasicFilters::on_actionRemoveFilter_triggered()
 
- {
 
- 	if (ui->asyncFilters->hasFocus())
 
- 		on_removeAsyncFilter_clicked();
 
- 	else if (ui->effectFilters->hasFocus())
 
- 		on_removeEffectFilter_clicked();
 
- }
 
- void OBSBasicFilters::on_actionMoveUp_triggered()
 
- {
 
- 	if (ui->asyncFilters->hasFocus())
 
- 		on_moveAsyncFilterUp_clicked();
 
- 	else if (ui->effectFilters->hasFocus())
 
- 		on_moveEffectFilterUp_clicked();
 
- }
 
- void OBSBasicFilters::on_actionMoveDown_triggered()
 
- {
 
- 	if (ui->asyncFilters->hasFocus())
 
- 		on_moveAsyncFilterDown_clicked();
 
- 	else if (ui->effectFilters->hasFocus())
 
- 		on_moveEffectFilterDown_clicked();
 
- }
 
- void OBSBasicFilters::CustomContextMenu(const QPoint &pos, bool async)
 
- {
 
- 	QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
 
- 	QListWidgetItem *item = list->itemAt(pos);
 
- 	QMenu popup(window());
 
- 	QPointer<QMenu> addMenu = CreateAddFilterPopupMenu(async);
 
- 	if (addMenu)
 
- 		popup.addMenu(addMenu);
 
- 	if (item) {
 
- 		const char *dulpicateSlot =
 
- 			async ? SLOT(DuplicateAsyncFilter())
 
- 			      : SLOT(DuplicateEffectFilter());
 
- 		const char *renameSlot = async ? SLOT(RenameAsyncFilter())
 
- 					       : SLOT(RenameEffectFilter());
 
- 		const char *removeSlot =
 
- 			async ? SLOT(on_removeAsyncFilter_clicked())
 
- 			      : SLOT(on_removeEffectFilter_clicked());
 
- 		popup.addSeparator();
 
- 		popup.addAction(QTStr("Duplicate"), this, dulpicateSlot);
 
- 		popup.addSeparator();
 
- 		popup.addAction(QTStr("Rename"), this, renameSlot);
 
- 		popup.addAction(QTStr("Remove"), this, removeSlot);
 
- 		popup.addSeparator();
 
- 		QAction *copyAction = new QAction(QTStr("Copy"));
 
- 		connect(copyAction, SIGNAL(triggered()), this,
 
- 			SLOT(CopyFilter()));
 
- 		copyAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_C));
 
- 		ui->effectWidget->addAction(copyAction);
 
- 		ui->asyncWidget->addAction(copyAction);
 
- 		popup.addAction(copyAction);
 
- 	}
 
- 	QAction *pasteAction = new QAction(QTStr("Paste"));
 
- 	pasteAction->setEnabled(main->copyFilter);
 
- 	connect(pasteAction, SIGNAL(triggered()), this, SLOT(PasteFilter()));
 
- 	pasteAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_V));
 
- 	ui->effectWidget->addAction(pasteAction);
 
- 	ui->asyncWidget->addAction(pasteAction);
 
- 	popup.addAction(pasteAction);
 
- 	popup.exec(QCursor::pos());
 
- }
 
- void OBSBasicFilters::EditItem(QListWidgetItem *item, bool async)
 
- {
 
- 	if (editActive)
 
- 		return;
 
- 	Qt::ItemFlags flags = item->flags();
 
- 	OBSSource filter = item->data(Qt::UserRole).value<OBSSource>();
 
- 	const char *name = obs_source_get_name(filter);
 
- 	QListWidget *list = async ? ui->asyncFilters : ui->effectFilters;
 
- 	item->setText(QT_UTF8(name));
 
- 	item->setFlags(flags | Qt::ItemIsEditable);
 
- 	list->removeItemWidget(item);
 
- 	list->editItem(item);
 
- 	item->setFlags(flags);
 
- 	editActive = true;
 
- }
 
- void OBSBasicFilters::DuplicateItem(QListWidgetItem *item)
 
- {
 
- 	OBSSource filter = item->data(Qt::UserRole).value<OBSSource>();
 
- 	string name = obs_source_get_name(filter);
 
- 	obs_source_t *existing_filter;
 
- 	QString placeholder = QString::fromStdString(name);
 
- 	QString text{placeholder};
 
- 	int i = 2;
 
- 	while ((existing_filter = obs_source_get_filter_by_name(
 
- 			source, QT_TO_UTF8(text)))) {
 
- 		obs_source_release(existing_filter);
 
- 		text = QString("%1 %2").arg(placeholder).arg(i++);
 
- 	}
 
- 	bool success = NameDialog::AskForName(
 
- 		this, QTStr("Basic.Filters.AddFilter.Title"),
 
- 		QTStr("Basic.Filters.AddFilter.Text"), name, text);
 
- 	if (!success)
 
- 		return;
 
- 	if (name.empty()) {
 
- 		OBSMessageBox::warning(this, QTStr("NoNameEntered.Title"),
 
- 				       QTStr("NoNameEntered.Text"));
 
- 		DuplicateItem(item);
 
- 		return;
 
- 	}
 
- 	existing_filter = obs_source_get_filter_by_name(source, name.c_str());
 
- 	if (existing_filter) {
 
- 		OBSMessageBox::warning(this, QTStr("NameExists.Title"),
 
- 				       QTStr("NameExists.Text"));
 
- 		obs_source_release(existing_filter);
 
- 		DuplicateItem(item);
 
- 		return;
 
- 	}
 
- 	bool enabled = obs_source_enabled(filter);
 
- 	obs_source_t *new_filter =
 
- 		obs_source_duplicate(filter, name.c_str(), false);
 
- 	if (new_filter) {
 
- 		const char *sourceName = obs_source_get_name(source);
 
- 		const char *id = obs_source_get_id(new_filter);
 
- 		blog(LOG_INFO,
 
- 		     "User duplicated filter '%s' (%s) from '%s' "
 
- 		     "to source '%s'",
 
- 		     name.c_str(), id, name.c_str(), sourceName);
 
- 		obs_source_set_enabled(new_filter, enabled);
 
- 		obs_source_filter_add(source, new_filter);
 
- 		obs_source_release(new_filter);
 
- 	}
 
- }
 
- void OBSBasicFilters::on_asyncFilters_customContextMenuRequested(
 
- 	const QPoint &pos)
 
- {
 
- 	CustomContextMenu(pos, true);
 
- }
 
- void OBSBasicFilters::on_effectFilters_customContextMenuRequested(
 
- 	const QPoint &pos)
 
- {
 
- 	CustomContextMenu(pos, false);
 
- }
 
- void OBSBasicFilters::RenameAsyncFilter()
 
- {
 
- 	EditItem(ui->asyncFilters->currentItem(), true);
 
- }
 
- void OBSBasicFilters::RenameEffectFilter()
 
- {
 
- 	EditItem(ui->effectFilters->currentItem(), false);
 
- }
 
- void OBSBasicFilters::DuplicateAsyncFilter()
 
- {
 
- 	DuplicateItem(ui->asyncFilters->currentItem());
 
- }
 
- void OBSBasicFilters::DuplicateEffectFilter()
 
- {
 
- 	DuplicateItem(ui->effectFilters->currentItem());
 
- }
 
- void OBSBasicFilters::FilterNameEdited(QWidget *editor, QListWidget *list)
 
- {
 
- 	QListWidgetItem *listItem = list->currentItem();
 
- 	OBSSource filter = listItem->data(Qt::UserRole).value<OBSSource>();
 
- 	QLineEdit *edit = qobject_cast<QLineEdit *>(editor);
 
- 	string name = QT_TO_UTF8(edit->text().trimmed());
 
- 	const char *prevName = obs_source_get_name(filter);
 
- 	bool sameName = (name == prevName);
 
- 	obs_source_t *foundFilter = nullptr;
 
- 	if (!sameName)
 
- 		foundFilter =
 
- 			obs_source_get_filter_by_name(source, name.c_str());
 
- 	if (foundFilter || name.empty() || sameName) {
 
- 		listItem->setText(QT_UTF8(prevName));
 
- 		if (foundFilter) {
 
- 			OBSMessageBox::information(window(),
 
- 						   QTStr("NameExists.Title"),
 
- 						   QTStr("NameExists.Text"));
 
- 			obs_source_release(foundFilter);
 
- 		} else if (name.empty()) {
 
- 			OBSMessageBox::information(window(),
 
- 						   QTStr("NoNameEntered.Title"),
 
- 						   QTStr("NoNameEntered.Text"));
 
- 		}
 
- 	} else {
 
- 		const char *sourceName = obs_source_get_name(source);
 
- 		blog(LOG_INFO,
 
- 		     "User renamed filter '%s' on source '%s' to '%s'",
 
- 		     prevName, sourceName, name.c_str());
 
- 		listItem->setText(QT_UTF8(name.c_str()));
 
- 		obs_source_set_name(filter, name.c_str());
 
- 	}
 
- 	listItem->setText(QString());
 
- 	SetupVisibilityItem(list, listItem, filter);
 
- 	editActive = false;
 
- }
 
- void OBSBasicFilters::AsyncFilterNameEdited(
 
- 	QWidget *editor, QAbstractItemDelegate::EndEditHint endHint)
 
- {
 
- 	FilterNameEdited(editor, ui->asyncFilters);
 
- 	UNUSED_PARAMETER(endHint);
 
- }
 
- void OBSBasicFilters::EffectFilterNameEdited(
 
- 	QWidget *editor, QAbstractItemDelegate::EndEditHint endHint)
 
- {
 
- 	FilterNameEdited(editor, ui->effectFilters);
 
- 	UNUSED_PARAMETER(endHint);
 
- }
 
- void OBSBasicFilters::ResetFilters()
 
- {
 
- 	QListWidget *list = isAsync ? ui->asyncFilters : ui->effectFilters;
 
- 	int row = list->currentRow();
 
- 	OBSSource filter = GetFilter(row, isAsync);
 
- 	if (!filter)
 
- 		return;
 
- 	obs_data_t *settings = obs_source_get_settings(filter);
 
- 	obs_data_clear(settings);
 
- 	obs_data_release(settings);
 
- 	if (!view->DeferUpdate())
 
- 		obs_source_update(filter, nullptr);
 
- 	view->RefreshProperties();
 
- }
 
- void OBSBasicFilters::CopyFilter()
 
- {
 
- 	OBSSource filter = nullptr;
 
- 	if (isAsync)
 
- 		filter = GetFilter(ui->asyncFilters->currentRow(), true);
 
- 	else
 
- 		filter = GetFilter(ui->effectFilters->currentRow(), false);
 
- 	main->copyFilter = OBSGetWeakRef(filter);
 
- }
 
- void OBSBasicFilters::PasteFilter()
 
- {
 
- 	OBSSource filter = OBSGetStrongRef(main->copyFilter);
 
- 	if (!filter)
 
- 		return;
 
- 	obs_source_copy_single_filter(source, filter);
 
- }
 
 
  |