/* PCSX2 - PS2 Emulator for PCs * Copyright (C) 2002-2022 PCSX2 Dev Team * * PCSX2 is free software: you can redistribute it and/or modify it under the terms * of the GNU Lesser General Public License as published by the Free Software Found- * ation, either version 3 of the License, or (at your option) any later version. * * PCSX2 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 PCSX2. * If not, see . */ #pragma once #include #include #include #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) #include #else #include #endif #include #include #include #include #include #include #include "EmuThread.h" #include "QtHost.h" namespace SettingWidgetBinder { template struct SettingAccessor { static bool getBoolValue(const T* widget); static void setBoolValue(T* widget, bool value); static int getIntValue(const T* widget); static void setIntValue(T* widget, int value); static int getFloatValue(const T* widget); static void setFloatValue(T* widget, int value); static QString getStringValue(const T* widget); static void setStringValue(T* widget, const QString& value); template static void connectValueChanged(T* widget, F func); }; template <> struct SettingAccessor { static bool getBoolValue(const QLineEdit* widget) { return widget->text().toInt() != 0; } static void setBoolValue(QLineEdit* widget, bool value) { widget->setText(value ? QStringLiteral("1") : QStringLiteral("0")); } static int getIntValue(const QLineEdit* widget) { return widget->text().toInt(); } static void setIntValue(QLineEdit* widget, int value) { widget->setText(QString::number(value)); } static float getFloatValue(const QLineEdit* widget) { return widget->text().toFloat(); } static void setFloatValue(QLineEdit* widget, float value) { widget->setText(QString::number(value)); } static QString getStringValue(const QLineEdit* widget) { return widget->text(); } static void setStringValue(QLineEdit* widget, const QString& value) { widget->setText(value); } template static void connectValueChanged(QLineEdit* widget, F func) { widget->connect(widget, &QLineEdit::textChanged, func); } }; template <> struct SettingAccessor { static bool getBoolValue(const QComboBox* widget) { return widget->currentIndex() > 0; } static void setBoolValue(QComboBox* widget, bool value) { widget->setCurrentIndex(value ? 1 : 0); } static int getIntValue(const QComboBox* widget) { return widget->currentIndex(); } static void setIntValue(QComboBox* widget, int value) { widget->setCurrentIndex(value); } static float getFloatValue(const QComboBox* widget) { return static_cast(widget->currentIndex()); } static void setFloatValue(QComboBox* widget, float value) { widget->setCurrentIndex(static_cast(value)); } static QString getStringValue(const QComboBox* widget) { const QVariant currentData(widget->currentData()); #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) if (currentData.type() == QVariant::String) return currentData.toString(); #else if (currentData.metaType().id() == QMetaType::QString) return currentData.toString(); #endif return widget->currentText(); } static void setStringValue(QComboBox* widget, const QString& value) { const int index = widget->findData(value); if (index >= 0) { widget->setCurrentIndex(index); return; } widget->setCurrentText(value); } template static void connectValueChanged(QComboBox* widget, F func) { widget->connect(widget, static_cast(&QComboBox::currentIndexChanged), func); } }; template <> struct SettingAccessor { static bool getBoolValue(const QCheckBox* widget) { return widget->isChecked(); } static void setBoolValue(QCheckBox* widget, bool value) { widget->setChecked(value); } static int getIntValue(const QCheckBox* widget) { return widget->isChecked() ? 1 : 0; } static void setIntValue(QCheckBox* widget, int value) { widget->setChecked(value != 0); } static float getFloatValue(const QCheckBox* widget) { return widget->isChecked() ? 1.0f : 0.0f; } static void setFloatValue(QCheckBox* widget, float value) { widget->setChecked(value != 0.0f); } static QString getStringValue(const QCheckBox* widget) { return widget->isChecked() ? QStringLiteral("1") : QStringLiteral("0"); } static void setStringValue(QCheckBox* widget, const QString& value) { widget->setChecked(value.toInt() != 0); } template static void connectValueChanged(QCheckBox* widget, F func) { widget->connect(widget, &QCheckBox::stateChanged, func); } }; template <> struct SettingAccessor { static bool getBoolValue(const QSlider* widget) { return widget->value() > 0; } static void setBoolValue(QSlider* widget, bool value) { widget->setValue(value ? 1 : 0); } static int getIntValue(const QSlider* widget) { return widget->value(); } static void setIntValue(QSlider* widget, int value) { widget->setValue(value); } static float getFloatValue(const QSlider* widget) { return static_cast(widget->value()); } static void setFloatValue(QSlider* widget, float value) { widget->setValue(static_cast(value)); } static QString getStringValue(const QSlider* widget) { return QString::number(widget->value()); } static void setStringValue(QSlider* widget, const QString& value) { widget->setValue(value.toInt()); } template static void connectValueChanged(QSlider* widget, F func) { widget->connect(widget, &QSlider::valueChanged, func); } }; template <> struct SettingAccessor { static bool getBoolValue(const QSpinBox* widget) { return widget->value() > 0; } static void setBoolValue(QSpinBox* widget, bool value) { widget->setValue(value ? 1 : 0); } static int getIntValue(const QSpinBox* widget) { return widget->value(); } static void setIntValue(QSpinBox* widget, int value) { widget->setValue(value); } static float getFloatValue(const QSpinBox* widget) { return static_cast(widget->value()); } static void setFloatValue(QSpinBox* widget, float value) { widget->setValue(static_cast(value)); } static QString getStringValue(const QSpinBox* widget) { return QString::number(widget->value()); } static void setStringValue(QSpinBox* widget, const QString& value) { widget->setValue(value.toInt()); } template static void connectValueChanged(QSpinBox* widget, F func) { widget->connect(widget, QOverload::of(&QSpinBox::valueChanged), func); } }; template <> struct SettingAccessor { static bool getBoolValue(const QDoubleSpinBox* widget) { return widget->value() > 0.0; } static void setBoolValue(QDoubleSpinBox* widget, bool value) { widget->setValue(value ? 1.0 : 0.0); } static int getIntValue(const QDoubleSpinBox* widget) { return static_cast(widget->value()); } static void setIntValue(QDoubleSpinBox* widget, int value) { widget->setValue(static_cast(value)); } static float getFloatValue(const QDoubleSpinBox* widget) { return static_cast(widget->value()); } static void setFloatValue(QDoubleSpinBox* widget, float value) { widget->setValue(static_cast(value)); } static QString getStringValue(const QDoubleSpinBox* widget) { return QString::number(widget->value()); } static void setStringValue(QDoubleSpinBox* widget, const QString& value) { widget->setValue(value.toDouble()); } template static void connectValueChanged(QDoubleSpinBox* widget, F func) { widget->connect(widget, QOverload::of(&QDoubleSpinBox::valueChanged), func); } }; template <> struct SettingAccessor { static bool getBoolValue(const QAction* widget) { return widget->isChecked(); } static void setBoolValue(QAction* widget, bool value) { widget->setChecked(value); } static int getIntValue(const QAction* widget) { return widget->isChecked() ? 1 : 0; } static void setIntValue(QAction* widget, int value) { widget->setChecked(value != 0); } static float getFloatValue(const QAction* widget) { return widget->isChecked() ? 1.0f : 0.0f; } static void setFloatValue(QAction* widget, float value) { widget->setChecked(value != 0.0f); } static QString getStringValue(const QAction* widget) { return widget->isChecked() ? QStringLiteral("1") : QStringLiteral("0"); } static void setStringValue(QAction* widget, const QString& value) { widget->setChecked(value.toInt() != 0); } template static void connectValueChanged(QAction* widget, F func) { widget->connect(widget, &QAction::toggled, func); } }; /// Binds a widget's value to a setting, updating it when the value changes. template static void BindWidgetToBoolSetting(WidgetType* widget, std::string section, std::string key, bool default_value) { using Accessor = SettingAccessor; bool value = QtHost::GetBaseBoolSettingValue(section.c_str(), key.c_str(), default_value); Accessor::setBoolValue(widget, value); Accessor::connectValueChanged(widget, [widget, section, key]() { const bool new_value = Accessor::getBoolValue(widget); QtHost::SetBaseBoolSettingValue(section.c_str(), key.c_str(), new_value); g_emu_thread->applySettings(); }); } template static void BindWidgetToIntSetting(WidgetType* widget, std::string section, std::string key, int default_value, int option_offset = 0) { using Accessor = SettingAccessor; s32 value = QtHost::GetBaseIntSettingValue(section.c_str(), key.c_str(), static_cast(default_value)) - option_offset; Accessor::setIntValue(widget, static_cast(value)); Accessor::connectValueChanged(widget, [widget, section, key, option_offset]() { const int new_value = Accessor::getIntValue(widget); QtHost::SetBaseIntSettingValue(section.c_str(), key.c_str(), new_value + option_offset); g_emu_thread->applySettings(); }); } template static void BindWidgetToFloatSetting(WidgetType* widget, std::string section, std::string key, float default_value) { using Accessor = SettingAccessor; float value = QtHost::GetBaseFloatSettingValue(section.c_str(), key.c_str(), default_value); Accessor::setFloatValue(widget, value); Accessor::connectValueChanged(widget, [widget, section, key]() { const float new_value = Accessor::getFloatValue(widget); QtHost::SetBaseFloatSettingValue(section.c_str(), key.c_str(), new_value); g_emu_thread->applySettings(); }); } template static void BindWidgetToNormalizedSetting(WidgetType* widget, std::string section, std::string key, float range, float default_value) { using Accessor = SettingAccessor; float value = QtHost::GetBaseFloatSettingValue(section.c_str(), key.c_str(), default_value); Accessor::setIntValue(widget, static_cast(value * range)); Accessor::connectValueChanged(widget, [widget, section, key, range]() { const float new_value = (static_cast(Accessor::getIntValue(widget)) / range); QtHost::SetBaseFloatSettingValue(section.c_str(), key.c_str(), new_value); g_emu_thread->applySettings(); }); } template static void BindWidgetToStringSetting(WidgetType* widget, std::string section, std::string key, std::string default_value = std::string()) { using Accessor = SettingAccessor; std::string value = QtHost::GetBaseStringSettingValue(section.c_str(), key.c_str(), default_value.c_str()); Accessor::setStringValue(widget, QString::fromStdString(value)); Accessor::connectValueChanged(widget, [widget, section, key]() { const QString new_value = Accessor::getStringValue(widget); if (!new_value.isEmpty()) QtHost::SetBaseStringSettingValue(section.c_str(), key.c_str(), new_value.toUtf8().constData()); else QtHost::RemoveBaseSettingValue(section.c_str(), key.c_str()); g_emu_thread->applySettings(); }); } template static void BindWidgetToEnumSetting(WidgetType* widget, std::string section, std::string key, std::optional (*from_string_function)(const char* str), const char* (*to_string_function)(DataType value), DataType default_value) { using Accessor = SettingAccessor; using UnderlyingType = std::underlying_type_t; // TODO: Clean this up? const std::string old_setting_string_value = QtHost::GetBaseStringSettingValue(section.c_str(), key.c_str(), to_string_function(default_value)); const std::optional old_setting_value = from_string_function(old_setting_string_value.c_str()); if (old_setting_value.has_value()) Accessor::setIntValue(widget, static_cast(static_cast(old_setting_value.value()))); else Accessor::setIntValue(widget, static_cast(static_cast(default_value))); Accessor::connectValueChanged(widget, [widget, section, key, to_string_function]() { const DataType value = static_cast(static_cast(Accessor::getIntValue(widget))); const char* string_value = to_string_function(value); QtHost::SetBaseStringSettingValue(section.c_str(), key.c_str(), string_value); g_emu_thread->applySettings(); }); } template static void BindWidgetToEnumSetting(WidgetType* widget, std::string section, std::string key, const char** enum_names, DataType default_value) { using Accessor = SettingAccessor; using UnderlyingType = std::underlying_type_t; const std::string old_setting_string_value = QtHost::GetBaseStringSettingValue( section.c_str(), key.c_str(), enum_names[static_cast(default_value)]); UnderlyingType enum_index = static_cast(default_value); for (UnderlyingType i = 0; enum_names[i] != nullptr; i++) { if (old_setting_string_value == enum_names[i]) { enum_index = i; break; } } Accessor::setIntValue(widget, static_cast(enum_index)); Accessor::connectValueChanged(widget, [widget, section, key, enum_names]() { const UnderlyingType value = static_cast(Accessor::getIntValue(widget)); const char* string_value = enum_names[value]; QtHost::SetBaseStringSettingValue(section.c_str(), key.c_str(), string_value); g_emu_thread->applySettings(); }); } template static void BindWidgetToEnumSetting(WidgetType* widget, std::string section, std::string key, const char** enum_names, const char** enum_values, const char* default_value) { using Accessor = SettingAccessor; const std::string old_setting_string_value = QtHost::GetBaseStringSettingValue(section.c_str(), key.c_str(), default_value); for (int i = 0; enum_names[i] != nullptr; i++) widget->addItem(QString::fromUtf8(enum_names[i])); int enum_index = -1; for (int i = 0; enum_values[i] != nullptr; i++) { if (old_setting_string_value == enum_values[i]) { enum_index = i; break; } } if (enum_index >= 0) Accessor::setIntValue(widget, enum_index); Accessor::connectValueChanged(widget, [widget, section, key, enum_values]() { const int value = Accessor::getIntValue(widget); QtHost::SetBaseStringSettingValue(section.c_str(), key.c_str(), enum_values[value]); g_emu_thread->applySettings(); }); } } // namespace SettingWidgetBinder