I am attempting to write two functions that add and remove a folder from a IShellLibrary
. I started with this, but the function produces an exception in System._IntfClear
:
First chance exception at $000007FEFE 168BC4. Exception class $C0000005 with Message 'c0000005 ACCESS_VIOLATION'.
The SHAddFolderPathToLibrary
is the line that causes the exception.
I guess I need to add the library name to the function?
function AddFolderToLibrary(AFolder: string): HRESULT;
{ Add AFolder to Windows 7 library. }
var
plib: IShellLibrary;
begin
Result := CoCreateInstance(CLSID_ShellLibrary, nil, CLSCTX_INPROC_SERVER,
IID_IShellLibrary, plib);
if SUCCEEDED(Result) then
begin
Result := SHAddFolderPathToLibrary(plib, PWideChar(AFolder));
end;
end;
function RemoveFolderFromLibrary(AFolder: string): HRESULT;
{ Remove AFolder from Windows 7 library. }
var
plib: IShellLibrary;
begin
Result := CoCreateInstance(CLSID_ShellLibrary, nil, CLSCTX_INPROC_SERVER,
IID_IShellLibrary, plib);
if SUCCEEDED(Result) then
begin
Result := SHRemoveFolderPathFromLibrary(plib, PWideChar(AFolder));
end;
end;
Tap or click to open File Explorer. Select the library where you want to remove a folder. Tap or click the Library Tools tab, and then tap or click Manage library. In the dialog box that appears, select the folder you want to remove, tap or click Remove, and then tap or click OK.
Click File Explorer icon. Go to the location where stores your file or folder. Click the name of the file or folder you wish to delete. Press the delete key (on the keyboard) or right click the file or folder and click Delete.
1] Via File Explorer Press Windows key + E to open File Explorer. Right-click or press and hold on an empty area inside the navigation pane. Click/tap on Show libraries to toggle add (checked) or remove (unchecked – default) per your requirement. Exit File Explorer when done.
The problem here is that the Embarcadero engineer who translated SHAddFolderPathToLibrary
does not understand COM reference counting, and how it is handled by different compilers.
Here's how SHAddFolderPathToLibrary
is implemented in the C++ header file Shobjidl.h
. It's actually an inline wrapper of other core API calls:
__inline HRESULT SHAddFolderPathToLibrary(_In_ IShellLibrary *plib,
_In_ PCWSTR pszFolderPath)
{
IShellItem *psiFolder;
HRESULT hr = SHCreateItemFromParsingName(pszFolderPath, NULL,
IID_PPV_ARGS(&psiFolder));
if (SUCCEEDED(hr))
{
hr = plib->AddFolder(psiFolder);
psiFolder->Release();
}
return hr;
}
And the Delphi translation is very faithful, indeed too faithful:
function SHAddFolderPathToLibrary(const plib: IShellLibrary;
pszFolderPath: LPCWSTR): HResult;
var
psiFolder: IShellItem;
begin
Result := SHCreateItemFromParsingName(pszFolderPath, nil, IID_IShellItem,
psiFolder);
if Succeeded(Result) then
begin
Result := plib.AddFolder(psiFolder);
psiFolder._Release();
end;
end;
The problem is the call to _Release
. The Delphi compiler manages reference counting, and so this explicit call to _Release
is bogus and should not be there. Since the compiler will arrange for a call to _Release
, this extra one simply unbalances the reference counting. The reason why _AddRef
and _Release
are prefixed with _
is to remind people not to call them and to let the compiler do that.
The call to Release
in the C++ version is accurate because C++ compilers don't automatically call Release
for you unless you wrap the interface in a COM smart pointer. But the Embarcadero engineer has blindly copied it across and you are left with the consequences. Clearly this code has never even been executed by the Embarcadero engineers.
You'll need to supply your own corrected implementation of this function. And also any other erroneously translated function. Search for _Release
in the ShlObj
unit, and remove them in your corrected versions. There are other bugs in the translation, so watch out. For example, SHLoadLibraryFromItem
(and others) declare local variable plib: ^IShellLibrary
which should be plib: IShellLibrary
.
I submitted a QC report: QC#117351.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With