I see that one should do this to compare PIDLs: IShellFolder::CompareIDs().
In particular, I'm trying to detect if a given absolute PIDL (or relative) is that of the Control Panel.
However, in practice I end up with two PIDLs which IShellFolder::CompareIDs() claims are not equal, when they should be (looking at the GetDisplayName() for each, I can see that we're indeed looking at the Control Panel).
Basically, I'm obtaining the absolute PIDL for the Control panel by:
PIDL iidControlPanel = nullptr;
SHGetSpecialFolderLocation(hwnd, CSIDL_CONTROLS, &iidControlPanel);
And then comparing the incoming enumerated shell object like so (see here for context - in a nutshell this is looking at the results of enumerating the desktop shell namespace inside of CMFCShellTreeCtrl):
bool bIsControlPanel = CompareAbsolutePIDLs(iidControlPanel, pItem->pidlFQ);
For reference, here's the comparison function:
bool CompareAbsolutePIDLs(PIDLIST_ABSOLUTE pidl1, PIDLIST_ABSOLUTE pidl2)
{
    CComPtr<IShellFolder> ishDesk;
    SHGetDesktopFolder(&ishDesk);
    HRESULT hr = ishDesk->CompareIDs(SHCIDS_CANONICALONLY, pidl1, pidl2);
    return SUCCEEDED(hr) && HRESULT_CODE(hr) == 0;
}
In the debugger, I can see that GetDisplayName() for each returns:
"::{26EE0668-A00A-44D7-9371-BEB064C98683}\0"
"::{26EE0668-A00A-44D7-9371-BEB064C98683}"
Here you can see the hex dump of the PIDLs:
1f 70 68 06 ee 26 0a a0 d7 44 93 71 be b0 64 c9 86 83 *0c* 00
1f 70 68 06 ee 26 0a a0 d7 44 93 71 be b0 64 c9 86 83 *00* 00
The underlying PIDLS are also binary identical excepting the penultimate value (00 vs. 0c). I'm currently at a loss as to why they're different, or what I can do to resolve this issue?!
These really are different shell objects. You can pass the PIDLs you got to SHGetNameFromIDList() with the SIGDN_NORMALDISPLAY option to convert them to readable strings. The long PIDL (with 0x0c) converts to "All Control Panel Items", the short one converts to "Control Panel".
This problem got started by using SHGetSpecialFolderLocation() to retrieve the virtual folder for the Control Panel controls. Distinct from the Control Panel object off the desktop root. I think you need to fix this by obtaining the PIDL for Control Panel and ignore the virtual folder. One way to do so is by using ILCloneFirst to convert the virtual folder to the root object:
 PITEMID_CHILD controlPanel = ILCloneFirst(iidControlPanel);
Or you can hard-code the control panel CLSID, "::{26EE0668-A00A-44D7-9371-BEB064C98683}", and convert it to a PIDL with SHParseDisplayName().
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