📄 pathutils.cc
字号:
#ifdef WIN32#include "win32.h"#include <shellapi.h>#include <shlobj.h>#include <tchar.h>#endif // WIN32#include "common.h"#include "pathutils.h"#include "stringutils.h"#include "urlencode.h"namespace utils_base {std::string const EMPTY_STR = "";// EXT_DELIM separates a file basename from extensionconst char EXT_DELIM = '.';// FOLDER_DELIMS separate folder segments and the filenameconst char* const FOLDER_DELIMS = "/\\";// DEFAULT_FOLDER_DELIM is the preferred delimiter for this platform#if WIN32const char DEFAULT_FOLDER_DELIM = '\\';#else // !WIN32const char DEFAULT_FOLDER_DELIM = '/';#endif // !WIN32///////////////////////////////////////////////////////////////////////////////// Pathname - parsing of pathnames into components, and vice versa///////////////////////////////////////////////////////////////////////////////bool Pathname::IsFolderDelimiter(char ch) { return (NULL != ::strchr(FOLDER_DELIMS, ch));}Pathname::Pathname() : folder_delimiter_(DEFAULT_FOLDER_DELIM) {}Pathname::Pathname(const std::string& pathname) : folder_delimiter_(DEFAULT_FOLDER_DELIM) { SetPathname(pathname);}void Pathname::SetFolderDelimiter(char delimiter) { ASSERT(IsFolderDelimiter(delimiter)); folder_delimiter_ = delimiter;}void Pathname::Normalize() { for (size_t i=0; i<folder_.length(); ++i) { if (IsFolderDelimiter(folder_[i])) { folder_[i] = folder_delimiter_; } }}void Pathname::clear() { folder_.clear(); basename_.clear(); extension_.clear();}std::string Pathname::pathname() const { std::string pathname(folder_); pathname.append(basename_); pathname.append(extension_); return pathname;}std::string Pathname::url() const { std::string s = "file://"; for (size_t i=0; i<folder_.length(); ++i) { if (i == 1 && folder_[i] == ':') // drive letter s += '|'; else if (IsFolderDelimiter(folder_[i])) s += '/'; else s += folder_[i]; } s += basename_; s += extension_; return UrlEncodeString(s);}void Pathname::SetPathname(const std::string &pathname) { std::string::size_type pos = pathname.find_last_of(FOLDER_DELIMS); if (pos != std::string::npos) { SetFolder(pathname.substr(0, pos + 1)); SetFilename(pathname.substr(pos + 1)); } else { SetFolder(EMPTY_STR); SetFilename(pathname); }}void Pathname::AppendPathname(const Pathname& pathname) { std::string full_pathname(folder_); full_pathname.append(pathname.pathname()); SetPathname(full_pathname);}std::string Pathname::folder() const { return folder_;}std::string Pathname::folder_name() const { std::string::size_type pos = std::string::npos; if (folder_.size() >= 2) { pos = folder_.find_last_of(FOLDER_DELIMS, folder_.length() - 2); } if (pos != std::string::npos) { return folder_.substr(pos + 1); } else { return folder_; }}std::string Pathname::parent_folder() const { std::string::size_type pos = std::string::npos; if (folder_.size() >= 2) { pos = folder_.find_last_of(FOLDER_DELIMS, folder_.length() - 2); } if (pos != std::string::npos) { return folder_.substr(0, pos + 1); } else { return EMPTY_STR; }}void Pathname::SetFolder(const std::string& folder) { folder_.assign(folder); // Ensure folder ends in a path delimiter if (!folder_.empty() && !IsFolderDelimiter(folder_[folder_.length()-1])) { folder_.push_back(folder_delimiter_); }}void Pathname::AppendFolder(const std::string& folder) { folder_.append(folder); // Ensure folder ends in a path delimiter if (!folder_.empty() && !IsFolderDelimiter(folder_[folder_.length()-1])) { folder_.push_back(folder_delimiter_); }}std::string Pathname::basename() const { return basename_;}void Pathname::SetBasename(const std::string& basename) { ASSERT(basename.find_first_of(FOLDER_DELIMS) == std::string::npos); basename_.assign(basename);}std::string Pathname::extension() const { return extension_;}void Pathname::SetExtension(const std::string& extension) { ASSERT(extension.find_first_of(FOLDER_DELIMS) == std::string::npos); ASSERT(extension.find_first_of(EXT_DELIM, 1) == std::string::npos); extension_.assign(extension); // Ensure extension begins with the extension delimiter if (!extension_.empty() && (extension_[0] != EXT_DELIM)) { extension_.insert(extension_.begin(), EXT_DELIM); }} std::string Pathname::filename() const { std::string filename(basename_); filename.append(extension_); return filename;}void Pathname::SetFilename(const std::string& filename) { std::string::size_type pos = filename.rfind(EXT_DELIM); if ((pos == std::string::npos) || (pos == 0)) { SetBasename(filename); SetExtension(EMPTY_STR); } else { SetBasename(filename.substr(0, pos)); SetExtension(filename.substr(pos)); }}///////////////////////////////////////////////////////////////////////////////// CreateUniqueFile///////////////////////////////////////////////////////////////////////////////std::string g_organization_name;std::string g_application_name;void SetOrganizationName(const std::string& organization) { g_organization_name = organization;}void SetApplicationName(const std::string& application) { g_application_name = application;}bool CreateFolder(const utils_base::Pathname& path) {#ifdef WIN32 if (!path.filename().empty()) return false; std::wstring pathname16 = ToUtf16(path.pathname()); if (!pathname16.empty() && (pathname16[0] != '\\')) { pathname16 = L"\\\\?\\" + pathname16; } DWORD res = ::GetFileAttributes(pathname16.c_str()); if (res != INVALID_FILE_ATTRIBUTES) { // Something exists at this location, check if it is a directory return ((res & FILE_ATTRIBUTE_DIRECTORY) != 0); } else if ((GetLastError() != ERROR_FILE_NOT_FOUND) && (GetLastError() != ERROR_PATH_NOT_FOUND)) { // Unexpected error return false; } // Directory doesn't exist, look up one directory level if (!path.parent_folder().empty()) { utils_base::Pathname parent(path); parent.SetFolder(path.parent_folder()); if (!CreateFolder(parent)) { return false; } } return (::CreateDirectory(pathname16.c_str(), NULL) != 0);#else // !WIN32 return false;#endif // !WIN32}bool FinishPath(utils_base::Pathname& path, bool create, const std::string& append) { if (!append.empty()) { path.AppendFolder(append); } if (create && !CreateFolder(path)) return false; return true;}bool GetTemporaryFolder(utils_base::Pathname& path, bool create, const std::string& append) { ASSERT(!g_application_name.empty());#ifdef WIN32 TCHAR buffer[MAX_PATH + 1]; if (!::GetTempPath(ARRAY_SIZE(buffer), buffer)) return false; if (!::GetLongPathName(buffer, buffer, ARRAY_SIZE(buffer))) return false; size_t len = strlen(buffer); if ((len > 0) && (buffer[len-1] != __T('\\'))) { len += utils_base::strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, __T("\\")); } len += utils_base::strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, ToUtf16(g_application_name).c_str()); if ((len > 0) && (buffer[len-1] != __T('\\'))) { len += utils_base::strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, __T("\\")); } if (len >= ARRAY_SIZE(buffer) - 1) return false; path.clear(); path.SetFolder(ToUtf8(buffer)); return FinishPath(path, create, append);#else // !WIN32 return false;#endif // !WIN32}bool GetAppDataFolder(utils_base::Pathname& path, bool create, const std::string& append) { ASSERT(!g_organization_name.empty()); ASSERT(!g_application_name.empty());#ifdef WIN32 TCHAR buffer[MAX_PATH + 1]; if (!::SHGetSpecialFolderPath(NULL, buffer, CSIDL_LOCAL_APPDATA, TRUE)) return false; if (!::GetLongPathName(buffer, buffer, ARRAY_SIZE(buffer))) return false; size_t len = utils_base::strcatn(buffer, ARRAY_SIZE(buffer), _T("\\")); len += utils_base::strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, ToUtf16(g_organization_name).c_str()); if ((len > 0) && (buffer[len-1] != __T('\\'))) { len += utils_base::strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, __T("\\")); } len += utils_base::strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, ToUtf16(g_application_name).c_str()); if ((len > 0) && (buffer[len-1] != __T('\\'))) { len += utils_base::strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, __T("\\")); } if (len >= ARRAY_SIZE(buffer) - 1) return false; path.clear(); path.SetFolder(ToUtf8(buffer)); return FinishPath(path, create, append);#else // !WIN32 return false;#endif // !WIN32}bool CleanupTemporaryFolder() {#ifdef WIN32 utils_base::Pathname temp_path; if (!GetTemporaryFolder(temp_path, false, "")) return false; std::wstring temp_path16 = ToUtf16(temp_path.pathname()); temp_path16.append(1, '*'); temp_path16.append(1, '\0'); SHFILEOPSTRUCT file_op = { 0 }; file_op.wFunc = FO_DELETE; file_op.pFrom = temp_path16.c_str(); file_op.fFlags = FOF_NOCONFIRMATION | FOF_NOERRORUI | FOF_SILENT; return (0 == SHFileOperation(&file_op));#else // !WIN32 return false;#endif // !WIN32}#if 0bool CreateUnijqueFile(utils_base::Pathname& path, bool create_empty) {#ifdef WIN32 // If not folder is supplied, use the temporary folder if (path.folder().empty()) { utils_base::Pathname temp_path; if (!GetTemporaryFolder(temp_path, true, "")) { return false; } path.SetFolder(temp_path.folder()); } printf("path: %s\n", path.pathname()); // If not filename is supplied, use a temporary name if (path.filename().empty()) { TCHAR filename[MAX_PATH]; std::wstring folder((ToUtf16)(path.folder())); if (!::GetTempFileName(folder.c_str(), __T("gt"), 0, filename)) return false; ASSERT(wcsncmp(folder.c_str(), filename, folder.length()) == 0); path.SetFilename(ToUtf8(filename + folder.length())); if (!create_empty) { VERIFY(::DeleteFile(ToUtf16(path.pathname()).c_str()) != FALSE); } return true; } // Otherwise, create a unique name based on the given filename // foo.txt -> foo-N.txt const std::string basename = path.basename(); const size_t MAX_VERSION = 100; size_t version = 0; while (version < MAX_VERSION) { std::string pathname = path.pathname(); std::wstring pathname16 = ToUtf16(pathname).c_str(); if (pathname16[0] != __T('\\')) pathname16 = __T("\\\\?\\") + pathname16; HANDLE hfile = CreateFile(pathname16.c_str(), GENERIC_WRITE, 0, NULL, CREATE_NEW, FILE_ATTRIBUTE_NORMAL, NULL); if (hfile != INVALID_HANDLE_VALUE) { CloseHandle(hfile); if (!create_empty) { VERIFY(::DeleteFile(pathname16.c_str()) != FALSE); } return true; } else { int err = GetLastError(); if (err != ERROR_FILE_EXISTS && err != ERROR_ACCESS_DENIED) { return false; } } version += 1; char version_base[MAX_PATH]; utils_base::sprintfn(version_base, ARRAY_SIZE(version_base), "%s-%u", basename.c_str(), version); path.SetBasename(version_base); } return false;#else // !WIN32 // TODO: Make this better. path.SetBasename("/tmp/temp-1");#endif // !WIN32}#endif///////////////////////////////////////////////////////////////////////////////} // namespace utils_base
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -