nestopia/source/fltkui/setmanager.cpp
HWXLR8 5e065f55af FLTK: Add option to mute audio
Adds a mute-on-startup entry in settings, a --mute (-m) command line
parameter, 'm' hotkey to mute on the fly, and a corresponding menubar
entry.
2025-01-01 21:24:37 -05:00

278 lines
9.1 KiB
C++

/*
Copyright (c) 2012-2024 R. Danbrook
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <cstdlib>
#include <cstring>
#include <filesystem>
#include <fstream>
#include <iostream>
#include <iterator>
#include <jg/jg.h>
#include <jg/jg_nes.h>
#include "ini.h"
#include "setmanager.h"
namespace {
jg_setting_t fe_settings[] = {
{ "v_renderer", "Video Renderer (Restart)",
"0 = Modern, 1 = Legacy",
"Use Modern (Core Profile, GLES) or Legacy (Compatibility Profile) OpenGL. "
"Use Modern unless you are on extremely weak or incapable hardware.",
0, 0, 1, FLAG_FRONTEND | JG_SETTING_RESTART
},
{ "v_postproc", "Post-processing",
"0 = Nearest Neighbour, 1 = Linear, 2 = Sharp Bilinear, 3 = CRT, 4 = MMPX, 5 = OmniScale",
"Select a video post-processing effect. Advanced effects are only available to the Modern renderer.",
2, 0, 5, FLAG_FRONTEND
},
{ "v_aspect", "Aspect Ratio",
"0 = Auto, 1 = 1:1, 2 = 4:3, 3 = 5:4",
"Set the aspect ratio to the correct TV aspect (Auto), 1:1 (square pixels), 4:3, or 5:4",
0, 0, 3, FLAG_FRONTEND
},
{ "v_fullscreen", "Start in Fullscreen Mode",
"0 = Disabled, 1 = Enabled",
"Start the emulator in fullscreen mode if a valid ROM is entered on the command line",
0, 0, 1, FLAG_FRONTEND | JG_SETTING_RESTART
},
{ "v_scale", "Initial Window Scale",
"N = Window scale factor at startup",
"Set the window's initial scale factor (multiple of NES resolution)",
3, 1, 16, FLAG_FRONTEND | JG_SETTING_RESTART
},
{ "a_mute", "Start with Audio Muted",
"0 = Disabled, 1 = Enabled",
"Start the emulator with audio muted",
0, 0, 1, FLAG_FRONTEND
},
{ "a_rsqual", "Audio Resampler",
"0 = Sinc (Best), 1 = Sinc (Medium), 2 = Sinc (Fast), 3 = Zero Order Hold, 4 = Linear",
"Set the frontend's audio resampling quality. Use Sinc unless you are on extremely weak hardware.",
2, 0, 4, FLAG_FRONTEND
},
{ "m_ffspeed", "Fast-forward Speed",
"N = Fast-forward speed multiplier",
"Set the speed multiplier to run emulation faster",
2, 2, 8, FLAG_FRONTEND
},
{ "m_hidecursor", "Hide Cursor",
"0 = Disabled, 1 = Enabled",
"Hide the cursor when hovering over the UI",
0, 0, 1, FLAG_FRONTEND
},
{ "m_hidecrosshair", "Hide Crosshair",
"0 = Disabled, 1 = Enabled",
"Hide the crosshair when a Zapper is present",
0, 0, 1, FLAG_FRONTEND
},
{ "m_syncmode", "Synchronization Mode (Restart)",
"0 = Timer, 1 = VSync",
"Set the Synchronization Mode: VSync to sync to VBLANK, Timer in cases where VSync is unreliable",
0, 0, 1, FLAG_FRONTEND | JG_SETTING_RESTART
},
{ "s_crtmasktype", "CRT Mask Type",
"0 = No Mask, 1 = Aperture Grille Lite, 2 = Aperture Grille, "
"3 = Shadow Mask",
"Set the type of CRT mask. Set no mask for pure scanlines, or use Aperture "
"Grille or Shadow Mask options to mimic the appearance of a real CRT screen.",
0, 0, 3, FLAG_FRONTEND
},
{ "s_crtmaskstr", "CRT Mask Strength",
"N = CRT Mask Strength",
"Set the strength of the CRT mask",
5, 0, 10, FLAG_FRONTEND
},
{ "s_crtscanstr", "CRT Scanline Strength",
"N = CRT Scanline Strength",
"Set the strength of the scanlines",
6, 0, 10, FLAG_FRONTEND
},
{ "s_crtsharp", "CRT Sharpness",
"N = CRT Sharpness",
"Set the level of blur/sharpness",
3, 0, 10, FLAG_FRONTEND
},
{ "s_crtcurve", "CRT Curve",
"N = CRT Curvature",
"Set the level of screen curvature on the horizontal and vertical axes",
3, 0, 10, FLAG_FRONTEND
},
{ "s_crtcorner", "CRT Corner",
"N = CRT Corner",
"Set the size of the corner mask",
3, 0, 10, FLAG_FRONTEND
},
{ "s_crttcurve", "CRT Trinitron Curve",
"N = CRT Trinitron Curvature",
"Set the level of Trinitron Curvature, which reduces the curve on the vertical axis",
10, 0, 10, FLAG_FRONTEND
},
{ "l_loglevel", "Console Log Level",
"0 = Debug, 1 = Info, 2 = Warn, 3 = Error",
"Set the level of logs printed to the console. Debug shows all, Error shows only critical errors.",
1, 0, 3, FLAG_FRONTEND
},
};
jg_setting_t nullsetting;
// Input config is really better staying alive in memory in .ini form
mINI::INIStructure inputini;
}
SettingManager::SettingManager() {
// Set defaults
size_t numsettings = sizeof(fe_settings) / sizeof(jg_setting_t);
for (size_t i = 0; i < numsettings; ++i ) {
settings.push_back(&fe_settings[i]);
}
// Create config directory
if (const char *env_xdg_config = std::getenv("XDG_CONFIG_HOME")) {
confpath = std::string(env_xdg_config) + "/nestopia";
}
else {
confpath = std::string(std::getenv("HOME")) + "/.config/nestopia";
}
// Create the directory if it does not exist
std::filesystem::create_directories(confpath);
}
void SettingManager::read(JGManager& jgm) {
// Read in any settings
mINI::INIFile file(confpath + "/nestopia.conf");
mINI::INIStructure ini;
file.read(ini);
for (const auto& setting : settings) {
std::string& strval = ini["frontend"][setting->name];
if (strval.empty()) {
continue;
}
int val = std::stoi(strval);
if (val >= setting->min && val <= setting->max) {
setting->val = val;
}
}
for (const auto& setting : *jgm.get_settings()) {
std::string& strval = ini["nestopia"][setting->name];
if (strval.empty()) {
continue;
}
int val = std::stoi(strval);
if (val >= setting->min && val <= setting->max) {
setting->val = val;
}
}
jgm.rehash();
// Read input config
mINI::INIFile inputfile(confpath + "/input.conf");
inputfile.read(inputini);
}
void SettingManager::write(JGManager& jgm) {
std::string filepath(confpath + "/nestopia.conf");
std::ofstream os(filepath);
if (!os.is_open()) {
return;
}
os << "; Nestopia UE Configuration File\n\n";
// Write out frontend settings
os << "[frontend]\n";
for (const auto& setting : settings) {
os << "; " << setting->desc << "\n";
os << "; " << setting->opts << "\n";
os << setting->name << " = " << setting->val << "\n\n";
}
// Write out emulator core settings
os << "[nestopia]\n";
for (const auto& setting : *jgm.get_settings()) {
os << "; " << setting->desc << "\n";
os << "; " << setting->opts << "\n";
os << setting->name << " = " << setting->val << "\n\n";
}
os.close();
mINI::INIFile inputfile(confpath + "/input.conf");
inputfile.write(inputini, true);
/*for (auto const& it : inputini) {
auto const& section = it.first;
auto const& collection = it.second;
std::cout << "[" << section << "]" << std::endl;
for (auto const& it2 : collection) {
auto const& key = it2.first;
auto const& value = it2.second;
std::cout << key << " = " << value << std::endl;
}
}*/
}
std::vector<jg_setting_t*> *SettingManager::get_settings() {
return &settings;
}
jg_setting_t* SettingManager::get_setting(std::string name) {
size_t numsettings = sizeof(fe_settings) / sizeof(jg_setting_t);
for (size_t i = 0; i < numsettings; ++i ) {
if (std::string(fe_settings[i].name) == name) {
return &fe_settings[i];
}
}
return &nullsetting;
}
std::string& SettingManager::get_input(std::string name, std::string def) {
return inputini[name][def];
}
void SettingManager::set_input(std::string name, std::string def, std::string val) {
inputini[name][def] = val;
}