我正在寻找一种将文件夹(带有子文件夹)放入回收站的方法,条件如下:
它必须静默地完成 -- 没有任何 Windows UI。
文件夹绝不能被永久删除。如果不能放入回收站,我认为 API 会失败。
为类似CopyFileEx 的进程获取回调例程会。
到目前为止,我能够想出这个:
SHFILEOPSTRUCT sfo = {0};
sfo.wFunc = FO_DELETE;
sfo.pFrom = L"K:\\test del from USB\0"; //Folder on a USB stick
sfo.fFlags = FOF_ALLOWUNDO |
FOF_SILENT | FOF_NOCONFIRMATION | FOF_NOERRORUI | FOF_NOCONFIRMMKDIR |
FOF_WANTNUKEWARNING;
int res = SHFileOperation(&sfo);
BOOL bFullSuccess = res == 0 && !sfo.fAnyOperationsAborted;
这对位于 USB 闪存驱动器上的文件夹非常失败,即它被永久删除,尽管有 FOF_ALLOWUNDO
标志。
所以无论我做错了什么,还是SHFileOperation API非常错误!
知道如何执行我上面概述的操作吗?
编辑:我按照@Denis Anisimov 的建议实现了IRecycleBinManager::WillRecycle
方法,但显然还有更多内容。这是我的 C++ 版本。我需要的方法的第一个接口(interface)定义:
#if defined(__cplusplus) && !defined(CINTERFACE)
MIDL_INTERFACE("5869092D-8AF9-4A6C-AE84-1F03BE2246CC")
IRecycleBinManager : public IUnknown
{
public:
//function WillRecycle(const pszPath: LPCWSTR): HRESULT; stdcall;
virtual HRESULT STDMETHODCALLTYPE WillRecycle(
/* [string][in] */ __RPC__in LPCWSTR pszFile) = 0;
};
#endif
然后调用本身:
HRESULT hr;
CoInitializeEx(NULL, COINIT_DISABLE_OLE1DDE | COINIT_APARTMENTTHREADED);
// {4A04656D-52AA-49DE-8A09-CB178760E748}
const CLSID CLSID_RecycleBinManager = {0x4A04656D, 0x52AA, 0x49DE, {0x8A, 0x09, 0xCB, 0x17, 0x87, 0x60, 0xE7, 0x48}};
// {5869092D-8AF9-4A6C-AE84-1F03BE2246CC}
const IID IID_IRecycleBinManager = {0x5869092D, 0x8AF9, 0x4A6C, {0xAE, 0x84, 0x1F, 0x03, 0xBE, 0x22, 0x46, 0xCC}};
IRecycleBinManager* pIRBM = NULL;
hr = CoCreateInstance(CLSID_RecycleBinManager, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER,
IID_IRecycleBinManager, (void**) &pIRBM);
// hr = SHCoCreateInstance(NULL, &CLSID_RecycleBinManager, NULL, IID_IRecycleBinManager, (void **)&pIRBM);
if (SUCCEEDED(hr))
{
hr = pIRBM->WillRecycle(L"C:\\test del"); //Crashes
pIRBM->Release();
}
不幸的是,我在应该调用 WillRecycle
方法的那一行遇到了这个错误:
Run-Time Check Failure #0 - The value of ESP was not properly saved across a function call. This is usually a result of calling a function declared with one calling convention with a function pointer declared with a different calling convention.
最佳答案
每个驱动器都有自己的回收站。当您从驱动器 С: 中删除文件时,它应该被移动到驱动器 С: 上的回收站。当您从 USB 驱动器中删除文件时,它应该被移动到 USB 驱动器上的回收站。但是当 USB 驱动器没有回收站时,文件将被永久删除。这是 Windows 的默认行为。
FOF_ALLOWUNDO 标志只是推荐。 MSDN 关于 FOF_ALLOWUNDO 标志说:
Preserve undo information, if possible.
因此,即使您使用 FOF_ALLOWUNDO 标志,Windows 永久删除文件时也不会出现任何错误。
我看到的唯一方法是在删除操作之前使用 SHQueryRecycleBin 函数检查驱动器上是否存在回收站(正如 Alex Farber 在评论中指出的那样)。但即使回收站出现,也不能完全保证文件将被删除到回收站。回收站有最大容量限制,可能已经满了。
更新
你可以使用 hack。您可以使用自己的代码模拟将文件删除到回收站中,这将在 C:\$Recycle.Bin\UserSID 文件夹中创建所有必要的系统记录。我在 Windows 7 上测试了这个方法,它工作正常。它允许忽略回收站最大大小的限制。它还允许将文件从 USB 移动到任何驱动器上的回收站。
更新 2
对于 Vista+,您可以使用未记录的接口(interface) IRecycleBinManager(俄语说明可在网页 http://rcrrad.com/2010/10/14/bitbucket-interfaces/ 上找到):
const
IID_IEnumRecycleItems: TGUID = '{6E325F88-D12F-49E5-895B-8EC98630C021}';
IID_IRecycle: TGUID = '{0125E62F-8349-443A-854B-A55FB84CFA35}';
IID_IRecycleBin: TGUID = '{F964AD97-96F4-48AB-B444-E8588BC7C7B3}';
IID_IRecycleBinManager: TGUID = '{5869092D-8AF9-4A6C-AE84-1F03BE2246CC}';
CLSID_RecycleBinManager: TGUID = '{4A04656D-52AA-49DE-8A09-CB178760E748}';
type
{ Тип Корзины }
tagRECYCLEBIN_TYPE = (RBTYPE_VOLUME, RBTYPE_KNOWNFOLDER);
TRecycleBinType = tagRECYCLEBIN_TYPE;
{ Данные об удаленном элементе }
PDeletedItem = ^TDeletedItem;
tagDELETEDITEM = packed record
dwFileSizeLow: DWORD;
dwFileSizeHigh: DWORD;
ftDeletionTime: TFileTime;
szOriginalPath: array[0..Pred(MAX_PATH)] of WideChar;
szDisplacedPath: array[0..Pred(MAX_PATH)] of WideChar;
end;
TDeletedItem = tagDELETEDITEM;
{ Перечислитель элементов Корзины }
IEnumRecycleItems = interface(IUnknown)
['{6E325F88-D12F-49E5-895B-8EC98630C021}']
{ celt может быть равен только единице }
function Next(celt: ULONG; out rgelt: TDeletedItem;
var pceltFetched: ULONG): HRESULT; stdcall;
{ Not Implemented }
function Skip(celt: ULONG): HRESULT; stdcall;
function Reset: HRESULT; stdcall;
{ Not Implemented }
function Clone(out ppenum: IEnumRecycleItems): HRESULT; stdcall;
end;
{ "Интерфейс-переходник" между IRecycleBin и IRecycleBinManager }
IRecycle = interface(IUnknown)
['{0125E62F-8349-443A-854B-A55FB84CFA35}']
function Compact(): HRESULT; stdcall;
function GetFileData(const pszPath: LPCWSTR;
out lpData: TDeletedItem): HRESULT; stdcall;
function GetItemCount(out lpCount: TLargeInteger): HRESULT; stdcall;
function GetUsedSpace(out lpUsedSpace: TLargeInteger): HRESULT; stdcall;
function IsEmpty(): HRESULT; stdcall;
function PurgeAll(pfo: IFileOperation): HRESULT; stdcall;
function PurgeItems(const lpstrItems: LPCWSTR;
pfo: IFileOperation): HRESULT; stdcall;
function SuspendUpdating(fSuspend: BOOL): HRESULT; stdcall;
function RecycleItem(const lpstrItem: LPCWSTR; const dwAttrs: DWORD;
const iFileSize: TLargeInteger; out psi: IShellItem): HRESULT; stdcall;
function RestoreItems(const lpstrItems: LPCWSTR;
pfo: IFileOperation): HRESULT; stdcall;
function IsRecycled(const pszPath: LPCWSTR;
lpRecycled: PBOOL): HRESULT; stdcall;
function EnumItems(dwFlags: DWORD;
out EnumRecycleItems: IEnumRecycleItems): HRESULT; stdcall;
function WillRecycle(const pszPath: LPCWSTR): HRESULT; stdcall;
end;
{ Представляет определенную Корзину на конкретном диске }
IRecycleBin = interface(IUnknown)
['{F964AD97-96F4-48AB-B444-E8588BC7C7B3}']
function Compact(): HRESULT; stdcall;
function GetFileData(const pszPath: LPCWSTR;
out lpData: TDeletedItem): HRESULT; stdcall;
function GetItemCount(out lpCount: TLargeInteger): HRESULT; stdcall;
function GetUsedSpace(out lpUsedSpace: TLargeInteger): HRESULT; stdcall;
function IsEmpty(): HRESULT; stdcall;
function PurgeAll(pfo: IFileOperation): HRESULT; stdcall;
function PurgeItems(const lpstrItems: LPCWSTR;
pfo: IFileOperation): HRESULT; stdcall;
function SuspendUpdating(fSuspend: BOOL): HRESULT; stdcall;
function RecycleItem(const lpstrItem: LPCWSTR; const dwAttrs: DWORD;
const iFileSize: TLargeInteger; out psi: IShellItem): HRESULT; stdcall;
function RestoreItems(const lpstrItems: LPCWSTR;
pfo: IFileOperation): HRESULT; stdcall;
function IsRecycled(const pszPath: LPCWSTR;
lpRecycled: PBOOL): HRESULT; stdcall;
function EnumItems(dwFlags: DWORD;
out EnumRecycleItems: IEnumRecycleItems): HRESULT; stdcall;
function WillRecycle(const pszPath: LPCWSTR): HRESULT; stdcall;
function Initialize(const rbType: TRecycleBinType;
const pszID: LPCWSTR): HRESULT; stdcall;
function GetTypeID(out rbType: TRecycleBinType;
var pszID: LPWSTR): HRESULT; stdcall;
function GetIDList(out ppidl: PItemIDList): HRESULT; stdcall;
function GetLocation(pszPathBuffer: LPWSTR;
cchMax: DWORD): HRESULT; stdcall;
function GetMaxCapacityRange(out lpMin: TLargeInteger;
out lpMax: TLargeInteger): HRESULT; stdcall;
function GetMaxCapacity(out lpCapacity: TLargeInteger): HRESULT; stdcall;
function SetMaxCapacity(const lpCapacity: TLargeInteger): HRESULT; stdcall;
function GetPurgeOnDelete(out fNukeOnDelete: BOOL): HRESULT; stdcall;
function SetPurgeOnDelete(const fNukeOnDelete: BOOL): HRESULT; stdcall;
end;
{ Менеджер всех Корзин данной ОС }
IRecycleBinManager = interface(IUnknown)
['{5869092D-8AF9-4A6C-AE84-1F03BE2246CC}']
function Compact(): HRESULT; stdcall;
function GetFileData(const pszPath: LPCWSTR;
out lpData: TDeletedItem): HRESULT; stdcall;
function GetItemCount(out lpCount: TLargeInteger): HRESULT; stdcall;
function GetUsedSpace(out lpUsedSpace: TLargeInteger): HRESULT; stdcall;
function IsEmpty(): HRESULT; stdcall;
function PurgeAll(pfo: IFileOperation): HRESULT; stdcall;
function PurgeItems(const lpstrItems: LPCWSTR;
pfo: IFileOperation): HRESULT; stdcall;
function SuspendUpdating(fSuspend: BOOL): HRESULT; stdcall;
{ Not Implemented }
function RecycleItem(const lpstrItem: LPCWSTR; const dwAttrs: DWORD;
const iFileSize: TLargeInteger; out psi: IShellItem): HRESULT; stdcall;
function RestoreItems(const lpstrItems: LPCWSTR;
pfo: IFileOperation): HRESULT; stdcall;
function IsRecycled(const pszPath: LPCWSTR;
lpRecycled: PBOOL): HRESULT; stdcall;
function EnumItems(dwFlags: DWORD;
out EnumRecycleItems: IEnumRecycleItems): HRESULT; stdcall;
function WillRecycle(const pszPath: LPCWSTR): HRESULT; stdcall;
function DelayCompaction(const fDelay: BOOL): HRESULT; stdcall;
function GetRecycleBinCount(out iCount: Integer): HRESULT; stdcall;
function GetRecycleBinAt(const index: Integer; const iid: TGUID;
out ppv): HRESULT; stdcall;
function GetRecycleBin(const pszPath: LPCWSTR; const iid: TGUID;
out ppv): HRESULT; stdcall;
function Refresh(): HRESULT; stdcall;
end;
您可以使用以下代码检查文件被删除到回收站的可能性:
function CanFileBeDeletedToRecycleBin(const AFileName: UnicodeString): Boolean;
var
RecycleBinManager: IRecycleBinManager;
begin
OleCheck(CoCreateInstance(CLSID_RecycleBinManager, nil, CLSCTX_INPROC_SERVER or CLSCTX_LOCAL_SERVER, IRecycleBinManager, RecycleBinManager));
try
Result := RecycleBinManager.WillRecycle(PWideChar(AFileName)) = S_OK;
finally
RecycleBinManager := nil;
end;
end;
更新 3
您也可以尝试使用以下代码将对象删除到回收站:
function GetObjectSize(const AFileName: UnicodeString): Int64;
var
FindHandle: THandle;
FindData: TWin32FindDataW;
S: Int64;
begin
Result := 0;
FindHandle := FindFirstFileW(PWideChar(AFileName), FindData);
if FindHandle = INVALID_HANDLE_VALUE then
RaiseLastOSError;
try
repeat
if (FindData.cFileName <> UnicodeString('.')) and (FindData.cFileName <> '..') then
begin
Int64Rec(S).Lo := FindData.nFileSizeLow;
Int64Rec(S).Hi := FindData.nFileSizeHigh;
Result := Result + S;
if FindData.dwFileAttributes and FILE_ATTRIBUTE_DIRECTORY <> 0 then
Result := Result + GetObjectSize(AFileName + '\*.*');
end;
until not FindNextFileW(FindHandle, FindData);
finally
FindClose(FindHandle);
end;
end;
procedure DeleteToRecycleBin(const AFileName: UnicodeString);
var
Attr: DWORD;
Size: Int64;
RecycleBinManager: IRecycleBinManager;
RecycleBin: IRecycleBin;
ShellItem: IShellItem;
begin
OleCheck(CoCreateInstance(CLSID_RecycleBinManager, nil, CLSCTX_INPROC_SERVER or CLSCTX_LOCAL_SERVER, IRecycleBinManager, RecycleBinManager));
try
OleCheck(RecycleBinManager.GetRecycleBin(PWideChar(AFileName), IRecycleBin, RecycleBin));
try
Attr := GetFileAttributes(PWideChar(AFileName));
Size := GetObjectSize(AFileName);
OleCheck(RecycleBin.RecycleItem(PWideChar(AFileName), Attr, Size, ShellItem));
ShellItem := nil;
finally
RecycleBin := nil;
end;
finally
RecycleBinManager := nil;
end;
end;
关于c++ - 如何将 "safely"文件夹删除到回收站,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23725225/