⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 pathutils.cc

📁 本人收集整理的一份c/c++跨平台网络库
💻 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 + -