[WIP] Qt: Add config window controls (#655)
* Qt: Add config window controls * Fix Windows build * Fix audio slider * Qt configs: Make thread-safe, properly update audio enable & renderdoc settings * Qt configs: Add `connectCheckbox` function * Qt configs: Add `connectCheckbox` function * Rename spuLayout * Add Discord RPC reloading * Allow configuring the app icon * Qt: Serialize icon & theme, properly set them * Add rnap and rcow icons * Qt: Fix forceShadergen config --------- Co-authored-by: wheremyfoodat <44909372+wheremyfoodat@users.noreply.github.com>
|
@ -280,6 +280,7 @@ set(SOURCE_FILES src/emulator.cpp src/io_file.cpp src/config.cpp
|
||||||
src/core/memory.cpp src/renderer.cpp src/core/renderer_null/renderer_null.cpp
|
src/core/memory.cpp src/renderer.cpp src/core/renderer_null/renderer_null.cpp
|
||||||
src/http_server.cpp src/stb_image_write.c src/core/cheats.cpp src/core/action_replay.cpp
|
src/http_server.cpp src/stb_image_write.c src/core/cheats.cpp src/core/action_replay.cpp
|
||||||
src/discord_rpc.cpp src/lua.cpp src/memory_mapped_file.cpp src/miniaudio.cpp src/renderdoc.cpp
|
src/discord_rpc.cpp src/lua.cpp src/memory_mapped_file.cpp src/miniaudio.cpp src/renderdoc.cpp
|
||||||
|
src/frontend_settings.cpp
|
||||||
)
|
)
|
||||||
set(CRYPTO_SOURCE_FILES src/core/crypto/aes_engine.cpp)
|
set(CRYPTO_SOURCE_FILES src/core/crypto/aes_engine.cpp)
|
||||||
set(KERNEL_SOURCE_FILES src/core/kernel/kernel.cpp src/core/kernel/resource_limits.cpp
|
set(KERNEL_SOURCE_FILES src/core/kernel/kernel.cpp src/core/kernel/resource_limits.cpp
|
||||||
|
@ -361,7 +362,7 @@ set(HEADER_FILES include/emulator.hpp include/helpers.hpp include/termcolor.hpp
|
||||||
include/PICA/pica_vert_config.hpp include/sdl_sensors.hpp include/PICA/draw_acceleration.hpp include/renderdoc.hpp
|
include/PICA/pica_vert_config.hpp include/sdl_sensors.hpp include/PICA/draw_acceleration.hpp include/renderdoc.hpp
|
||||||
include/align.hpp include/audio/aac_decoder.hpp include/PICA/pica_simd.hpp include/services/fonts.hpp
|
include/align.hpp include/audio/aac_decoder.hpp include/PICA/pica_simd.hpp include/services/fonts.hpp
|
||||||
include/audio/audio_interpolation.hpp include/audio/hle_mixer.hpp include/audio/dsp_simd.hpp
|
include/audio/audio_interpolation.hpp include/audio/hle_mixer.hpp include/audio/dsp_simd.hpp
|
||||||
include/services/dsp_firmware_db.hpp
|
include/services/dsp_firmware_db.hpp include/frontend_settings.hpp
|
||||||
)
|
)
|
||||||
|
|
||||||
cmrc_add_resource_library(
|
cmrc_add_resource_library(
|
||||||
|
@ -695,6 +696,9 @@ if(NOT BUILD_HYDRA_CORE AND NOT BUILD_LIBRETRO_CORE)
|
||||||
PREFIX "/"
|
PREFIX "/"
|
||||||
FILES
|
FILES
|
||||||
docs/img/rsob_icon.png docs/img/rstarstruck_icon.png docs/img/rpog_icon.png docs/img/rsyn_icon.png
|
docs/img/rsob_icon.png docs/img/rstarstruck_icon.png docs/img/rpog_icon.png docs/img/rsyn_icon.png
|
||||||
|
docs/img/settings_icon.png docs/img/display_icon.png docs/img/speaker_icon.png
|
||||||
|
docs/img/sparkling_icon.png docs/img/battery_icon.png docs/img/sdcard_icon.png
|
||||||
|
docs/img/rnap_icon.png docs/img/rcow_icon.png
|
||||||
)
|
)
|
||||||
else()
|
else()
|
||||||
set(FRONTEND_SOURCE_FILES src/panda_sdl/main.cpp src/panda_sdl/frontend_sdl.cpp src/panda_sdl/mappings.cpp)
|
set(FRONTEND_SOURCE_FILES src/panda_sdl/main.cpp src/panda_sdl/frontend_sdl.cpp src/panda_sdl/mappings.cpp)
|
||||||
|
|
BIN
docs/img/battery_icon.png
Normal file
After Width: | Height: | Size: 4.3 KiB |
BIN
docs/img/display_icon.png
Normal file
After Width: | Height: | Size: 166 B |
BIN
docs/img/rcow_icon.png
Normal file
After Width: | Height: | Size: 32 KiB |
BIN
docs/img/rnap_icon.png
Normal file
After Width: | Height: | Size: 27 KiB |
BIN
docs/img/sdcard_icon.png
Normal file
After Width: | Height: | Size: 4.3 KiB |
BIN
docs/img/settings_icon.png
Normal file
After Width: | Height: | Size: 697 B |
BIN
docs/img/sparkling_icon.png
Normal file
After Width: | Height: | Size: 549 B |
BIN
docs/img/speaker_icon.png
Normal file
After Width: | Height: | Size: 507 B |
|
@ -3,6 +3,7 @@
|
||||||
|
|
||||||
#include "audio/dsp_core.hpp"
|
#include "audio/dsp_core.hpp"
|
||||||
#include "renderer.hpp"
|
#include "renderer.hpp"
|
||||||
|
#include "frontend_settings.hpp"
|
||||||
|
|
||||||
struct AudioDeviceConfig {
|
struct AudioDeviceConfig {
|
||||||
float volumeRaw = 1.0f;
|
float volumeRaw = 1.0f;
|
||||||
|
@ -86,8 +87,9 @@ struct EmulatorConfig {
|
||||||
|
|
||||||
WindowSettings windowSettings;
|
WindowSettings windowSettings;
|
||||||
AudioDeviceConfig audioDeviceConfig;
|
AudioDeviceConfig audioDeviceConfig;
|
||||||
|
FrontendSettings frontendSettings;
|
||||||
|
|
||||||
EmulatorConfig(const std::filesystem::path& path);
|
EmulatorConfig(const std::filesystem::path& path);
|
||||||
void load();
|
void load();
|
||||||
void save();
|
void save();
|
||||||
};
|
};
|
|
@ -17,6 +17,8 @@ namespace Discord {
|
||||||
void init();
|
void init();
|
||||||
void update(RPCStatus status, const std::string& title);
|
void update(RPCStatus status, const std::string& title);
|
||||||
void stop();
|
void stop();
|
||||||
|
|
||||||
|
bool running() const { return enabled; }
|
||||||
};
|
};
|
||||||
} // namespace Discord
|
} // namespace Discord
|
||||||
|
|
||||||
|
|
|
@ -118,6 +118,9 @@ class Emulator {
|
||||||
void setOutputSize(u32 width, u32 height) { gpu.setOutputSize(width, height); }
|
void setOutputSize(u32 width, u32 height) { gpu.setOutputSize(width, height); }
|
||||||
void deinitGraphicsContext() { gpu.deinitGraphicsContext(); }
|
void deinitGraphicsContext() { gpu.deinitGraphicsContext(); }
|
||||||
|
|
||||||
|
// Reloads some settings that require special handling, such as audio enable
|
||||||
|
void reloadSettings();
|
||||||
|
|
||||||
EmulatorConfig& getConfig() { return config; }
|
EmulatorConfig& getConfig() { return config; }
|
||||||
Cheats& getCheats() { return cheats; }
|
Cheats& getCheats() { return cheats; }
|
||||||
ServiceManager& getServiceManager() { return kernel.getServiceManager(); }
|
ServiceManager& getServiceManager() { return kernel.getServiceManager(); }
|
||||||
|
|
32
include/frontend_settings.hpp
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
#pragma once
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
// Some UI settings that aren't fully frontend-dependent. Note: Not all frontends will support the same settings.
|
||||||
|
// Note: Any enums should ideally be ordered in the same order we want to show them in UI dropdown menus, so that we can cast indices to enums
|
||||||
|
// directly.
|
||||||
|
struct FrontendSettings {
|
||||||
|
enum class Theme : int {
|
||||||
|
System = 0,
|
||||||
|
Light = 1,
|
||||||
|
Dark = 2,
|
||||||
|
GreetingsCat = 3,
|
||||||
|
Cream = 4,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Different panda-themed window icons
|
||||||
|
enum class WindowIcon : int {
|
||||||
|
Rpog = 0,
|
||||||
|
Rsyn = 1,
|
||||||
|
Rnap = 2,
|
||||||
|
Rcow = 3,
|
||||||
|
};
|
||||||
|
|
||||||
|
Theme theme = Theme::Dark;
|
||||||
|
WindowIcon icon = WindowIcon::Rpog;
|
||||||
|
|
||||||
|
static Theme themeFromString(std::string inString);
|
||||||
|
static const char* themeToString(Theme theme);
|
||||||
|
|
||||||
|
static WindowIcon iconFromString(std::string inString);
|
||||||
|
static const char* iconToString(WindowIcon icon);
|
||||||
|
};
|
|
@ -1,30 +1,56 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
|
#include <QCheckBox>
|
||||||
#include <QComboBox>
|
#include <QComboBox>
|
||||||
#include <QDialog>
|
#include <QDialog>
|
||||||
|
#include <QListWidget>
|
||||||
#include <QPalette>
|
#include <QPalette>
|
||||||
|
#include <QStackedWidget>
|
||||||
|
#include <QTextEdit>
|
||||||
#include <QWidget>
|
#include <QWidget>
|
||||||
#include <QtWidgets>
|
#include <QtWidgets>
|
||||||
|
#include <array>
|
||||||
|
#include <functional>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#include "emulator.hpp"
|
||||||
|
#include "frontend_settings.hpp"
|
||||||
|
|
||||||
class ConfigWindow : public QDialog {
|
class ConfigWindow : public QDialog {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
private:
|
private:
|
||||||
enum class Theme : int {
|
using ConfigCallback = std::function<void()>;
|
||||||
System = 0,
|
using IconCallback = std::function<void(const QString&)>;
|
||||||
Light = 1,
|
|
||||||
Dark = 2,
|
|
||||||
GreetingsCat = 3,
|
|
||||||
Cream = 4,
|
|
||||||
};
|
|
||||||
|
|
||||||
Theme currentTheme;
|
using Theme = FrontendSettings::Theme;
|
||||||
QComboBox* themeSelect = nullptr;
|
using WindowIcon = FrontendSettings::WindowIcon;
|
||||||
|
|
||||||
void setTheme(Theme theme);
|
QTextEdit* helpText = nullptr;
|
||||||
|
QListWidget* widgetList = nullptr;
|
||||||
|
QStackedWidget* widgetContainer = nullptr;
|
||||||
|
|
||||||
|
static constexpr size_t settingWidgetCount = 6;
|
||||||
|
std::array<QString, settingWidgetCount> helpTexts;
|
||||||
|
|
||||||
|
// The config class holds a copy of the emulator config which it edits and sends
|
||||||
|
// over to the emulator in a thread-safe manner
|
||||||
|
EmulatorConfig config;
|
||||||
|
|
||||||
|
ConfigCallback updateConfig;
|
||||||
|
IconCallback updateIcon;
|
||||||
|
|
||||||
|
void addWidget(QWidget* widget, QString title, QString icon, QString helpText);
|
||||||
|
void setTheme(FrontendSettings::Theme theme);
|
||||||
|
void setIcon(FrontendSettings::WindowIcon icon);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ConfigWindow(QWidget* parent = nullptr);
|
ConfigWindow(ConfigCallback configCallback, IconCallback iconCallback, const EmulatorConfig& config, QWidget* parent = nullptr);
|
||||||
~ConfigWindow();
|
~ConfigWindow();
|
||||||
|
|
||||||
|
EmulatorConfig& getConfig() { return config; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
Emulator* emu;
|
||||||
};
|
};
|
||||||
|
|
|
@ -51,6 +51,7 @@ class MainWindow : public QMainWindow {
|
||||||
ReleaseTouchscreen,
|
ReleaseTouchscreen,
|
||||||
ReloadUbershader,
|
ReloadUbershader,
|
||||||
SetScreenSize,
|
SetScreenSize,
|
||||||
|
UpdateConfig,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Tagged union representing our message queue messages
|
// Tagged union representing our message queue messages
|
||||||
|
|
|
@ -23,6 +23,9 @@ namespace Renderdoc {
|
||||||
// Sets output directory for captures
|
// Sets output directory for captures
|
||||||
void setOutputDir(const std::string& path, const std::string& prefix);
|
void setOutputDir(const std::string& path, const std::string& prefix);
|
||||||
|
|
||||||
|
// Returns whether Renderdoc has been loaded
|
||||||
|
bool isLoaded();
|
||||||
|
|
||||||
// Returns whether we've compiled with Renderdoc support
|
// Returns whether we've compiled with Renderdoc support
|
||||||
static constexpr bool isSupported() { return true; }
|
static constexpr bool isSupported() { return true; }
|
||||||
} // namespace Renderdoc
|
} // namespace Renderdoc
|
||||||
|
@ -34,6 +37,7 @@ namespace Renderdoc {
|
||||||
static void triggerCapture() { Helpers::panic("Tried to trigger a Renderdoc capture while support for renderdoc is disabled"); }
|
static void triggerCapture() { Helpers::panic("Tried to trigger a Renderdoc capture while support for renderdoc is disabled"); }
|
||||||
static void setOutputDir(const std::string& path, const std::string& prefix) {}
|
static void setOutputDir(const std::string& path, const std::string& prefix) {}
|
||||||
static constexpr bool isSupported() { return false; }
|
static constexpr bool isSupported() { return false; }
|
||||||
|
static constexpr bool isLoaded() { return false; }
|
||||||
} // namespace Renderdoc
|
} // namespace Renderdoc
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -130,6 +130,16 @@ void EmulatorConfig::load() {
|
||||||
sdWriteProtected = toml::find_or<toml::boolean>(sd, "WriteProtectVirtualSD", false);
|
sdWriteProtected = toml::find_or<toml::boolean>(sd, "WriteProtectVirtualSD", false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (data.contains("UI")) {
|
||||||
|
auto uiResult = toml::expect<toml::value>(data.at("UI"));
|
||||||
|
if (uiResult.is_ok()) {
|
||||||
|
auto ui = uiResult.unwrap();
|
||||||
|
|
||||||
|
frontendSettings.theme = FrontendSettings::themeFromString(toml::find_or<std::string>(ui, "Theme", "dark"));
|
||||||
|
frontendSettings.icon = FrontendSettings::iconFromString(toml::find_or<std::string>(ui, "WindowIcon", "rpog"));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void EmulatorConfig::save() {
|
void EmulatorConfig::save() {
|
||||||
|
@ -186,6 +196,9 @@ void EmulatorConfig::save() {
|
||||||
data["SD"]["UseVirtualSD"] = sdCardInserted;
|
data["SD"]["UseVirtualSD"] = sdCardInserted;
|
||||||
data["SD"]["WriteProtectVirtualSD"] = sdWriteProtected;
|
data["SD"]["WriteProtectVirtualSD"] = sdWriteProtected;
|
||||||
|
|
||||||
|
data["UI"]["Theme"] = std::string(FrontendSettings::themeToString(frontendSettings.theme));
|
||||||
|
data["UI"]["WindowIcon"] = std::string(FrontendSettings::iconToString(frontendSettings.icon));
|
||||||
|
|
||||||
std::ofstream file(path, std::ios::out);
|
std::ofstream file(path, std::ios::out);
|
||||||
file << data;
|
file << data;
|
||||||
file.close();
|
file.close();
|
||||||
|
|
|
@ -444,3 +444,24 @@ void Emulator::loadRenderdoc() {
|
||||||
Renderdoc::loadRenderdoc();
|
Renderdoc::loadRenderdoc();
|
||||||
Renderdoc::setOutputDir(capturePath, "");
|
Renderdoc::setOutputDir(capturePath, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Emulator::reloadSettings() {
|
||||||
|
setAudioEnabled(config.audioEnabled);
|
||||||
|
|
||||||
|
if (Renderdoc::isSupported() && config.enableRenderdoc && !Renderdoc::isLoaded()) {
|
||||||
|
loadRenderdoc();
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef PANDA3DS_ENABLE_DISCORD_RPC
|
||||||
|
// Reload RPC setting if we're compiling with RPC support
|
||||||
|
|
||||||
|
if (discordRpc.running() != config.discordRpcEnabled) {
|
||||||
|
if (config.discordRpcEnabled) {
|
||||||
|
discordRpc.init();
|
||||||
|
updateDiscord();
|
||||||
|
} else {
|
||||||
|
discordRpc.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
64
src/frontend_settings.cpp
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
#include "frontend_settings.hpp"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <cctype>
|
||||||
|
#include <unordered_map>
|
||||||
|
|
||||||
|
// Frontend setting serialization/deserialization functions
|
||||||
|
|
||||||
|
FrontendSettings::Theme FrontendSettings::themeFromString(std::string inString) {
|
||||||
|
// Transform to lower-case to make the setting case-insensitive
|
||||||
|
std::transform(inString.begin(), inString.end(), inString.begin(), [](unsigned char c) { return std::tolower(c); });
|
||||||
|
|
||||||
|
static const std::unordered_map<std::string, Theme> map = {
|
||||||
|
{"system", Theme::System}, {"light", Theme::Light}, {"dark", Theme::Dark}, {"greetingscat", Theme::GreetingsCat}, {"cream", Theme::Cream},
|
||||||
|
};
|
||||||
|
|
||||||
|
if (auto search = map.find(inString); search != map.end()) {
|
||||||
|
return search->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default to dark theme
|
||||||
|
return Theme::Dark;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* FrontendSettings::themeToString(Theme theme) {
|
||||||
|
switch (theme) {
|
||||||
|
case Theme::System: return "system";
|
||||||
|
case Theme::Light: return "light";
|
||||||
|
case Theme::GreetingsCat: return "greetingscat";
|
||||||
|
case Theme::Cream: return "cream";
|
||||||
|
|
||||||
|
case Theme::Dark:
|
||||||
|
default: return "dark";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FrontendSettings::WindowIcon FrontendSettings::iconFromString(std::string inString) { // Transform to lower-case to make the setting case-insensitive
|
||||||
|
std::transform(inString.begin(), inString.end(), inString.begin(), [](unsigned char c) { return std::tolower(c); });
|
||||||
|
|
||||||
|
static const std::unordered_map<std::string, WindowIcon> map = {
|
||||||
|
{"rpog", WindowIcon::Rpog},
|
||||||
|
{"rsyn", WindowIcon::Rsyn},
|
||||||
|
{"rcow", WindowIcon::Rcow},
|
||||||
|
{"rnap", WindowIcon::Rnap},
|
||||||
|
};
|
||||||
|
|
||||||
|
if (auto search = map.find(inString); search != map.end()) {
|
||||||
|
return search->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default to the icon rpog icon
|
||||||
|
return WindowIcon::Rpog;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* FrontendSettings::iconToString(WindowIcon icon) {
|
||||||
|
switch (icon) {
|
||||||
|
case WindowIcon::Rsyn: return "rsyn";
|
||||||
|
case WindowIcon::Rcow: return "rcow";
|
||||||
|
case WindowIcon::Rnap: return "rnap";
|
||||||
|
|
||||||
|
case WindowIcon::Rpog:
|
||||||
|
default: return "rpog";
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,26 +1,286 @@
|
||||||
#include "panda_qt/config_window.hpp"
|
#include "panda_qt/config_window.hpp"
|
||||||
|
|
||||||
ConfigWindow::ConfigWindow(QWidget* parent) : QDialog(parent) {
|
ConfigWindow::ConfigWindow(ConfigCallback configCallback, IconCallback iconCallback, const EmulatorConfig& emuConfig, QWidget* parent)
|
||||||
|
: QDialog(parent), config(emuConfig) {
|
||||||
setWindowTitle(tr("Configuration"));
|
setWindowTitle(tr("Configuration"));
|
||||||
|
|
||||||
|
updateConfig = std::move(configCallback);
|
||||||
|
updateIcon = std::move(iconCallback);
|
||||||
|
|
||||||
// Set up theme selection
|
// Set up theme selection
|
||||||
setTheme(Theme::Dark);
|
setTheme(config.frontendSettings.theme);
|
||||||
themeSelect = new QComboBox(this);
|
setIcon(config.frontendSettings.icon);
|
||||||
|
|
||||||
|
// Initialize the widget list and the widget container widgets
|
||||||
|
widgetList = new QListWidget(this);
|
||||||
|
widgetContainer = new QStackedWidget(this);
|
||||||
|
|
||||||
|
helpText = new QTextEdit(this);
|
||||||
|
helpText->setReadOnly(true);
|
||||||
|
|
||||||
|
helpText->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed);
|
||||||
|
helpText->setFixedHeight(50);
|
||||||
|
|
||||||
|
widgetList->setMinimumWidth(100);
|
||||||
|
widgetList->setMaximumWidth(100);
|
||||||
|
widgetList->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
|
||||||
|
widgetList->setPalette(QPalette(QColor(25, 25, 25)));
|
||||||
|
|
||||||
|
widgetList->setCurrentRow(0);
|
||||||
|
widgetContainer->setCurrentIndex(0);
|
||||||
|
|
||||||
|
connect(widgetList, &QListWidget::currentRowChanged, this, [&](int row) {
|
||||||
|
widgetContainer->setCurrentIndex(row);
|
||||||
|
helpText->setText(helpTexts[row]);
|
||||||
|
});
|
||||||
|
|
||||||
|
auto connectCheckbox = [&](QCheckBox* checkbox, bool& setting) {
|
||||||
|
checkbox->setChecked(setting);
|
||||||
|
|
||||||
|
connect(checkbox, &QCheckBox::toggled, this, [&](bool checked) {
|
||||||
|
setting = checked;
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
QVBoxLayout* mainLayout = new QVBoxLayout();
|
||||||
|
QHBoxLayout* hLayout = new QHBoxLayout();
|
||||||
|
|
||||||
|
// Set up widget layouts
|
||||||
|
setLayout(mainLayout);
|
||||||
|
mainLayout->addLayout(hLayout);
|
||||||
|
mainLayout->addWidget(helpText);
|
||||||
|
|
||||||
|
hLayout->setAlignment(Qt::AlignLeft);
|
||||||
|
hLayout->addWidget(widgetList);
|
||||||
|
hLayout->addWidget(widgetContainer);
|
||||||
|
|
||||||
|
// Interface settings
|
||||||
|
QGroupBox* guiGroupBox = new QGroupBox(tr("Interface Settings"), this);
|
||||||
|
QFormLayout* guiLayout = new QFormLayout(guiGroupBox);
|
||||||
|
guiLayout->setHorizontalSpacing(20);
|
||||||
|
guiLayout->setVerticalSpacing(10);
|
||||||
|
|
||||||
|
QComboBox* themeSelect = new QComboBox();
|
||||||
themeSelect->addItem(tr("System"));
|
themeSelect->addItem(tr("System"));
|
||||||
themeSelect->addItem(tr("Light"));
|
themeSelect->addItem(tr("Light"));
|
||||||
themeSelect->addItem(tr("Dark"));
|
themeSelect->addItem(tr("Dark"));
|
||||||
themeSelect->addItem(tr("Greetings Cat"));
|
themeSelect->addItem(tr("Greetings Cat"));
|
||||||
themeSelect->addItem(tr("Cream"));
|
themeSelect->addItem(tr("Cream"));
|
||||||
themeSelect->setCurrentIndex(static_cast<int>(currentTheme));
|
themeSelect->setCurrentIndex(static_cast<int>(config.frontendSettings.theme));
|
||||||
|
connect(themeSelect, &QComboBox::currentIndexChanged, this, [&](int index) {
|
||||||
|
config.frontendSettings.theme = static_cast<Theme>(index);
|
||||||
|
setTheme(static_cast<Theme>(index));
|
||||||
|
|
||||||
themeSelect->setGeometry(40, 40, 100, 50);
|
updateConfig();
|
||||||
themeSelect->show();
|
});
|
||||||
connect(themeSelect, &QComboBox::currentIndexChanged, this, [&](int index) { setTheme(static_cast<Theme>(index)); });
|
guiLayout->addRow(tr("Color theme"), themeSelect);
|
||||||
|
|
||||||
|
QComboBox* iconSelect = new QComboBox();
|
||||||
|
iconSelect->addItem(tr("Happy panda"));
|
||||||
|
iconSelect->addItem(tr("Happy panda (colourful)"));
|
||||||
|
iconSelect->addItem(tr("Sleepy panda"));
|
||||||
|
iconSelect->addItem(tr("Cow panda"));
|
||||||
|
iconSelect->setCurrentIndex(static_cast<int>(config.frontendSettings.icon));
|
||||||
|
|
||||||
|
connect(iconSelect, &QComboBox::currentIndexChanged, this, [&](int index) {
|
||||||
|
config.frontendSettings.icon = static_cast<WindowIcon>(index);
|
||||||
|
setIcon(static_cast<WindowIcon>(index));
|
||||||
|
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
guiLayout->addRow(tr("Window icon"), iconSelect);
|
||||||
|
|
||||||
|
QCheckBox* showAppVersion = new QCheckBox(tr("Show version on window title"));
|
||||||
|
connectCheckbox(showAppVersion, config.windowSettings.showAppVersion);
|
||||||
|
guiLayout->addRow(showAppVersion);
|
||||||
|
|
||||||
|
QCheckBox* rememberPosition = new QCheckBox(tr("Remember window position"));
|
||||||
|
connectCheckbox(rememberPosition, config.windowSettings.rememberPosition);
|
||||||
|
guiLayout->addRow(rememberPosition);
|
||||||
|
|
||||||
|
// General settings
|
||||||
|
QGroupBox* genGroupBox = new QGroupBox(tr("General Settings"), this);
|
||||||
|
QFormLayout* genLayout = new QFormLayout(genGroupBox);
|
||||||
|
genLayout->setHorizontalSpacing(20);
|
||||||
|
genLayout->setVerticalSpacing(10);
|
||||||
|
|
||||||
|
QLineEdit* defaultRomPath = new QLineEdit;
|
||||||
|
defaultRomPath->setText(QString::fromStdU16String(config.defaultRomPath.u16string()));
|
||||||
|
connect(defaultRomPath, &QLineEdit::textChanged, this, [&](const QString& text) {
|
||||||
|
config.defaultRomPath = text.toStdString();
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
QPushButton* browseRomPath = new QPushButton(tr("Browse..."));
|
||||||
|
browseRomPath->setAutoDefault(false);
|
||||||
|
connect(browseRomPath, &QPushButton::pressed, this, [&, defaultRomPath]() {
|
||||||
|
QString newPath = QFileDialog::getExistingDirectory(
|
||||||
|
this, tr("Select Directory"), QString::fromStdU16String(config.defaultRomPath.u16string()),
|
||||||
|
QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks
|
||||||
|
);
|
||||||
|
if (!newPath.isEmpty()) {
|
||||||
|
defaultRomPath->setText(newPath);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
QHBoxLayout* romLayout = new QHBoxLayout;
|
||||||
|
romLayout->setSpacing(4);
|
||||||
|
romLayout->addWidget(defaultRomPath);
|
||||||
|
romLayout->addWidget(browseRomPath);
|
||||||
|
genLayout->addRow(tr("Default ROMs path"), romLayout);
|
||||||
|
|
||||||
|
QCheckBox* discordRpcEnabled = new QCheckBox(tr("Enable Discord RPC"));
|
||||||
|
connectCheckbox(discordRpcEnabled, config.discordRpcEnabled);
|
||||||
|
genLayout->addRow(discordRpcEnabled);
|
||||||
|
|
||||||
|
QCheckBox* usePortableBuild = new QCheckBox(tr("Use portable build"));
|
||||||
|
connectCheckbox(usePortableBuild, config.usePortableBuild);
|
||||||
|
genLayout->addRow(usePortableBuild);
|
||||||
|
|
||||||
|
QCheckBox* printAppVersion = new QCheckBox(tr("Print version in console output"));
|
||||||
|
connectCheckbox(printAppVersion, config.printAppVersion);
|
||||||
|
genLayout->addRow(printAppVersion);
|
||||||
|
|
||||||
|
// Graphics settings
|
||||||
|
QGroupBox* gpuGroupBox = new QGroupBox(tr("Graphics Settings"), this);
|
||||||
|
QFormLayout* gpuLayout = new QFormLayout(gpuGroupBox);
|
||||||
|
gpuLayout->setHorizontalSpacing(20);
|
||||||
|
gpuLayout->setVerticalSpacing(10);
|
||||||
|
|
||||||
|
QComboBox* rendererType = new QComboBox;
|
||||||
|
rendererType->addItem(tr("Null"));
|
||||||
|
rendererType->addItem(tr("OpenGL"));
|
||||||
|
rendererType->addItem(tr("Vulkan"));
|
||||||
|
rendererType->setCurrentIndex(static_cast<int>(config.rendererType));
|
||||||
|
connect(rendererType, &QComboBox::currentIndexChanged, this, [&](int index) {
|
||||||
|
config.rendererType = static_cast<RendererType>(index);
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
gpuLayout->addRow(tr("GPU renderer"), rendererType);
|
||||||
|
|
||||||
|
QCheckBox* enableRenderdoc = new QCheckBox(tr("Enable Renderdoc"));
|
||||||
|
connectCheckbox(enableRenderdoc, config.enableRenderdoc);
|
||||||
|
gpuLayout->addRow(enableRenderdoc);
|
||||||
|
|
||||||
|
QCheckBox* shaderJitEnabled = new QCheckBox(tr("Enable shader JIT"));
|
||||||
|
connectCheckbox(shaderJitEnabled, config.shaderJitEnabled);
|
||||||
|
gpuLayout->addRow(shaderJitEnabled);
|
||||||
|
|
||||||
|
QCheckBox* vsyncEnabled = new QCheckBox(tr("Enable VSync"));
|
||||||
|
connectCheckbox(vsyncEnabled, config.vsyncEnabled);
|
||||||
|
gpuLayout->addRow(vsyncEnabled);
|
||||||
|
|
||||||
|
QCheckBox* useUbershaders = new QCheckBox(tr("Use ubershaders (No stutter, maybe slower)"));
|
||||||
|
connectCheckbox(useUbershaders, config.useUbershaders);
|
||||||
|
gpuLayout->addRow(useUbershaders);
|
||||||
|
|
||||||
|
QCheckBox* accurateShaderMul = new QCheckBox(tr("Accurate shader multiplication"));
|
||||||
|
connectCheckbox(accurateShaderMul, config.accurateShaderMul);
|
||||||
|
gpuLayout->addRow(accurateShaderMul);
|
||||||
|
|
||||||
|
QCheckBox* accelerateShaders = new QCheckBox(tr("Accelerate shaders"));
|
||||||
|
connectCheckbox(accelerateShaders, config.accelerateShaders);
|
||||||
|
gpuLayout->addRow(accelerateShaders);
|
||||||
|
|
||||||
|
QCheckBox* forceShadergenForLights = new QCheckBox(tr("Force shadergen when rendering lights"));
|
||||||
|
connectCheckbox(forceShadergenForLights, config.forceShadergenForLights);
|
||||||
|
gpuLayout->addRow(forceShadergenForLights);
|
||||||
|
|
||||||
|
QSpinBox* lightShadergenThreshold = new QSpinBox;
|
||||||
|
lightShadergenThreshold->setRange(1, 8);
|
||||||
|
lightShadergenThreshold->setValue(config.lightShadergenThreshold);
|
||||||
|
connect(lightShadergenThreshold, &QSpinBox::valueChanged, this, [&](int value) {
|
||||||
|
config.lightShadergenThreshold = static_cast<int>(value);
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
gpuLayout->addRow(tr("Light threshold for forcing shadergen"), lightShadergenThreshold);
|
||||||
|
|
||||||
|
// Audio settings
|
||||||
|
QGroupBox* spuGroupBox = new QGroupBox(tr("Audio Settings"), this);
|
||||||
|
QFormLayout* audioLayout = new QFormLayout(spuGroupBox);
|
||||||
|
audioLayout->setHorizontalSpacing(20);
|
||||||
|
audioLayout->setVerticalSpacing(10);
|
||||||
|
|
||||||
|
QComboBox* dspType = new QComboBox;
|
||||||
|
dspType->addItem(tr("Null"));
|
||||||
|
dspType->addItem(tr("LLE"));
|
||||||
|
dspType->addItem(tr("HLE"));
|
||||||
|
dspType->setCurrentIndex(static_cast<int>(config.dspType));
|
||||||
|
connect(dspType, &QComboBox::currentIndexChanged, this, [&](int index) {
|
||||||
|
config.dspType = static_cast<Audio::DSPCore::Type>(index);
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
audioLayout->addRow(tr("DSP emulation"), dspType);
|
||||||
|
|
||||||
|
QCheckBox* audioEnabled = new QCheckBox(tr("Enable audio"));
|
||||||
|
connectCheckbox(audioEnabled, config.audioEnabled);
|
||||||
|
audioLayout->addRow(audioEnabled);
|
||||||
|
|
||||||
|
QCheckBox* aacEnabled = new QCheckBox(tr("Enable AAC audio"));
|
||||||
|
connectCheckbox(aacEnabled, config.aacEnabled);
|
||||||
|
audioLayout->addRow(aacEnabled);
|
||||||
|
|
||||||
|
QCheckBox* printDSPFirmware = new QCheckBox(tr("Print DSP firmware"));
|
||||||
|
connectCheckbox(printDSPFirmware, config.printDSPFirmware);
|
||||||
|
audioLayout->addRow(printDSPFirmware);
|
||||||
|
|
||||||
|
QCheckBox* muteAudio = new QCheckBox(tr("Mute audio device"));
|
||||||
|
connectCheckbox(muteAudio, config.audioDeviceConfig.muteAudio);
|
||||||
|
audioLayout->addRow(muteAudio);
|
||||||
|
|
||||||
|
QSpinBox* volumeRaw = new QSpinBox();
|
||||||
|
volumeRaw->setRange(0, 200);
|
||||||
|
volumeRaw->setValue(config.audioDeviceConfig.volumeRaw* 100);
|
||||||
|
connect(volumeRaw, &QSpinBox::valueChanged, this, [&](int value) {
|
||||||
|
config.audioDeviceConfig.volumeRaw = static_cast<float>(value) / 100.0f;
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
audioLayout->addRow(tr("Audio device volume"), volumeRaw);
|
||||||
|
|
||||||
|
// Battery settings
|
||||||
|
QGroupBox* batGroupBox = new QGroupBox(tr("Battery Settings"), this);
|
||||||
|
QFormLayout* batLayout = new QFormLayout(batGroupBox);
|
||||||
|
batLayout->setHorizontalSpacing(20);
|
||||||
|
batLayout->setVerticalSpacing(10);
|
||||||
|
|
||||||
|
QSpinBox* batteryPercentage = new QSpinBox;
|
||||||
|
batteryPercentage->setRange(1, 100);
|
||||||
|
batteryPercentage->setValue(config.batteryPercentage);
|
||||||
|
connect(batteryPercentage, &QSpinBox::valueChanged, this, [&](int value) {
|
||||||
|
config.batteryPercentage = static_cast<int>(value);
|
||||||
|
updateConfig();
|
||||||
|
});
|
||||||
|
batLayout->addRow(tr("Battery percentage"), batteryPercentage);
|
||||||
|
|
||||||
|
QCheckBox* chargerPlugged = new QCheckBox(tr("Charger plugged"));
|
||||||
|
connectCheckbox(chargerPlugged, config.chargerPlugged);
|
||||||
|
batLayout->addRow(chargerPlugged);
|
||||||
|
|
||||||
|
// SD Card settings
|
||||||
|
QGroupBox* sdcGroupBox = new QGroupBox(tr("SD Card Settings"), this);
|
||||||
|
QFormLayout* sdcLayout = new QFormLayout(sdcGroupBox);
|
||||||
|
sdcLayout->setHorizontalSpacing(20);
|
||||||
|
sdcLayout->setVerticalSpacing(10);
|
||||||
|
|
||||||
|
QCheckBox* sdCardInserted = new QCheckBox(tr("Enable virtual SD card"));
|
||||||
|
connectCheckbox(sdCardInserted, config.sdCardInserted);
|
||||||
|
sdcLayout->addRow(sdCardInserted);
|
||||||
|
|
||||||
|
QCheckBox* sdWriteProtected = new QCheckBox(tr("Write protect virtual SD card"));
|
||||||
|
connectCheckbox(sdWriteProtected, config.sdWriteProtected);
|
||||||
|
sdcLayout->addRow(sdWriteProtected);
|
||||||
|
|
||||||
|
// Add all our settings widgets to our widget list
|
||||||
|
addWidget(guiGroupBox, tr("Interface"), ":/docs/img/sparkling_icon.png", tr("User Interface settings"));
|
||||||
|
addWidget(genGroupBox, tr("General"), ":/docs/img/settings_icon.png", tr("General emulator settings"));
|
||||||
|
addWidget(gpuGroupBox, tr("Graphics"), ":/docs/img/display_icon.png", tr("Graphics emulation and output settings"));
|
||||||
|
addWidget(spuGroupBox, tr("Audio"), ":/docs/img/speaker_icon.png", tr("Audio emulation and output settings"));
|
||||||
|
addWidget(batGroupBox, tr("Battery"), ":/docs/img/battery_icon.png", tr("Battery emulation settings"));
|
||||||
|
addWidget(sdcGroupBox, tr("SD Card"), ":/docs/img/sdcard_icon.png", tr("SD Card emulation settings"));
|
||||||
|
|
||||||
|
widgetList->setCurrentRow(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ConfigWindow::setTheme(Theme theme) {
|
void ConfigWindow::setTheme(Theme theme) {
|
||||||
currentTheme = theme;
|
|
||||||
|
|
||||||
switch (theme) {
|
switch (theme) {
|
||||||
case Theme::Dark: {
|
case Theme::Dark: {
|
||||||
QApplication::setStyle(QStyleFactory::create("Fusion"));
|
QApplication::setStyle(QStyleFactory::create("Fusion"));
|
||||||
|
@ -119,4 +379,36 @@ void ConfigWindow::setTheme(Theme theme) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ConfigWindow::~ConfigWindow() { delete themeSelect; }
|
void ConfigWindow::setIcon(WindowIcon icon) {
|
||||||
|
switch (icon) {
|
||||||
|
case WindowIcon::Rsyn: updateIcon(":/docs/img/rsyn_icon.png"); break;
|
||||||
|
case WindowIcon::Rnap: updateIcon(":/docs/img/rnap_icon.png"); break;
|
||||||
|
case WindowIcon::Rcow: updateIcon(":/docs/img/rcow_icon.png"); break;
|
||||||
|
|
||||||
|
case WindowIcon::Rpog:
|
||||||
|
default: updateIcon(":/docs/img/rpog_icon.png"); break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConfigWindow::addWidget(QWidget* widget, QString title, QString icon, QString helpText) {
|
||||||
|
const int index = widgetList->count();
|
||||||
|
|
||||||
|
QListWidgetItem* item = new QListWidgetItem(widgetList);
|
||||||
|
item->setText(title);
|
||||||
|
if (!icon.isEmpty()) {
|
||||||
|
item->setIcon(QIcon::fromTheme(icon));
|
||||||
|
}
|
||||||
|
|
||||||
|
widgetContainer->addWidget(widget);
|
||||||
|
|
||||||
|
if (index >= settingWidgetCount) {
|
||||||
|
Helpers::panic("Qt: ConfigWindow::settingWidgetCount has not been updated correctly!");
|
||||||
|
}
|
||||||
|
helpTexts[index] = std::move(helpText);
|
||||||
|
}
|
||||||
|
|
||||||
|
ConfigWindow::~ConfigWindow() {
|
||||||
|
delete helpText;
|
||||||
|
delete widgetList;
|
||||||
|
delete widgetContainer;
|
||||||
|
}
|
||||||
|
|
|
@ -15,7 +15,6 @@
|
||||||
|
|
||||||
MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent), keyboardMappings(InputMappings::defaultKeyboardMappings()) {
|
MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent), keyboardMappings(InputMappings::defaultKeyboardMappings()) {
|
||||||
setWindowTitle("Alber");
|
setWindowTitle("Alber");
|
||||||
setWindowIcon(QIcon(":/docs/img/rpog_icon.png"));
|
|
||||||
|
|
||||||
// Enable drop events for loading ROMs
|
// Enable drop events for loading ROMs
|
||||||
setAcceptDrops(true);
|
setAcceptDrops(true);
|
||||||
|
@ -81,7 +80,6 @@ MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent)
|
||||||
|
|
||||||
// Set up misc objects
|
// Set up misc objects
|
||||||
aboutWindow = new AboutWindow(nullptr);
|
aboutWindow = new AboutWindow(nullptr);
|
||||||
configWindow = new ConfigWindow(this);
|
|
||||||
cheatsEditor = new CheatsWindow(emu, {}, this);
|
cheatsEditor = new CheatsWindow(emu, {}, this);
|
||||||
patchWindow = new PatchWindow(this);
|
patchWindow = new PatchWindow(this);
|
||||||
luaEditor = new TextEditorWindow(this, "script.lua", "");
|
luaEditor = new TextEditorWindow(this, "script.lua", "");
|
||||||
|
@ -92,6 +90,14 @@ MainWindow::MainWindow(QApplication* app, QWidget* parent) : QMainWindow(parent)
|
||||||
shaderEditor->setText(emu->getRenderer()->getUbershader());
|
shaderEditor->setText(emu->getRenderer()->getUbershader());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
configWindow = new ConfigWindow(
|
||||||
|
[&]() {
|
||||||
|
EmulatorMessage message{.type = MessageType::UpdateConfig};
|
||||||
|
sendMessage(message);
|
||||||
|
},
|
||||||
|
[&](const QString& icon) { setWindowIcon(QIcon(icon)); }, emu->getConfig(), this
|
||||||
|
);
|
||||||
|
|
||||||
auto args = QCoreApplication::arguments();
|
auto args = QCoreApplication::arguments();
|
||||||
if (args.size() > 1) {
|
if (args.size() > 1) {
|
||||||
auto romPath = std::filesystem::current_path() / args.at(1).toStdU16String();
|
auto romPath = std::filesystem::current_path() / args.at(1).toStdU16String();
|
||||||
|
@ -410,6 +416,14 @@ void MainWindow::dispatchMessage(const EmulatorMessage& message) {
|
||||||
screen->resizeSurface(width, height);
|
screen->resizeSurface(width, height);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case MessageType::UpdateConfig:
|
||||||
|
emu->getConfig() = configWindow->getConfig();
|
||||||
|
emu->reloadSettings();
|
||||||
|
|
||||||
|
// Save new settings to disk
|
||||||
|
emu->getConfig().save();
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,8 @@ namespace Renderdoc {
|
||||||
};
|
};
|
||||||
|
|
||||||
static CaptureState captureState{CaptureState::Idle};
|
static CaptureState captureState{CaptureState::Idle};
|
||||||
|
static bool renderdocLoaded{false};
|
||||||
|
|
||||||
RENDERDOC_API_1_6_0* rdocAPI{};
|
RENDERDOC_API_1_6_0* rdocAPI{};
|
||||||
|
|
||||||
void loadRenderdoc() {
|
void loadRenderdoc() {
|
||||||
|
@ -73,6 +75,8 @@ namespace Renderdoc {
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
if (rdocAPI) {
|
if (rdocAPI) {
|
||||||
|
renderdocLoaded = true;
|
||||||
|
|
||||||
// Disable default capture keys as they suppose to trigger present-to-present capturing
|
// Disable default capture keys as they suppose to trigger present-to-present capturing
|
||||||
// and it is not what we want
|
// and it is not what we want
|
||||||
rdocAPI->SetCaptureKeys(nullptr, 0);
|
rdocAPI->SetCaptureKeys(nullptr, 0);
|
||||||
|
@ -115,5 +119,7 @@ namespace Renderdoc {
|
||||||
rdocAPI->SetCaptureFilePathTemplate((path + '\\' + prefix).c_str());
|
rdocAPI->SetCaptureFilePathTemplate((path + '\\' + prefix).c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool isLoaded() { return renderdocLoaded; }
|
||||||
} // namespace Renderdoc
|
} // namespace Renderdoc
|
||||||
#endif
|
#endif
|