/****************************************************************************** Copyright (C) 2014 by Hugh Bailey 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 . ******************************************************************************/ #include "obs-app.hpp" #include "window-basic-properties.hpp" #include "window-basic-main.hpp" #include "qt-wrappers.hpp" #include "display-helpers.hpp" #include "properties-view.hpp" #include #include #include #include using namespace std; OBSBasicProperties::OBSBasicProperties(QWidget *parent, OBSSource source_) : QDialog (parent), preview (new OBSQTDisplay(this)), main (qobject_cast(parent)), acceptClicked (false), source (source_), removedSignal (obs_source_get_signal_handler(source), "remove", OBSBasicProperties::SourceRemoved, this), renamedSignal (obs_source_get_signal_handler(source), "rename", OBSBasicProperties::SourceRenamed, this), oldSettings (obs_data_create()), buttonBox (new QDialogButtonBox(this)) { int cx = (int)config_get_int(App()->GlobalConfig(), "PropertiesWindow", "cx"); int cy = (int)config_get_int(App()->GlobalConfig(), "PropertiesWindow", "cy"); buttonBox->setStandardButtons(QDialogButtonBox::Ok | QDialogButtonBox::Cancel); buttonBox->setObjectName(QStringLiteral("buttonBox")); if (cx > 400 && cy > 400) resize(cx, cy); QMetaObject::connectSlotsByName(this); /* The OBSData constructor increments the reference once */ obs_data_release(oldSettings); OBSData settings = obs_source_get_settings(source); obs_data_apply(oldSettings, settings); obs_data_release(settings); view = new OBSPropertiesView(settings, source, (PropertiesReloadCallback)obs_source_properties, (PropertiesUpdateCallback)obs_source_update); preview->setMinimumSize(20, 20); preview->setSizePolicy(QSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding)); setLayout(new QVBoxLayout(this)); layout()->addWidget(preview); layout()->addWidget(view); layout()->addWidget(buttonBox); layout()->setAlignment(buttonBox, Qt::AlignRight | Qt::AlignBottom); layout()->setAlignment(view, Qt::AlignBottom); view->setMaximumHeight(250); view->setMinimumHeight(150); view->show(); installEventFilter(CreateShortcutFilter()); connect(view, SIGNAL(PropertiesResized()), this, SLOT(OnPropertiesResized())); connect(windowHandle(), &QWindow::screenChanged, [this]() { QSize size = GetPixelSize(preview); obs_display_resize(display, size.width(), size.height()); }); const char *name = obs_source_get_name(source); setWindowTitle(QTStr("Basic.PropertiesWindow").arg(QT_UTF8(name))); obs_source_inc_showing(source); updatePropertiesSignal.Connect(obs_source_get_signal_handler(source), "update_properties", OBSBasicProperties::UpdateProperties, this); } OBSBasicProperties::~OBSBasicProperties() { obs_source_dec_showing(source); main->SaveProject(); } void OBSBasicProperties::SourceRemoved(void *data, calldata_t *params) { QMetaObject::invokeMethod(static_cast(data), "close"); UNUSED_PARAMETER(params); } void OBSBasicProperties::SourceRenamed(void *data, calldata_t *params) { const char *name = calldata_string(params, "new_name"); QString title = QTStr("Basic.PropertiesWindow").arg(QT_UTF8(name)); QMetaObject::invokeMethod(static_cast(data), "setWindowTitle", Q_ARG(QString, title)); } void OBSBasicProperties::UpdateProperties(void *data, calldata_t *) { QMetaObject::invokeMethod(static_cast(data)->view, "ReloadProperties"); } void OBSBasicProperties::on_buttonBox_clicked(QAbstractButton *button) { QDialogButtonBox::ButtonRole val = buttonBox->buttonRole(button); if (val == QDialogButtonBox::AcceptRole) { acceptClicked = true; close(); if (view->DeferUpdate()) view->UpdateSettings(); } if (val == QDialogButtonBox::RejectRole) { obs_data_t *settings = obs_source_get_settings(source); obs_data_clear(settings); obs_data_release(settings); if (view->DeferUpdate()) obs_data_apply(settings, oldSettings); else obs_source_update(source, oldSettings); close(); } } void OBSBasicProperties::DrawPreview(void *data, uint32_t cx, uint32_t cy) { OBSBasicProperties *window = static_cast(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(); } void OBSBasicProperties::OnPropertiesResized() { QSize size = GetPixelSize(preview); obs_display_resize(display, size.width(), size.height()); } void OBSBasicProperties::resizeEvent(QResizeEvent *event) { if (isVisible()) { QSize size = GetPixelSize(preview); obs_display_resize(display, size.width(), size.height()); } QDialog::resizeEvent(event); } void OBSBasicProperties::Cleanup() { // remove draw callback and release display in case our drawable // surfaces go away before the destructor gets called obs_display_remove_draw_callback(display, OBSBasicProperties::DrawPreview, this); display = nullptr; config_set_int(App()->GlobalConfig(), "PropertiesWindow", "cx", width()); config_set_int(App()->GlobalConfig(), "PropertiesWindow", "cy", height()); } void OBSBasicProperties::reject() { if (!acceptClicked && (CheckSettings() != 0)) { if (!ConfirmQuit()) { return; } } Cleanup(); done(0); } void OBSBasicProperties::closeEvent(QCloseEvent *event) { if (!acceptClicked && (CheckSettings() != 0)) { if (!ConfirmQuit()) { event->ignore(); return; } } QDialog::closeEvent(event); if (!event->isAccepted()) return; Cleanup(); } void OBSBasicProperties::Init() { gs_init_data init_data = {}; show(); QSize previewSize = GetPixelSize(preview); init_data.cx = uint32_t(previewSize.width()); init_data.cy = uint32_t(previewSize.height()); init_data.format = GS_RGBA; QTToGSWindow(preview->winId(), init_data.window); display = obs_display_create(&init_data); if (display) obs_display_add_draw_callback(display, OBSBasicProperties::DrawPreview, this); } int OBSBasicProperties::CheckSettings() { OBSData currentSettings = obs_source_get_settings(source); const char *oldSettingsJson = obs_data_get_json(oldSettings); const char *currentSettingsJson = obs_data_get_json(currentSettings); int ret = strcmp(currentSettingsJson, oldSettingsJson); obs_data_release(currentSettings); return ret; } bool OBSBasicProperties::ConfirmQuit() { QMessageBox::StandardButton button; button = QMessageBox::question(this, QTStr("Basic.PropertiesWindow.ConfirmTitle"), QTStr("Basic.PropertiesWindow.Confirm"), QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel); switch (button) { case QMessageBox::Save: if (view->DeferUpdate()) view->UpdateSettings(); // Do nothing because the settings are already updated break; case QMessageBox::Discard: obs_source_update(source, oldSettings); break; case QMessageBox::Cancel: return false; break; default: /* If somehow the dialog fails to show, just default to * saving the settings. */ break; } return true; }