/***************************************************************************** * Copyright (c) 2014-2019 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 #ifndef _WIN32 # include #endif #include "../localisation/Language.h" #include "../platform/platform.h" #include "../util/Util.h" #include "File.h" #include "FileSystem.hpp" #include "Memory.hpp" #include "Path.hpp" #include "String.hpp" #include namespace Path { utf8* Append(utf8* buffer, size_t bufferSize, const utf8* src) { return safe_strcat_path(buffer, src, bufferSize); } std::string Combine(const std::string& a, const std::string& b) { utf8 buffer[MAX_PATH]; String::Set(buffer, sizeof(buffer), a.c_str()); Path::Append(buffer, sizeof(buffer), b.c_str()); return std::string(buffer); } std::string GetDirectory(const std::string& path) { const utf8* directory = GetDirectory(path.c_str()); std::string result(directory); Memory::Free(directory); return result; } utf8* GetDirectory(const utf8* path) { size_t maxSize = String::SizeOf(path) + 1; utf8* result = Memory::Allocate(maxSize); GetDirectory(result, maxSize, path); size_t reducedSize = String::SizeOf(path) + 1; result = Memory::Reallocate(result, reducedSize); return result; } utf8* GetDirectory(utf8* buffer, size_t bufferSize, const utf8* path) { auto lastPathSepIndex = std::max(String::LastIndexOf(path, *PATH_SEPARATOR), String::LastIndexOf(path, '/')); if (lastPathSepIndex < 0) { return String::Set(buffer, bufferSize, String::Empty); } size_t copyLength = std::min(lastPathSepIndex, static_cast(bufferSize - 1)); std::copy_n(path, copyLength, buffer); buffer[copyLength] = '\0'; return buffer; } void CreateDirectory(const std::string& path) { platform_ensure_directory_exists(path.c_str()); } bool DirectoryExists(const std::string& path) { return platform_directory_exists(path.c_str()); } std::string GetFileName(const std::string& path) { return GetFileName(path.c_str()); } const utf8* GetFileName(const utf8* path) { const utf8* lastPathSeparator = nullptr; for (const utf8* ch = path; *ch != '\0'; ch++) { if (*ch == *PATH_SEPARATOR || *ch == '/') { lastPathSeparator = ch; } } return lastPathSeparator == nullptr ? path : lastPathSeparator + 1; } std::string GetFileNameWithoutExtension(const std::string& path) { utf8* cstr = GetFileNameWithoutExtension(path.c_str()); std::string result = String::ToStd(cstr); Memory::Free(cstr); return result; } utf8* GetFileNameWithoutExtension(const utf8* path) { size_t maxSize = String::SizeOf(path) + 1; utf8* result = Memory::Allocate(maxSize); GetFileNameWithoutExtension(result, maxSize, path); size_t reducedSize = String::SizeOf(path) + 1; result = Memory::Reallocate(result, reducedSize); return result; } utf8* GetFileNameWithoutExtension(utf8* buffer, size_t bufferSize, const utf8* path) { path = GetFileName(path); const utf8* lastDot = nullptr; const utf8* ch = path; for (; *ch != '\0'; ch++) { if (*ch == '.') { lastDot = ch; } } if (lastDot == nullptr) { return String::Set(buffer, bufferSize, path); } size_t truncatedLength = std::min(bufferSize - 1, lastDot - path); std::copy_n(path, truncatedLength, buffer); buffer[truncatedLength] = '\0'; return buffer; } const std::string GetExtension(const std::string& path) { return fs::u8path(path).extension().string(); } const utf8* GetExtension(const utf8* path) { const utf8* lastDot = nullptr; const utf8* ch = GetFileName(path); for (; *ch != '\0'; ch++) { if (*ch == '.') { lastDot = ch; } } if (lastDot == nullptr) { // Return the null terminator, i.e. a blank extension return ch; } // Return the extension including the dot return lastDot; } utf8* GetAbsolute(utf8* buffer, size_t bufferSize, const utf8* relativePath) { #ifdef _WIN32 auto relativePathW = String::ToWideChar(relativePath); wchar_t absolutePathW[MAX_PATH]; DWORD length = GetFullPathNameW(relativePathW.c_str(), (DWORD)std::size(absolutePathW), absolutePathW, nullptr); if (length == 0) { return String::Set(buffer, bufferSize, relativePath); } else { auto absolutePath = String::ToUtf8(absolutePathW); String::Set(buffer, bufferSize, absolutePath.c_str()); return buffer; } #else utf8* absolutePath = realpath(relativePath, nullptr); if (absolutePath == nullptr) { return String::Set(buffer, bufferSize, relativePath); } else { String::Set(buffer, bufferSize, absolutePath); Memory::Free(absolutePath); return buffer; } #endif } std::string GetAbsolute(const std::string& relative) { utf8 absolute[MAX_PATH]; return GetAbsolute(absolute, sizeof(absolute), relative.c_str()); } bool Equals(const std::string& a, const std::string& b) { return String::Equals(a.c_str(), b.c_str()); } bool Equals(const utf8* a, const utf8* b) { bool ignoreCase = false; #ifdef _WIN32 ignoreCase = true; #endif return String::Equals(a, b, ignoreCase); } std::string ResolveCasing(const std::string& path) { std::string result; if (File::Exists(path)) { // Windows is case insensitive so it will exist and that is all that matters // for now. We can properly resolve the casing if we ever need to. result = path; } #ifndef _WIN32 else { std::string fileName = Path::GetFileName(path); std::string directory = Path::GetDirectory(path); struct dirent** files; auto count = scandir(directory.c_str(), &files, nullptr, alphasort); if (count != -1) { // Find a file which matches by name (case insensitive) for (int32_t i = 0; i < count; i++) { if (String::Equals(files[i]->d_name, fileName.c_str(), true)) { result = Path::Combine(directory, std::string(files[i]->d_name)); break; } } // Free memory for (int32_t i = 0; i < count; i++) { free(files[i]); } free(files); } } #endif return result; } } // namespace Path