#include "settings.hpp" #include "parser.hpp" #include "values.hpp" #include #include #include #include #if !(defined(_MSC_VER) && (_MSC_VER >= 1924)) && !(defined(__GNUC__) && __GNUC__ >= 11) || defined(__clang__) \ || defined(__apple_build_version__) #include #include #include #endif #include #include #include namespace Settings { namespace { template T parseNumberFromSetting(const std::string& value, std::string_view setting, std::string_view category) { T number{}; const auto result = std::from_chars(value.data(), value.data() + value.size(), number); if (result.ec != std::errc()) { throw std::system_error(std::make_error_code(result.ec), "Failed to parse number from setting [" + std::string(category) + "] " + std::string(setting) + " value \"" + value + "\""); } return number; } #if !(defined(_MSC_VER) && (_MSC_VER >= 1924)) && !(defined(__GNUC__) && __GNUC__ >= 11) || defined(__clang__) \ || defined(__apple_build_version__) template <> float parseNumberFromSetting( const std::string& value, std::string_view setting, std::string_view category) { std::istringstream iss(value); iss.imbue(std::locale::classic()); float floatValue = 0.0f; if (!(iss >> floatValue)) { throw std::system_error(errno, std::generic_category(), "Failed to parse number from setting [" + std::string(category) + "] " + std::string(setting) + " value \"" + value + "\""); } return floatValue; } template <> double parseNumberFromSetting( const std::string& value, std::string_view setting, std::string_view category) { std::istringstream iss(value); iss.imbue(std::locale::classic()); double doubleValue = 0.0; if (!(iss >> doubleValue)) { throw std::system_error(errno, std::generic_category(), "Failed to parse number from setting [" + std::string(category) + "] " + std::string(setting) + " value \"" + value + "\""); } return doubleValue; } #endif template std::string serialize(const T& value) { std::ostringstream stream; stream << value; return stream.str(); } } CategorySettingValueMap Manager::mDefaultSettings = CategorySettingValueMap(); CategorySettingValueMap Manager::mUserSettings = CategorySettingValueMap(); CategorySettingVector Manager::mChangedSettings = CategorySettingVector(); std::set> Manager::sInitialized; void Manager::clear() { mDefaultSettings.clear(); mUserSettings.clear(); mChangedSettings.clear(); } std::filesystem::path Manager::load(const Files::ConfigurationManager& cfgMgr, bool loadEditorSettings) { SettingsFileParser parser; const std::vector& paths = cfgMgr.getActiveConfigPaths(); if (paths.empty()) throw std::runtime_error("No config dirs! ConfigurationManager::readConfiguration must be called first."); // Create file name strings for either the engine or the editor. std::string defaultSettingsFile; std::string userSettingsFile; if (!loadEditorSettings) { defaultSettingsFile = "defaults.bin"; userSettingsFile = "settings.cfg"; } else { defaultSettingsFile = "defaults-cs.bin"; userSettingsFile = "openmw-cs.cfg"; } // Create the settings manager and load default settings file. const auto defaultsBin = paths.front() / defaultSettingsFile; if (!std::filesystem::exists(defaultsBin)) throw std::runtime_error("No default settings file found! Make sure the file \"" + defaultSettingsFile + "\" was properly installed."); parser.loadSettingsFile(defaultsBin, mDefaultSettings, true, false); const CategorySettingValueMap originalDefaultSettings = mDefaultSettings; // Load "settings.cfg" or "openmw-cs.cfg" from every config dir except the last one as additional default // settings. for (int i = 0; i < static_cast(paths.size()) - 1; ++i) { const auto additionalDefaults = paths[i] / userSettingsFile; if (std::filesystem::exists(additionalDefaults)) parser.loadSettingsFile(additionalDefaults, mDefaultSettings, false, true); } if (!loadEditorSettings) Settings::Values::initDefaults(); // Load "settings.cfg" or "openmw-cs.cfg" from the last config dir as user settings. This path will be used to // save modified settings. auto settingspath = paths.back() / userSettingsFile; if (std::filesystem::exists(settingspath)) parser.loadSettingsFile(settingspath, mUserSettings, false, false); if (!loadEditorSettings) Settings::Values::init(); for (const auto& [key, value] : originalDefaultSettings) if (!sInitialized.contains(key)) throw std::runtime_error("Default setting [" + key.first + "] " + key.second + " is not initialized"); return settingspath; } void Manager::saveUser(const std::filesystem::path& file) { SettingsFileParser parser; parser.saveSettingsFile(file, mUserSettings); } const std::string& Manager::getString(std::string_view setting, std::string_view category) { const auto key = std::make_pair(category, setting); CategorySettingValueMap::iterator it = mUserSettings.find(key); if (it != mUserSettings.end()) return it->second; it = mDefaultSettings.find(key); if (it != mDefaultSettings.end()) return it->second; throw std::runtime_error("Trying to retrieve a non-existing setting: [" + std::string(category) + "] " + std::string(setting) + ".\nMake sure the defaults.bin file was properly installed."); } std::vector Manager::getStringArray(std::string_view setting, std::string_view category) { // TODO: it is unclear how to handle empty value - // there is no difference between empty serialized array // and a serialized array which has one empty value std::vector values; const std::string& value = getString(setting, category); if (value.empty()) return values; Misc::StringUtils::split(value, values, ","); for (auto& item : values) Misc::StringUtils::trim(item); return values; } float Manager::getFloat(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } double Manager::getDouble(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } int Manager::getInt(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } std::uint64_t Manager::getUInt64(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } std::size_t Manager::getSize(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } unsigned Manager::getUnsigned(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } unsigned long Manager::getUnsignedLong(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } unsigned long long Manager::getUnsignedLongLong(std::string_view setting, std::string_view category) { return parseNumberFromSetting(getString(setting, category), setting, category); } bool Manager::getBool(std::string_view setting, std::string_view category) { const std::string& string = getString(setting, category); return Misc::StringUtils::ciEqual(string, "true"); } osg::Vec2f Manager::getVector2(std::string_view setting, std::string_view category) { const std::string& value = getString(setting, category); std::vector components; Misc::StringUtils::split(value, components); if (components.size() == 2) { auto x = Misc::StringUtils::toNumeric(components[0]); auto y = Misc::StringUtils::toNumeric(components[1]); if (x && y) { return { x.value(), y.value() }; } } throw std::runtime_error(std::string("Can't parse 2d vector: " + value)); } osg::Vec3f Manager::getVector3(std::string_view setting, std::string_view category) { const std::string& value = getString(setting, category); std::vector components; Misc::StringUtils::split(value, components); if (components.size() == 3) { auto x = Misc::StringUtils::toNumeric(components[0]); auto y = Misc::StringUtils::toNumeric(components[1]); auto z = Misc::StringUtils::toNumeric(components[2]); if (x && y && z) { return { x.value(), y.value(), z.value() }; } } throw std::runtime_error(std::string("Can't parse 3d vector: " + value)); } void Manager::setString(std::string_view setting, std::string_view category, const std::string& value) { auto found = mUserSettings.find(std::make_pair(category, setting)); if (found != mUserSettings.end()) { if (found->second == value) return; } CategorySettingValueMap::key_type key(category, setting); mUserSettings[key] = value; mChangedSettings.insert(std::move(key)); } void Manager::setStringArray( std::string_view setting, std::string_view category, const std::vector& value) { std::stringstream stream; // TODO: escape delimeters, new line characters, etc. for (size_t i = 0; i < value.size(); ++i) { std::string item = value[i]; Misc::StringUtils::trim(item); stream << item; if (i < value.size() - 1) stream << ","; } setString(setting, category, stream.str()); } void Manager::setInt(std::string_view setting, std::string_view category, const int value) { std::ostringstream stream; stream << value; setString(setting, category, stream.str()); } void Manager::setUInt64(std::string_view setting, std::string_view category, const std::uint64_t value) { std::ostringstream stream; stream << value; setString(setting, category, stream.str()); } void Manager::setFloat(std::string_view setting, std::string_view category, const float value) { std::ostringstream stream; stream << value; setString(setting, category, stream.str()); } void Manager::setDouble(std::string_view setting, std::string_view category, const double value) { std::ostringstream stream; stream << value; setString(setting, category, stream.str()); } void Manager::setBool(std::string_view setting, std::string_view category, const bool value) { setString(setting, category, value ? "true" : "false"); } void Manager::setVector2(std::string_view setting, std::string_view category, const osg::Vec2f value) { std::ostringstream stream; stream << value.x() << " " << value.y(); setString(setting, category, stream.str()); } void Manager::setVector3(std::string_view setting, std::string_view category, const osg::Vec3f value) { std::ostringstream stream; stream << value.x() << ' ' << value.y() << ' ' << value.z(); setString(setting, category, stream.str()); } CategorySettingVector Manager::getPendingChanges() { return mChangedSettings; } CategorySettingVector Manager::getPendingChanges(const CategorySettingVector& filter) { CategorySettingVector intersection; std::set_intersection(mChangedSettings.begin(), mChangedSettings.end(), filter.begin(), filter.end(), std::inserter(intersection, intersection.begin())); return intersection; } void Manager::resetPendingChanges() { mChangedSettings.clear(); } void Manager::resetPendingChanges(const CategorySettingVector& filter) { for (const auto& key : filter) { mChangedSettings.erase(key); } } void Manager::set(std::string_view setting, std::string_view category, int value) { setInt(setting, category, value); } void Manager::set(std::string_view setting, std::string_view category, unsigned value) { setString(setting, category, serialize(value)); } void Manager::set(std::string_view setting, std::string_view category, unsigned long value) { setString(setting, category, serialize(value)); } void Manager::set(std::string_view setting, std::string_view category, unsigned long long value) { setString(setting, category, serialize(value)); } void Manager::set(std::string_view setting, std::string_view category, float value) { setFloat(setting, category, value); } void Manager::set(std::string_view setting, std::string_view category, double value) { setDouble(setting, category, value); } void Manager::set(std::string_view setting, std::string_view category, const std::string& value) { setString(setting, category, value); } void Manager::set(std::string_view setting, std::string_view category, bool value) { setBool(setting, category, value); } void Manager::set(std::string_view setting, std::string_view category, const osg::Vec2f& value) { setVector2(setting, category, value); } void Manager::set(std::string_view setting, std::string_view category, const osg::Vec3f& value) { setVector3(setting, category, value); } void Manager::recordInit(std::string_view setting, std::string_view category) { sInitialized.emplace(category, setting); } }