// Path.cpp: implementation of the CPath class. // ////////////////////////////////////////////////////////////////////// #include "stdafx.h" #include #include ////////////////////////////////////////////////////////////////////// // Constants ////////////////////////////////////////////////////////////////////// const char * const DLL_EXTENSION = "dll"; const char * const INI_EXTENSION = "ini"; const char * const EXE_EXTENSION = "exe"; const char * const WILD_NAME_EXTENSION = "*.*"; const TCHAR WILD_ONE = '?'; const TCHAR WILD_ANY = '*'; const char * const WILD_SET = "?*"; const char * const DIR_DOUBLEDELIM = "\\\\"; const TCHAR DRIVE_DELIMITER = ':'; const TCHAR DIRECTORY_DELIMITER = '\\'; const TCHAR EXTENSION_DELIMITER = '.'; const TCHAR DIRECTORY_DELIMITER2 = '/'; void * CPath::m_hInst = NULL; ////////////////////////////////////////////////////////////////////// // Helpers ////////////////////////////////////////////////////////////////////// void CPath::SethInst ( void * hInst ) { m_hInst = hInst; } void * CPath::GethInst() { return m_hInst; } ////////////////////////////////////////////////////////////////////// // Initialisation ////////////////////////////////////////////////////////////////////// //------------------------------------------------------------- // Task : Helper function for the various CPath constructors. // Initializes the data members and establishes various // class invariants //------------------------------------------------------------- inline void CPath::Init() { m_dwFindFileAttributes =0; m_hFindFile =NULL; } //------------------------------------------------------------- // Task : Helper function for the various CPath destructors. // Cleans up varios internals //------------------------------------------------------------- inline void CPath::Exit() { if(m_hFindFile != NULL) { FindClose(m_hFindFile); m_hFindFile =NULL; } } ////////////////////////////////////////////////////////////////////// // Construction/Destruction ////////////////////////////////////////////////////////////////////// //------------------------------------------------------------- // Task : Constructs a path //------------------------------------------------------------- CPath::CPath() { Init(); Empty(); } //------------------------------------------------------------- // Task : Constructs a path as copy of another //------------------------------------------------------------- CPath::CPath(const CPath& rPath) { Init(); m_strPath =rPath.m_strPath; } //------------------------------------------------------------- // Task : Constructs a path and points it 2 lpszPath //------------------------------------------------------------- CPath::CPath(const char * lpszPath) { Init(); m_strPath =lpszPath ? lpszPath : ""; cleanPathString(m_strPath); } CPath::CPath(const char * lpszPath, const char * NameExten) { Init(); SetDriveDirectory(lpszPath); SetNameExtension(NameExten); } //------------------------------------------------------------- // Task : Constructs a path and points it 2 strPath //------------------------------------------------------------- CPath::CPath(const std::string& strPath) { Init(); m_strPath =strPath; cleanPathString(m_strPath); } //------------------------------------------------------------- // Task : Constructs a path and points it 2 strPath //------------------------------------------------------------- CPath::CPath(const std::string& strPath, const char * NameExten) { Init(); SetDriveDirectory(strPath.c_str()); SetNameExtension(NameExten); } //------------------------------------------------------------- // Task : Constructs a path and points it 2 strPath //------------------------------------------------------------- CPath::CPath(const std::string& strPath, const std::string& NameExten) { Init(); SetDriveDirectory(strPath.c_str()); SetNameExtension(NameExten.c_str()); } //------------------------------------------------------------- // Task : Cleanup and destruct a path object //------------------------------------------------------------- CPath::~CPath() { Exit(); } //------------------------------------------------------------- // Post : Return TRUE if paths are equal // Task : Check if the two path are the same //------------------------------------------------------------- bool CPath::operator ==(const CPath& rPath) const { // Get fully qualified versions of the paths std::string FullyQualified1; std::string FullyQualified2; GetFullyQualified(FullyQualified1); rPath.GetFullyQualified(FullyQualified2); // Compare them return _stricmp(FullyQualified1.c_str(), FullyQualified2.c_str()) == 0; } //------------------------------------------------------------- // Post : Return TRUE if paths are different // Task : Check if the two path are different //------------------------------------------------------------- bool CPath::operator !=(const CPath& rPath) const { return !(*this == rPath); } //------------------------------------------------------------- // Task : Assign a path 2 another //------------------------------------------------------------- CPath& CPath::operator =(const CPath& rPath) { if(this != &rPath) m_strPath =rPath.m_strPath; return *this; } //------------------------------------------------------------- // Post : Return the path, so that assignements can be chained // Task : Assign a string 2 a path //------------------------------------------------------------- CPath& CPath::operator =(const char * lpszPath) { m_strPath =lpszPath ? lpszPath : ""; return *this; } //------------------------------------------------------------- // Post : Return the path, so that assignements can be chained // Task : Assign a string 2 a path //------------------------------------------------------------- CPath& CPath::operator =(const std::string& strPath) { m_strPath =strPath; return *this; } //------------------------------------------------------------- // Post : Converts path 2 string // Task : Convert path 2 string // Warning: because this pointer 2 string point in the data // of this class, it is possible 2 cast the result of this // function in any non-constant pointer and alter the data. // Very dangerous //------------------------------------------------------------- CPath::operator const char *() const { return (const char *)m_strPath.c_str(); } CPath::CPath(DIR_CURRENT_DIRECTORY /*sdt*/, const char * NameExten) { // Application's current directory Init(); CurrentDirectory(); if (NameExten) { SetNameExtension(NameExten); } } CPath::CPath(DIR_MODULE_DIRECTORY /*sdt*/, const char * NameExten) { // The directory where the executable of this app is Init(); ModuleDirectory(); if (NameExten) { SetNameExtension(NameExten); } } CPath::CPath(DIR_MODULE_FILE /*sdt*/) { // The directory where the executable of this app is Init(); Module(); } //------------------------------------------------------------- // Post : Returns the drive component without a colon, e.g. "c" // Returns the directory component with a leading backslash, // but no trailing backslash, e.g. "\dir\subdir" // Returns name compleletely without delimiters, e.g "letter" // Returns extension completely without delimiters, e.g. "doc" // Globals : // I/O : // Task : Return the individual components of this path. // For any given argument, you can pass NULL if you are not // interested in that component. // Do not rely on pNames being <= 8 characters, extensions // being <= 3 characters, or drives being 1 character //------------------------------------------------------------- void CPath::GetComponents(std::string* pDrive, std::string* pDirectory, std::string* pName, std::string* pExtension) const { TCHAR buff_drive[_MAX_DRIVE + 1]; TCHAR buff_dir [_MAX_DIR + 1]; TCHAR buff_name [_MAX_FNAME + 1]; TCHAR buff_ext [_MAX_EXT + 1]; ZeroMemory(buff_drive,sizeof(buff_drive)); ZeroMemory(buff_dir, sizeof(buff_dir)); ZeroMemory(buff_name, sizeof(buff_name)); ZeroMemory(buff_ext, sizeof(buff_ext)); _splitpath(m_strPath.c_str(), pDrive ? buff_drive : NULL, pDirectory ? buff_dir : NULL, pName ? buff_name : NULL, pExtension ? buff_ext : NULL); if(pDrive) *pDrive =buff_drive; if(pDirectory) *pDirectory =buff_dir; if(pName) *pName =buff_name; if(pExtension) *pExtension =buff_ext; // DOS's _splitpath returns "d:", we return "d" if(pDrive) StripTrailingChar(*pDrive,DRIVE_DELIMITER); // DOS's _splitpath returns "\dir\subdir\", we return "\dir\subdir" if(pDirectory) StripTrailingBackslash(*pDirectory); // DOS's _splitpath returns ".ext", we return "ext" if(pExtension) StripLeadingChar(*pExtension,EXTENSION_DELIMITER); } //------------------------------------------------------------- // Task : Get drive and directory from path //------------------------------------------------------------- void CPath::GetDriveDirectory(std::string& rDriveDirectory) const { std::string Drive; std::string Directory; GetComponents(&Drive,&Directory); rDriveDirectory =Drive; if(!Drive.empty()) { rDriveDirectory += DRIVE_DELIMITER; rDriveDirectory += Directory; } } std::string CPath::GetDriveDirectory(void) const { std::string rDriveDirectory; GetDriveDirectory(rDriveDirectory); return rDriveDirectory; } //------------------------------------------------------------- // Task : Get directory from path //------------------------------------------------------------- void CPath::GetDirectory(std::string& rDirectory) const { GetComponents(NULL,&rDirectory); } std::string CPath::GetDirectory(void) const { std::string rDirectory; GetComponents(NULL,&rDirectory); return rDirectory; } //------------------------------------------------------------- // Task : Get filename and extension from path //------------------------------------------------------------- void CPath::GetNameExtension(std::string& rNameExtension) const { std::string Name; std::string Extension; GetComponents(NULL,NULL,&Name,&Extension); rNameExtension =Name; if(!Extension.empty()) { rNameExtension += EXTENSION_DELIMITER; rNameExtension += Extension; } } std::string CPath::GetNameExtension(void) const { std::string rNameExtension; GetNameExtension(rNameExtension); return rNameExtension; } //------------------------------------------------------------- // Task : Get filename from path //------------------------------------------------------------- void CPath::GetName(std::string& rName) const { GetComponents(NULL,NULL,&rName); } std::string CPath::GetName(void) const { std::string rName; GetComponents(NULL,NULL,&rName); return rName; } //------------------------------------------------------------- // Task : Get file extension from path //------------------------------------------------------------- void CPath::GetExtension(std::string& rExtension) const { GetComponents(NULL,NULL,NULL,&rExtension); } std::string CPath::GetExtension(void) const { std::string rExtension; GetComponents(NULL,NULL,NULL,&rExtension); return rExtension; } //------------------------------------------------------------- // Task : Get current directory //------------------------------------------------------------- void CPath::GetLastDirectory(std::string& rDirectory) const { std::string Directory; rDirectory = ""; GetDirectory(Directory); StripTrailingBackslash(Directory); if(Directory.empty()) { return; } std::string::size_type nDelimiter = Directory.rfind(DIRECTORY_DELIMITER); rDirectory = Directory.substr(nDelimiter); StripLeadingBackslash(rDirectory); } std::string CPath::GetLastDirectory(void) const { std::string rDirecotry; GetLastDirectory(rDirecotry); return rDirecotry; } //------------------------------------------------------------- // Task : Get fully qualified path //------------------------------------------------------------- void CPath::GetFullyQualified(std::string& rFullyQualified) const { TCHAR buff_fullname[MAX_PATH]; memset(buff_fullname, 0, sizeof(buff_fullname)); _fullpath(buff_fullname, m_strPath.c_str(), MAX_PATH - 1); rFullyQualified =buff_fullname; } //------------------------------------------------------------- // Post : Return TRUE if path does not start from filesystem root // Task : Check if path is a relative one (e.g. doesn't start with C:\...) //------------------------------------------------------------- bool CPath::IsRelative() const { if (m_strPath.length() > 1 && m_strPath[1] == DRIVE_DELIMITER) { return false; } if (m_strPath.length() > 1 && m_strPath[0] == DIRECTORY_DELIMITER && m_strPath[1] == DIRECTORY_DELIMITER) { return false; } return true; } //------------------------------------------------------------- // Task : Set path components //------------------------------------------------------------- void CPath::SetComponents(const char * lpszDrive, const char * lpszDirectory, const char * lpszName, const char * lpszExtension) { TCHAR buff_fullname[MAX_PATH]; memset(buff_fullname, 0, sizeof(buff_fullname)); _makepath(buff_fullname, lpszDrive, lpszDirectory, lpszName, lpszExtension); m_strPath.erase(); m_strPath =buff_fullname; } //------------------------------------------------------------- // Task : Set path's drive //------------------------------------------------------------- void CPath::SetDrive(TCHAR chDrive) { stdstr_f Drive("%c",chDrive); std::string Directory; std::string Name; std::string Extension; GetComponents(NULL,&Directory,&Name,&Extension); SetComponents(Drive.c_str(),Directory.c_str(),Name.c_str(),Extension.c_str()); } //------------------------------------------------------------- // Task : Set path's directory //------------------------------------------------------------- void CPath::SetDirectory(const char * lpszDirectory, bool bEnsureAbsolute /*= FALSE*/) { std::string Drive; std::string Directory =lpszDirectory; std::string Name; std::string Extension; if(bEnsureAbsolute) EnsureLeadingBackslash(Directory); EnsureTrailingBackslash(Directory); GetComponents(&Drive,NULL,&Name,&Extension); SetComponents(Drive.c_str(),Directory.c_str(),Name.c_str(),Extension.c_str()); } //------------------------------------------------------------- // Task : Set path's drive and directory //------------------------------------------------------------- void CPath::SetDriveDirectory(const char * lpszDriveDirectory) { std::string DriveDirectory =lpszDriveDirectory; std::string Name; std::string Extension; EnsureTrailingBackslash(DriveDirectory); cleanPathString(DriveDirectory); GetComponents(NULL,NULL,&Name,&Extension); SetComponents(NULL,DriveDirectory.c_str(),Name.c_str(),Extension.c_str()); } //------------------------------------------------------------- // Task : Set path's filename //------------------------------------------------------------- void CPath::SetName(const char * lpszName) { std::string Drive; std::string Directory; std::string Extension; GetComponents(&Drive,&Directory,NULL,&Extension); SetComponents(Drive.c_str(),Directory.c_str(),lpszName,Extension.c_str()); } //------------------------------------------------------------- // Task : Set path's filename //------------------------------------------------------------- void CPath::SetName(int iName) { std::string Drive; std::string Directory; std::string Extension; TCHAR sName[33]; memset(sName, 0, sizeof(sName)); _itoa(iName, sName, 10); GetComponents(&Drive,&Directory,NULL,&Extension); SetComponents(Drive.c_str(),Directory.c_str(),sName,Extension.c_str()); } //------------------------------------------------------------- // Task : Set path's file extension //------------------------------------------------------------- void CPath::SetExtension(const char * lpszExtension) { std::string Drive; std::string Directory; std::string Name; GetComponents(&Drive,&Directory,&Name); SetComponents(Drive.c_str(),Directory.c_str(),Name.c_str(),lpszExtension); } //------------------------------------------------------------- // Task : Set path's file extension //------------------------------------------------------------- void CPath::SetExtension(int iExtension) { std::string Drive; std::string Directory; std::string Name; TCHAR sExtension[20]; memset(sExtension, 0, sizeof(sExtension)); _itoa(iExtension, sExtension, 10); GetComponents(&Drive,&Directory,&Name); SetComponents(Drive.c_str(),Directory.c_str(),Name.c_str(),sExtension); } //------------------------------------------------------------- // Task : Set path's filename and extension //------------------------------------------------------------- void CPath::SetNameExtension(const char * lpszNameExtension) { std::string Drive; std::string Directory; GetComponents(&Drive,&Directory); SetComponents(Drive.c_str(),Directory.c_str(),lpszNameExtension,NULL); } //------------------------------------------------------------- // Task : Append a subdirectory 2 path's directory //------------------------------------------------------------- void CPath::AppendDirectory(const char * lpszSubDirectory) { std::string Drive; std::string Directory; std::string SubDirectory =lpszSubDirectory; std::string Name; std::string Extension; if(SubDirectory.empty()) return; // Strip out any preceeding backslash StripLeadingBackslash(SubDirectory); EnsureTrailingBackslash(SubDirectory); GetComponents(&Drive,&Directory,&Name,&Extension); EnsureTrailingBackslash(Directory); Directory +=SubDirectory; SetComponents(Drive.c_str(),Directory.c_str(),Name.c_str(),Extension.c_str()); } //------------------------------------------------------------- // Pre : If pLastDirectory is given we will store the name of the // deepest directory (the one we're just exiting) in it // Task : Remove deepest subdirectory from path //------------------------------------------------------------- void CPath::UpDirectory(std::string *pLastDirectory /*= NULL*/) { std::string Directory; GetDirectory(Directory); StripTrailingBackslash(Directory); if(Directory.empty()) return; std::string::size_type nDelimiter =Directory.rfind(DIRECTORY_DELIMITER); if(pLastDirectory != NULL) { *pLastDirectory =Directory.substr(nDelimiter); StripLeadingBackslash(*pLastDirectory); } if(nDelimiter != std::string::npos) Directory =Directory.substr(0,nDelimiter); SetDirectory(Directory.c_str()); } //------------------------------------------------------------- // Task : Set path 2 current directory //------------------------------------------------------------- void CPath::CurrentDirectory() { TCHAR buff_path[MAX_PATH]; memset(buff_path, 0, sizeof(buff_path)); ::GetCurrentDirectory(MAX_PATH,buff_path); Empty(); SetDriveDirectory(buff_path); } //------------------------------------------------------------- // Task : Set path 2 the name of specified module //------------------------------------------------------------- void CPath::Module(void * hInstance) { TCHAR buff_path[MAX_PATH]; memset(buff_path, 0, sizeof(buff_path)); GetModuleFileName((HINSTANCE)hInstance, buff_path, MAX_PATH); m_strPath =buff_path; } //------------------------------------------------------------- // Task : Set path 2 the name of current module //------------------------------------------------------------- void CPath::Module() { TCHAR buff_path[MAX_PATH]; memset(buff_path, 0, sizeof(buff_path)); GetModuleFileName((HMODULE)m_hInst,buff_path,MAX_PATH); m_strPath =buff_path; } //------------------------------------------------------------- // Task : Set path 2 the directory of specified module //------------------------------------------------------------- void CPath::ModuleDirectory(void * hInstance) { Module((HINSTANCE)hInstance); SetNameExtension(""); } //------------------------------------------------------------- // Task : Set path 2 the directory of current module //------------------------------------------------------------- void CPath::ModuleDirectory() { Module(); SetNameExtension(""); } //--------------------------------------------------------------------------- // Post : Return TRUE if a directory // Task : Check if this path represents a directory //--------------------------------------------------------------------------- bool CPath::IsDirectory() const { // Check if this path has a filename std::string file_name; GetNameExtension(file_name); return file_name.empty(); } //------------------------------------------------------------- // Post : Return TRUE if directory exists // Task : To determine if the directory exists, we need to // create a test path with a wildcard (*.*) extension // and see if FindFirstFile returns anything. We don't // use CPath::FindFirst() because that routine parses out // '.' and '..', which fails for empty directories //------------------------------------------------------------- bool CPath::DirectoryExists() const { // Create test path CPath TestPath(m_strPath.c_str()); std::string DirName; TestPath.UpDirectory(&DirName); TestPath.SetNameExtension(DirName.c_str()); WIN32_FIND_DATA FindData; HANDLE hFindFile =FindFirstFile((const char *)TestPath,&FindData); // Find anything bool bGotFile =(hFindFile != INVALID_HANDLE_VALUE); if(hFindFile != NULL) // Make sure we close the search FindClose(hFindFile); return bGotFile; } //------------------------------------------------------------- // Post : Return TRUE if these is such a file // Task : Check if file exists //------------------------------------------------------------- bool CPath::Exists() const { WIN32_FIND_DATA FindData; HANDLE hFindFile =FindFirstFile(m_strPath.c_str(),&FindData); bool bSuccess =(hFindFile != INVALID_HANDLE_VALUE); if(hFindFile != NULL) // Make sure we close the search FindClose(hFindFile); return bSuccess; } //------------------------------------------------------------- // Post : Return TRUE on success // Task : Delete file //------------------------------------------------------------- bool CPath::Delete(bool bEvenIfReadOnly) const { uint32_t dwAttr =::GetFileAttributes(m_strPath.c_str()); if(dwAttr == (uint32_t)-1) // File does not exists return FALSE; if(((dwAttr & FILE_ATTRIBUTE_READONLY) == FILE_ATTRIBUTE_READONLY) && !bEvenIfReadOnly) // File is read-only, and we're not allowed 2 delete it return FALSE; SetFileAttributes(m_strPath.c_str(),FILE_ATTRIBUTE_NORMAL); return DeleteFile(m_strPath.c_str()) != 0; } //------------------------------------------------------------- // Post : Return TRUE on success, FALSE if there is such a target file // and we weren't granted permission 2 overwrite file or some error // Task : Copy file // Since ::CopyFile will not overwrite read only files // we will make sure the target file is writable first //------------------------------------------------------------- bool CPath::CopyTo(const char * lpcszTargetFile, bool bOverwrite) { // Check if the target file exists CPath TargetFile(lpcszTargetFile); if(TargetFile.Exists()) { // Yeah there is already such a target file // Decide if we should overwrite if(!bOverwrite) return FALSE; // Delete any previous target if(!TargetFile.Delete(TRUE)) return FALSE; } // CopyFile will set the target's attributes 2 the same as // the source after copying return CopyFile(m_strPath.c_str(),lpcszTargetFile,!bOverwrite) != 0; } //------------------------------------------------------------- // Post : Return TRUE on success, FALSE if there is such a target file // and we weren't granted permission 2 overwrite file or some error // Task : Move file //------------------------------------------------------------- bool CPath::MoveTo(const char * lpcszTargetFile, bool bOverwrite) { // Check if the target file exists CPath TargetFile(lpcszTargetFile); if(TargetFile.Exists()) { // Yeah there is already such a target file // Decide if we should overwrite if(!bOverwrite) return FALSE; // Delete any previous target if(!TargetFile.Delete(TRUE)) return FALSE; } return MoveFile(m_strPath.c_str(),lpcszTargetFile) != 0; } //------------------------------------------------------------- // Post : Return TRUE if attributes do match // Task : Compare finder attributes //------------------------------------------------------------- bool CPath::AttributesMatch(uint32_t dwTargetAttributes, uint32_t dwFileAttributes) { if (dwTargetAttributes == _A_ALLFILES) { return true; } if(dwTargetAttributes == _A_NORMAL) { return ((_A_SUBDIR & dwFileAttributes) == 0); } return ( ((dwTargetAttributes & dwFileAttributes) != 0) && ((_A_SUBDIR & dwTargetAttributes) == (_A_SUBDIR & dwFileAttributes)) ); } //------------------------------------------------------------- // Post : Return TRUE if any match found // Task : Find the first file that meets this path and the specified attributes // You can specify the current attributes of the file or directory // The attributes are represented by a combination (|) of the following // constants: // // _A_ARCH Archive. Set whenever the file is // changed, and cleared by the BACKUP command // _A_HIDDEN Hidden file. Not normally seen with // the DIR command, unless the /AH option // is used. Returns information about normal // files as well as files with this attribute // _A_NORMAL Normal. File can be read or written to // without restriction // _A_RDONLY Read-only. File cannot be opened for writing, // and a file with the same name cannot be created // _A_SUBDIR Subdirectory // _A_SYSTEM System file. Not normally seen with the DIR // command, unless the /AS option is used // // These attributes do not follow a simple additive logic // Note that _A_NORMAL is 0x00, so it effectively cannot be // removed from the attribute set. You will therefore always // get normal files, and may also get Archive, Hidden, etc. // if you specify those attributes // See aso: FindFirstFile, FindNextFile //------------------------------------------------------------- bool CPath::FindFirst(uint32_t dwAttributes /*= _A_NORMAL*/) { m_dwFindFileAttributes =dwAttributes; BOOL bGotFile; BOOL bWantSubdirectory =(BOOL)(_A_SUBDIR & dwAttributes); // Close handle to any previous enumeration Exit(); // i.) Finding first candidate file WIN32_FIND_DATA FindData; m_hFindFile =FindFirstFile(m_strPath.c_str(),&FindData); bGotFile =(m_hFindFile != INVALID_HANDLE_VALUE); while(bGotFile) { // ii.) Compare candidate to attributes, and filter out the "." and ".." folders if(!AttributesMatch(m_dwFindFileAttributes,FindData.dwFileAttributes)) goto LABEL_GetAnother; if(bWantSubdirectory && (FindData.cFileName[0] == '.')) goto LABEL_GetAnother; // iii.) Found match, prepare result if((_A_SUBDIR & FindData.dwFileAttributes) != 0) StripTrailingBackslash(m_strPath); SetNameExtension(FindData.cFileName); if((_A_SUBDIR & FindData.dwFileAttributes) != 0) EnsureTrailingBackslash(m_strPath); return TRUE; // iv.) Not found match, get another LABEL_GetAnother: bGotFile =FindNextFile(m_hFindFile,&FindData); } return FALSE; } //------------------------------------------------------------- // Post : Return TRUE if a new match found // Task : Find the next file that meets the conditions specified // in the last FindFirst call //------------------------------------------------------------- bool CPath::FindNext() { if (m_hFindFile == NULL) return FALSE; WIN32_FIND_DATA FindData; while(FindNextFile(m_hFindFile,&FindData) != FALSE) { // while(FindNext(...)) if(AttributesMatch(m_dwFindFileAttributes,FindData.dwFileAttributes)) { // if(AttributesMatch(...) if((_A_SUBDIR & FindData.dwFileAttributes) == _A_SUBDIR) { if (IsDirectory()) { // Found a directory UpDirectory(); } else { SetNameExtension(""); } AppendDirectory(FindData.cFileName); } else { // Found a file if (IsDirectory()) { // Found a directory UpDirectory(); } SetNameExtension(FindData.cFileName); } if((_A_SUBDIR & FindData.dwFileAttributes) == _A_SUBDIR) EnsureTrailingBackslash(m_strPath); return TRUE; } } return FALSE; } //------------------------------------------------------------- // Post : Return TRUE on success // Task : Change current working directory of application 2 path //------------------------------------------------------------- bool CPath::ChangeDirectory() { std::string DriveDirectory; GetDriveDirectory(DriveDirectory); return SetCurrentDirectory(DriveDirectory.c_str()) != 0; } //------------------------------------------------------------- // Pre : If bCreateIntermediates is TRUE, create all eventually // missing parent directories too // Post : Return TRUE on success // Task : Create new directory //------------------------------------------------------------- bool CPath::DirectoryCreate(bool bCreateIntermediates /*= TRUE*/) { std::string PathText; bool bSuccess; GetDriveDirectory(PathText); StripTrailingBackslash(PathText); bSuccess =::CreateDirectory(PathText.c_str(),NULL) != 0; if(!bSuccess) { CPath CurrentDir(CPath::CURRENT_DIRECTORY); bSuccess = ChangeDirectory() != 0; CurrentDir.ChangeDirectory(); } if(!bSuccess && bCreateIntermediates) { std::string::size_type nDelimiter =PathText.rfind(DIRECTORY_DELIMITER); if(nDelimiter == std::string::npos) return FALSE; PathText.resize(nDelimiter + 1); CPath SubPath(PathText); if (SubPath.DirectoryCreate()) return DirectoryCreate(false); else return FALSE; } return bSuccess; } //Helpers //------------------------------------------------------------------------ // Task : Remove first character (if any) if it's chLeading //------------------------------------------------------------------------ void CPath::cleanPathString(std::string& rDirectory) const { LPCSTR const DIR_DOUBLEDELIM = "\\\\"; std::string::size_type pos = rDirectory.find( DIRECTORY_DELIMITER2 ); while ( pos != std::string::npos ) { rDirectory.replace( pos, 1, &DIRECTORY_DELIMITER ); pos = rDirectory.find( DIRECTORY_DELIMITER2, pos + 1 ); } bool AppendEnd = !_strnicmp(rDirectory.c_str(), "\\\\", 2); pos = rDirectory.find( DIR_DOUBLEDELIM ); while ( pos != std::string::npos ) { rDirectory.replace( pos, 2, &DIRECTORY_DELIMITER ); pos = rDirectory.find( DIR_DOUBLEDELIM, pos + 1 ); } if (AppendEnd) { rDirectory.insert(0, "\\"); } } void CPath::StripLeadingChar(std::string& rText, TCHAR chLeading) const { std::string::size_type nLength =rText.length(); if(nLength == 0) return; if(rText[0] == chLeading) rText =rText.substr(1); } //------------------------------------------------------------------------ // Task : Remove first character if \ //------------------------------------------------------------------------ void CPath::StripLeadingBackslash(std::string& Directory) const { std::string::size_type nLength =Directory.length(); // If Directory is of the form '\', don't do it if(nLength <= 1) return; if(Directory[0] == DIRECTORY_DELIMITER) Directory =Directory.substr(1); } //------------------------------------------------------------------------ // Task : Remove last character (if any) if it's chTrailing //------------------------------------------------------------------------ void CPath::StripTrailingChar(std::string& rText, TCHAR chTrailing) const { std::string::size_type nLength =rText.length(); if(nLength == 0) return; if(rText[nLength-1] == chTrailing) rText.resize(nLength-1); } //------------------------------------------------------------------------ // Task : Remove last character if \ //------------------------------------------------------------------------ void CPath::StripTrailingBackslash(std::string& rDirectory) const { for (;;) { std::string::size_type nLength = rDirectory.length(); if(nLength <= 1) { return; } if(rDirectory[nLength-1] == DIRECTORY_DELIMITER || rDirectory[nLength-1] == DIRECTORY_DELIMITER2) { rDirectory.resize(nLength-1); continue; } return; } } //------------------------------------------------------------------------ // Task : Add a backslash to the end of Directory if there is // not already one there //------------------------------------------------------------------------ void CPath::EnsureTrailingBackslash(std::string& Directory) const { std::string::size_type nLength = Directory.length(); if (Directory.empty() || (Directory[nLength - 1] != DIRECTORY_DELIMITER)) { Directory += DIRECTORY_DELIMITER; } } //------------------------------------------------------------------------ // Task : Add a backslash to the beginning of Directory if there // is not already one there //------------------------------------------------------------------------ void CPath::EnsureLeadingBackslash(std::string & Directory) const { if(Directory.empty() || (Directory[0] != DIRECTORY_DELIMITER)) { Directory = stdstr_f("%c%s", DIRECTORY_DELIMITER, Directory.c_str()); } }