#include "launchersettings.hpp" #include #include #include #include #include #include #include #include #include #include "gamesettings.hpp" namespace Config { namespace { constexpr char sSettingsSection[] = "Settings"; constexpr char sGeneralSection[] = "General"; constexpr char sProfilesSection[] = "Profiles"; constexpr char sImporterSection[] = "Importer"; constexpr char sLanguageKey[] = "language"; constexpr char sCurrentProfileKey[] = "currentprofile"; constexpr char sDataKey[] = "data"; constexpr char sArchiveKey[] = "fallback-archive"; constexpr char sContentKey[] = "content"; constexpr char sFirstRunKey[] = "firstrun"; constexpr char sImportContentSetupKey[] = "importcontentsetup"; constexpr char sImportFontSetupKey[] = "importfontsetup"; constexpr char sMainWindowWidthKey[] = "MainWindow/width"; constexpr char sMainWindowHeightKey[] = "MainWindow/height"; constexpr char sMainWindowPosXKey[] = "MainWindow/posx"; constexpr char sMainWindowPosYKey[] = "MainWindow/posy"; QString makeNewContentListName() { // basically, use date and time as the name e.g. YYYY-MM-DDThh:mm:ss const std::time_t rawtime = std::chrono::system_clock::to_time_t(std::chrono::system_clock::now()); tm timeinfo{}; #ifdef _WIN32 (void)localtime_s(&timeinfo, &rawtime); #else (void)localtime_r(&rawtime, &timeinfo); #endif constexpr int base = 10; QChar zeroPad('0'); return QString("%1-%2-%3T%4:%5:%6") .arg(timeinfo.tm_year + 1900, 4) .arg(timeinfo.tm_mon + 1, 2, base, zeroPad) .arg(timeinfo.tm_mday, 2, base, zeroPad) .arg(timeinfo.tm_hour, 2, base, zeroPad) .arg(timeinfo.tm_min, 2, base, zeroPad) .arg(timeinfo.tm_sec, 2, base, zeroPad); } bool parseBool(const QString& value, bool& out) { if (value == "false") { out = false; return true; } if (value == "true") { out = true; return true; } return false; } bool parseInt(const QString& value, int& out) { bool ok = false; const int converted = value.toInt(&ok); if (ok) out = converted; return ok; } bool parseProfilePart( const QString& key, const QString& value, std::map& profiles) { const int separator = key.lastIndexOf('/'); if (separator == -1) return false; const QString profileName = key.mid(0, separator); if (key.endsWith(sArchiveKey)) { profiles[profileName].mArchives.append(value); return true; } if (key.endsWith(sDataKey)) { profiles[profileName].mData.append(value); return true; } if (key.endsWith(sContentKey)) { profiles[profileName].mContent.append(value); return true; } return false; } bool parseSettingsSection(const QString& key, const QString& value, LauncherSettings::Settings& settings) { if (key == sLanguageKey) { settings.mLanguage = value; return true; } return false; } bool parseProfilesSection(const QString& key, const QString& value, LauncherSettings::Profiles& profiles) { if (key == sCurrentProfileKey) { profiles.mCurrentProfile = value; return true; } return parseProfilePart(key, value, profiles.mValues); } bool parseGeneralSection(const QString& key, const QString& value, LauncherSettings::General& general) { if (key == sFirstRunKey) return parseBool(value, general.mFirstRun); if (key == sMainWindowWidthKey) return parseInt(value, general.mMainWindow.mWidth); if (key == sMainWindowHeightKey) return parseInt(value, general.mMainWindow.mHeight); if (key == sMainWindowPosXKey) return parseInt(value, general.mMainWindow.mPosX); if (key == sMainWindowPosYKey) return parseInt(value, general.mMainWindow.mPosY); return false; } bool parseImporterSection(const QString& key, const QString& value, LauncherSettings::Importer& importer) { if (key == sImportContentSetupKey) return parseBool(value, importer.mImportContentSetup); if (key == sImportFontSetupKey) return parseBool(value, importer.mImportFontSetup); return false; } template void writeSectionHeader(const char (&name)[size], QTextStream& stream) { stream << "\n[" << name << "]\n"; } template void writeKeyValue(const char (&key)[size], const QString& value, QTextStream& stream) { stream << key << '=' << value << '\n'; } template void writeKeyValue(const char (&key)[size], bool value, QTextStream& stream) { stream << key << '=' << (value ? "true" : "false") << '\n'; } template void writeKeyValue(const char (&key)[size], int value, QTextStream& stream) { stream << key << '=' << value << '\n'; } template void writeKeyValues( const QString& prefix, const char (&suffix)[size], const QStringList& values, QTextStream& stream) { for (const auto& v : values) stream << prefix << '/' << suffix << '=' << v << '\n'; } void writeSettings(const LauncherSettings::Settings& value, QTextStream& stream) { writeSectionHeader(sSettingsSection, stream); writeKeyValue(sLanguageKey, value.mLanguage, stream); } void writeProfiles(const LauncherSettings::Profiles& value, QTextStream& stream) { writeSectionHeader(sProfilesSection, stream); writeKeyValue(sCurrentProfileKey, value.mCurrentProfile, stream); for (auto it = value.mValues.rbegin(); it != value.mValues.rend(); ++it) { writeKeyValues(it->first, sArchiveKey, it->second.mArchives, stream); writeKeyValues(it->first, sDataKey, it->second.mData, stream); writeKeyValues(it->first, sContentKey, it->second.mContent, stream); } } void writeGeneral(const LauncherSettings::General& value, QTextStream& stream) { writeSectionHeader(sGeneralSection, stream); writeKeyValue(sFirstRunKey, value.mFirstRun, stream); writeKeyValue(sMainWindowWidthKey, value.mMainWindow.mWidth, stream); writeKeyValue(sMainWindowPosYKey, value.mMainWindow.mPosY, stream); writeKeyValue(sMainWindowPosXKey, value.mMainWindow.mPosX, stream); writeKeyValue(sMainWindowHeightKey, value.mMainWindow.mHeight, stream); } void writeImporter(const LauncherSettings::Importer& value, QTextStream& stream) { writeSectionHeader(sImporterSection, stream); writeKeyValue(sImportContentSetupKey, value.mImportContentSetup, stream); writeKeyValue(sImportFontSetupKey, value.mImportFontSetup, stream); } } } void Config::LauncherSettings::writeFile(QTextStream& stream) const { writeSettings(mSettings, stream); writeProfiles(mProfiles, stream); writeGeneral(mGeneral, stream); writeImporter(mImporter, stream); } QStringList Config::LauncherSettings::getContentLists() { QStringList result; result.reserve(mProfiles.mValues.size()); for (const auto& [k, v] : mProfiles.mValues) result.push_back(k); return result; } void Config::LauncherSettings::setContentList(const GameSettings& gameSettings) { // obtain content list from game settings (if present) QList dirs(gameSettings.getDataDirs()); dirs.erase(std::remove_if( dirs.begin(), dirs.end(), [&](const SettingValue& dir) { return !gameSettings.isUserSetting(dir); }), dirs.end()); // archives and content files aren't preprocessed, so we don't need to track their original form const QList archivesOriginal(gameSettings.getArchiveList()); QStringList archives; for (const auto& archive : archivesOriginal) { if (gameSettings.isUserSetting(archive)) archives.push_back(archive.value); } const QList filesOriginal(gameSettings.getContentList()); QStringList files; for (const auto& file : filesOriginal) { if (gameSettings.isUserSetting(file)) files.push_back(file.value); } // if openmw.cfg has no content, exit so we don't create an empty content list. if (dirs.isEmpty() || files.isEmpty()) { return; } // local data directory and resources/vfs are not part of any profile const auto resourcesVfs = gameSettings.getResourcesVfs(); const auto dataLocal = gameSettings.getDataLocal(); dirs.erase( std::remove_if(dirs.begin(), dirs.end(), [&](const SettingValue& dir) { return dir.value == resourcesVfs; }), dirs.end()); dirs.erase( std::remove_if(dirs.begin(), dirs.end(), [&](const SettingValue& dir) { return dir.value == dataLocal; }), dirs.end()); // if any existing profile in launcher matches the content list, make that profile the default for (const QString& listName : getContentLists()) { const auto& listDirs = getDataDirectoryList(listName); constexpr auto compareDataDirectories = [](const SettingValue& dir, const QString& listDir) { #ifdef Q_OS_WINDOWS constexpr auto caseSensitivity = Qt::CaseInsensitive; #else constexpr auto caseSensitivity = Qt::CaseSensitive; #endif return dir.originalRepresentation == listDir || QDir::cleanPath(dir.originalRepresentation).compare(QDir::cleanPath(listDir), caseSensitivity) == 0; }; if (!std::ranges::equal(dirs, listDirs, compareDataDirectories)) continue; constexpr auto compareFiles = [](const QString& a, const QString& b) { return a.compare(b, Qt::CaseInsensitive) == 0; }; if (!std::ranges::equal(files, getContentListFiles(listName), compareFiles)) continue; if (!std::ranges::equal(archives, getArchiveList(listName), compareFiles)) continue; setCurrentContentListName(listName); return; } // otherwise, add content list QString newContentListName(makeNewContentListName()); setCurrentContentListName(newContentListName); QStringList newListDirs; for (const auto& dir : dirs) newListDirs.push_back(dir.originalRepresentation); setContentList(newContentListName, newListDirs, archives, files); } void Config::LauncherSettings::setContentList(const QString& contentListName, const QStringList& dirNames, const QStringList& archiveNames, const QStringList& fileNames) { Profile& profile = mProfiles.mValues[contentListName]; profile.mData = dirNames; profile.mArchives = archiveNames; profile.mContent = fileNames; } QStringList Config::LauncherSettings::getDataDirectoryList(const QString& contentListName) const { const Profile* profile = findProfile(contentListName); if (profile == nullptr) return {}; return profile->mData; } QStringList Config::LauncherSettings::getArchiveList(const QString& contentListName) const { const Profile* profile = findProfile(contentListName); if (profile == nullptr) return {}; return profile->mArchives; } QStringList Config::LauncherSettings::getContentListFiles(const QString& contentListName) const { const Profile* profile = findProfile(contentListName); if (profile == nullptr) return {}; return profile->mContent; } bool Config::LauncherSettings::setValue(const QString& sectionPrefix, const QString& key, const QString& value) { if (sectionPrefix == sSettingsSection) return parseSettingsSection(key, value, mSettings); if (sectionPrefix == sProfilesSection) return parseProfilesSection(key, value, mProfiles); if (sectionPrefix == sGeneralSection) return parseGeneralSection(key, value, mGeneral); if (sectionPrefix == sImporterSection) return parseImporterSection(key, value, mImporter); return false; } void Config::LauncherSettings::readFile(QTextStream& stream) { const QRegularExpression sectionRe("^\\[([^]]+)\\]$"); const QRegularExpression keyRe("^([^=]+)\\s*=\\s*(.+)$"); QString section; while (!stream.atEnd()) { const QString line = stream.readLine(); if (line.isEmpty() || line.startsWith("#")) continue; const QRegularExpressionMatch sectionMatch = sectionRe.match(line); if (sectionMatch.hasMatch()) { section = sectionMatch.captured(1); continue; } if (section.isEmpty()) continue; const QRegularExpressionMatch keyMatch = keyRe.match(line); if (!keyMatch.hasMatch()) continue; const QString key = keyMatch.captured(1).trimmed(); const QString value = keyMatch.captured(2).trimmed(); if (!setValue(section, key, value)) Log(Debug::Warning) << "Unsupported setting in the launcher config file: section: " << section.toUtf8().constData() << " key: " << key.toUtf8().constData() << " value: " << value.toUtf8().constData(); } } const Config::LauncherSettings::Profile* Config::LauncherSettings::findProfile(const QString& name) const { const auto it = mProfiles.mValues.find(name); if (it == mProfiles.mValues.end()) return nullptr; return &it->second; } void Config::LauncherSettings::clear() { mSettings = Settings{}; mGeneral = General{}; mProfiles = Profiles{}; mImporter = Importer{}; }