/***************************************************************************** * Copyright (c) 2014-2020 OpenRCT2 developers * * For a complete list of all authors, please refer to contributors.md * Interested in contributing? Visit https://github.com/OpenRCT2/OpenRCT2 * * OpenRCT2 is licensed under the GNU General Public License version 3. *****************************************************************************/ #include "PlatformEnvironment.h" #include "OpenRCT2.h" #include "config/Config.h" #include "core/File.h" #include "core/Path.hpp" #include "core/String.hpp" #include "platform/Platform.h" using namespace OpenRCT2; class PlatformEnvironment final : public IPlatformEnvironment { private: u8string _basePath[DIRBASE_COUNT]; bool _usingRctClassic{}; public: explicit PlatformEnvironment(DIRBASE_VALUES basePaths) { for (size_t i = 0; i < DIRBASE_COUNT; i++) { _basePath[i] = basePaths[i]; } } u8string GetDirectoryPath(DIRBASE base) const override { return _basePath[static_cast(base)]; } u8string GetDirectoryPath(DIRBASE base, DIRID did) const override { auto basePath = GetDirectoryPath(base); u8string_view directoryName; switch (base) { default: case DIRBASE::RCT1: directoryName = DirectoryNamesRCT2[static_cast(did)]; break; case DIRBASE::RCT2: directoryName = _usingRctClassic ? "Assets" : DirectoryNamesRCT2[static_cast(did)]; break; case DIRBASE::OPENRCT2: case DIRBASE::USER: case DIRBASE::CONFIG: directoryName = DirectoryNamesOpenRCT2[static_cast(did)]; break; } return Path::Combine(basePath, directoryName); } u8string GetFilePath(PATHID pathid) const override { auto dirbase = GetDefaultBaseDirectory(pathid); auto basePath = GetDirectoryPath(dirbase); auto fileName = FileNames[static_cast(pathid)]; return Path::Combine(basePath, fileName); } u8string FindFile(DIRBASE base, DIRID did, u8string_view fileName) const override { auto dataPath = GetDirectoryPath(base, did); auto path = Path::ResolveCasing(Path::Combine(dataPath, fileName)); if (base == DIRBASE::RCT1 && did == DIRID::DATA && !File::Exists(path)) { // Special case, handle RCT1 steam layout where some data files are under a CD root auto basePath = GetDirectoryPath(base); auto alternativePath = Path::ResolveCasing(Path::Combine(basePath, "RCTdeluxe_install", "Data", fileName)); if (File::Exists(alternativePath)) { path = alternativePath; } } return path; } void SetBasePath(DIRBASE base, u8string_view path) override { _basePath[static_cast(base)] = path; if (base == DIRBASE::RCT2) { // Check whether the RCT2 base is RCT2 or RCT Classic _usingRctClassic = Platform::IsRCT2ClassicPath(path); } } private: static const char* DirectoryNamesRCT2[]; static const u8string DirectoryNamesOpenRCT2[]; static const u8string FileNames[]; static DIRBASE GetDefaultBaseDirectory(PATHID pathid) { switch (pathid) { case PATHID::CONFIG: case PATHID::CONFIG_SHORTCUTS_LEGACY: case PATHID::CONFIG_SHORTCUTS: return DIRBASE::CONFIG; case PATHID::CACHE_OBJECTS: case PATHID::CACHE_TRACKS: case PATHID::CACHE_SCENARIOS: return DIRBASE::CACHE; case PATHID::SCORES_RCT2: return DIRBASE::RCT2; case PATHID::CHANGELOG: return DIRBASE::DOCUMENTATION; case PATHID::NETWORK_GROUPS: case PATHID::NETWORK_SERVERS: case PATHID::NETWORK_USERS: case PATHID::SCORES: case PATHID::SCORES_LEGACY: default: return DIRBASE::USER; } } }; std::unique_ptr OpenRCT2::CreatePlatformEnvironment(DIRBASE_VALUES basePaths) { return std::make_unique(basePaths); } static u8string GetOpenRCT2DirectoryName() { #if defined(__ANDROID__) return u8"openrct2-user"; #else return u8"OpenRCT2"; #endif } std::unique_ptr OpenRCT2::CreatePlatformEnvironment() { auto subDirectory = GetOpenRCT2DirectoryName(); // Set default paths std::string basePaths[DIRBASE_COUNT]; basePaths[static_cast(DIRBASE::OPENRCT2)] = Platform::GetInstallPath(); basePaths[static_cast(DIRBASE::USER)] = Path::Combine( Platform::GetFolderPath(SPECIAL_FOLDER::USER_DATA), subDirectory); basePaths[static_cast(DIRBASE::CONFIG)] = Path::Combine( Platform::GetFolderPath(SPECIAL_FOLDER::USER_CONFIG), subDirectory); basePaths[static_cast(DIRBASE::CACHE)] = Path::Combine( Platform::GetFolderPath(SPECIAL_FOLDER::USER_CACHE), subDirectory); basePaths[static_cast(DIRBASE::DOCUMENTATION)] = Platform::GetDocsPath(); // Override paths that have been specified via the command line if (!gCustomRCT1DataPath.empty()) { basePaths[static_cast(DIRBASE::RCT1)] = gCustomRCT1DataPath; } if (!gCustomRCT2DataPath.empty()) { basePaths[static_cast(DIRBASE::RCT2)] = gCustomRCT2DataPath; } if (!gCustomOpenRCT2DataPath.empty()) { basePaths[static_cast(DIRBASE::OPENRCT2)] = gCustomOpenRCT2DataPath; } if (!gCustomUserDataPath.empty()) { basePaths[static_cast(DIRBASE::USER)] = gCustomUserDataPath; basePaths[static_cast(DIRBASE::CONFIG)] = gCustomUserDataPath; basePaths[static_cast(DIRBASE::CACHE)] = gCustomUserDataPath; } if (basePaths[static_cast(DIRBASE::DOCUMENTATION)].empty()) { basePaths[static_cast(DIRBASE::DOCUMENTATION)] = basePaths[static_cast(DIRBASE::OPENRCT2)]; } auto env = OpenRCT2::CreatePlatformEnvironment(basePaths); // Now load the config so we can get the RCT1 and RCT2 paths auto configPath = env->GetFilePath(PATHID::CONFIG); config_set_defaults(); if (!config_open(configPath)) { config_save(configPath); } if (gCustomRCT1DataPath.empty()) { env->SetBasePath(DIRBASE::RCT1, gConfigGeneral.rct1_path); } if (gCustomRCT2DataPath.empty()) { env->SetBasePath(DIRBASE::RCT2, gConfigGeneral.rct2_path); } // Log base paths log_verbose("DIRBASE::RCT1 : %s", env->GetDirectoryPath(DIRBASE::RCT1).c_str()); log_verbose("DIRBASE::RCT2 : %s", env->GetDirectoryPath(DIRBASE::RCT2).c_str()); log_verbose("DIRBASE::OPENRCT2: %s", env->GetDirectoryPath(DIRBASE::OPENRCT2).c_str()); log_verbose("DIRBASE::USER : %s", env->GetDirectoryPath(DIRBASE::USER).c_str()); log_verbose("DIRBASE::CONFIG : %s", env->GetDirectoryPath(DIRBASE::CONFIG).c_str()); log_verbose("DIRBASE::CACHE : %s", env->GetDirectoryPath(DIRBASE::CACHE).c_str()); return env; } const char* PlatformEnvironment::DirectoryNamesRCT2[] = { "Data", // DATA "Landscapes", // LANDSCAPE nullptr, // LANGUAGE nullptr, // LOG_CHAT nullptr, // LOG_SERVER nullptr, // NETWORK_KEY "ObjData", // OBJECT nullptr, // PLUGIN "Saved Games", // SAVE "Scenarios", // SCENARIO nullptr, // SCREENSHOT nullptr, // SEQUENCE nullptr, // SHADER nullptr, // THEME "Tracks", // TRACK }; const u8string PlatformEnvironment::DirectoryNamesOpenRCT2[] = { u8"data", // DATA u8"landscape", // LANDSCAPE u8"language", // LANGUAGE u8"chatlogs", // LOG_CHAT u8"serverlogs", // LOG_SERVER u8"keys", // NETWORK_KEY u8"object", // OBJECT u8"plugin", // PLUGIN u8"save", // SAVE u8"scenario", // SCENARIO u8"screenshot", // SCREENSHOT u8"sequence", // SEQUENCE u8"shaders", // SHADER u8"themes", // THEME u8"track", // TRACK u8"heightmap", // HEIGHTMAP u8"replay", // REPLAY u8"desyncs", // DESYNCS u8"crash", // CRASH }; const u8string PlatformEnvironment::FileNames[] = { u8"config.ini", // CONFIG u8"hotkeys.dat", // CONFIG_SHORTCUTS_LEGACY u8"shortcuts.json", // CONFIG_SHORTCUTS u8"objects.idx", // CACHE_OBJECTS u8"tracks.idx", // CACHE_TRACKS u8"scenarios.idx", // CACHE_SCENARIOS u8"groups.json", // NETWORK_GROUPS u8"servers.cfg", // NETWORK_SERVERS u8"users.json", // NETWORK_USERS u8"highscores.dat", // SCORES u8"scores.dat", // SCORES (LEGACY) u8"Saved Games" PATH_SEPARATOR "scores.dat", // SCORES (RCT2) u8"changelog.txt", // CHANGELOG u8"plugin.store.json" // PLUGIN_STORE };