#pragma region Copyright (c) 2014-2017 OpenRCT2 Developers /***************************************************************************** * OpenRCT2, an open source clone of Roller Coaster Tycoon 2. * * OpenRCT2 is the work of many authors, a full list can be found in contributors.md * For more information, visit https://github.com/OpenRCT2/OpenRCT2 * * OpenRCT2 is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * A full copy of the GNU General Public License can be found in licence.txt *****************************************************************************/ #pragma endregion #include #ifndef _WIN32 #include #endif #include "../platform/platform.h" #include "../util/Util.h" #include "../localisation/Language.h" #include "File.h" #include "Math.hpp" #include "Memory.hpp" #include "Path.hpp" #include "Util.hpp" #include "String.hpp" 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 = Math::Max( String::LastIndexOf(path, *PATH_SEPARATOR), String::LastIndexOf(path, '/') ); if (lastPathSepIndex < 0) { return String::Set(buffer, bufferSize, String::Empty); } size_t copyLength = Math::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 = Math::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 GetExtension(path.c_str()); } 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 wchar_t * relativePathW = utf8_to_widechar(relativePath); wchar_t absolutePathW[MAX_PATH]; DWORD length = GetFullPathNameW(relativePathW, (DWORD)Util::CountOf(absolutePathW), absolutePathW, nullptr); Memory::Free(relativePathW); if (length == 0) { return String::Set(buffer, bufferSize, relativePath); } else { utf8 * absolutePath = widechar_to_utf8(absolutePathW); String::Set(buffer, bufferSize, absolutePath); Memory::Free(absolutePath); 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 (sint32 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 (sint32 i = 0; i < count; i++) { free(files[i]); } free(files); } } #endif return result; } } // namespace Path