Поиск корзины на локальном диске NTFS
-
06-09-2019 - |
Вопрос
Я пытаюсь написать какой-нибудь простой код, который вернет каталог для корзины на локальном диске.Кажется, это было бы просто - в Google должна быть тысяча ответов.Пока ничего не нашел :(
Я обнаружил, что диски FAT и NTFS имеют разные базовые названия (RECYCLED и RECYCLER).Я обнаружил, что "корзина" - это виртуальная папка, которая объединяет корзины всех дисков на компьютере.
Чего я не нашел, так это способа найти C:каталог корзины диска - даже на вьетнамском (или любом другом неанглоязычном) компьютере.(Ни в каких сообщениях, которые я могу найти, не указано, будет ли "RECYCLER" интернационализирован или нет)
Кто-нибудь может указать мне на окончательный ответ?
Спасибо
Обновить:Осведомленный о CSIDL_BITBUCKET
и функции, которые его используют.Однако из всего, что я прочитал, это указывает на виртуальный каталог, который представляет собой объединение всех удаленных файлов этим пользователем на всех дисках.Ищу каталог физической корзины (в моей Vista он выглядит как C: \ $ Recycle.Bin, насколько я могу судить)
Решение 2
Используя совет Рэймонда Чена и чью-то еще технику (не помню, где я ее нашел) Я представляю функцию, которая найдет каталог корзины на диске.Функция циклически перебирает каталоги в корневом каталоге, просматривая скрытые и / или системные каталоги.Когда он находит один, он проверяет дочерние подкаталоги в поисках того, у которого есть CLSID_Recycle Bin.
Обратите внимание, что я включил две функции GetFolderCLSID ниже.У Раймонда Чена проще, но он не работает в Windows 2000.Другая реализация длиннее, но, похоже, работает везде.
Звоните, как:CString recycleDir = Найти recyclebinondrive(L"C:\");
CString FindRecycleBinOnDrive(LPCWSTR path)
{
CString search;
search.Format(L"%c:\\*", path[0]);
WIN32_FIND_DATA fd = {0};
HANDLE fHandle = FindFirstFile(search, &fd);
while(INVALID_HANDLE_VALUE != fHandle)
{
if(FILE_ATTRIBUTE_DIRECTORY == (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) //only check directories
{
if(0 != (fd.dwFileAttributes & (FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN))) //only check hidden and/or system directories
{
//the recycle bin directory itself won't be marked, but a SID-specific child directory will, so now look at them
CString childSearch;
childSearch.Format(L"%c:\\%s\\*", path[0], fd.cFileName);
WIN32_FIND_DATA childFD = {0};
HANDLE childHandle = FindFirstFile(childSearch, &childFD);
while(INVALID_HANDLE_VALUE != childHandle)
{
if((FILE_ATTRIBUTE_DIRECTORY == (childFD.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) && //only check directories
(childFD.cFileName[0] != L'.')) //don't check . and .. dirs
{
CString fullPath;
fullPath.Format(L"%c:\\%s\\%s", path[0], fd.cFileName, childFD.cFileName);
CLSID id = {0};
HRESULT hr = GetFolderCLSID(fullPath, id);
if(SUCCEEDED(hr))
{
if(IsEqualGUID(CLSID_RecycleBin, id))
{
FindClose(childHandle);
FindClose(fHandle);
//return the parent (recycle bin) directory
fullPath.Format(L"%c:\\%s", path[0], fd.cFileName);
return fullPath;
}
}
else
{
Log(logERROR, L"GetFolderCLSID returned %08X for %s", hr, fullPath);
}
}
if(FALSE == FindNextFile(childHandle, &childFD))
{
FindClose(childHandle);
childHandle = INVALID_HANDLE_VALUE;
}
}
}
}
if(FALSE == FindNextFile(fHandle, &fd))
{
FindClose(fHandle);
fHandle = INVALID_HANDLE_VALUE;
}
}
_ASSERT(0);
return L"";
}
//Works on Windows 2000, and even as Local System account
HRESULT GetFolderCLSID(LPCWSTR path, CLSID& pathCLSID)
{
LPMALLOC pMalloc = NULL;
HRESULT hr = 0;
if (SUCCEEDED(hr = SHGetMalloc(&pMalloc)))
{
LPSHELLFOLDER pshfDesktop = NULL;
if (SUCCEEDED(hr = SHGetDesktopFolder(&pshfDesktop)))
{
LPITEMIDLIST pidl = NULL;
DWORD dwAttributes = SFGAO_FOLDER;
if (SUCCEEDED(hr = pshfDesktop->ParseDisplayName(NULL, NULL, (LPWSTR)path, NULL, &pidl, &dwAttributes)))
{
LPPERSIST pPersist = NULL;
if (SUCCEEDED(hr = pshfDesktop->BindToObject(pidl, NULL, IID_IPersist, (LPVOID *) &pPersist)))
{
hr = pPersist->GetClassID(&pathCLSID);
pPersist->Release();
}
pMalloc->Free(pidl);
}
pshfDesktop->Release();
}
pMalloc->Release();
}
return hr;
}
//Not supported on Windows 2000 since SHParseDisplayName wasn't implemented then
//HRESULT GetFolderCLSID(LPCWSTR pszPath, CLSID& pathCLSID)
//{
// SHDESCRIPTIONID did = {0};
// HRESULT hr = 0;
// LPITEMIDLIST pidl = NULL;
// if (SUCCEEDED(hr = SHParseDisplayName(pszPath, NULL, &pidl, 0, NULL))) //not supported by Windows 2000
// {
// IShellFolder *psf = NULL;
// LPCITEMIDLIST pidlChild = NULL;
// if (SUCCEEDED(hr = SHBindToParent(pidl, IID_IShellFolder, (void**)&psf, &pidlChild)))
// {
// hr = SHGetDataFromIDList(psf, pidlChild, SHGDFIL_DESCRIPTIONID, &did, sizeof(did));
// psf->Release();
// pathCLSID = did.clsid;
// }
// CoTaskMemFree(pidl);
// }
// return hr;
//}
Другие советы
У Рэймонда Чена есть ответ - Как я могу определить, что каталог на самом деле является корзиной?
Немного поздно, но, возможно, лучше поздно, чем никогда...
После отладки shell32.dll я обнаружил, что для каждой версии Windows путь к возврату жестко задан и, кроме того, зависит от файловой системы этого диска.Я тестировал это на Windows XP, Vista и Windows7:
Пусть X:мы хотим получить путь к корзине, и пусть SID будет SID текущего пользователя, тогда:
switchif(OsType) {
case WindowsXP:
{
if(PartitionType("X:") == NTFS)
{
printf("Path is: X:\\Recycler\\SID\\");
}
else
{
printf("Path is X:\\RECYCLED\\");
}
}
case WindowsVista:
case Windows7:
{
if(PartitionType("X:") == NTFS)
{
printf("Path is: X:\\$Recycle.bin\\SID\\");
}
else
{
printf("Path is X:\\$RECYCLE.BIN\\");
}
}
}
В статье вики представлены те же факты:http://en.wikipedia.org/wiki/Recycle_Bin_%28Windows%29
В Win32 используйте SHGetSpecialFolderLocation ( Местоположение специальной папки).Пройти CSIDL_БИТБАКЕТ в качестве параметра CDIL.