我正在使用 SHCreateItemFromParsingName
将路径转换为 IShellItem
:
IShellItem ParseName(String path)
{
IShellItem shellItem;
HRESULT hr = SHCreateItemFromParsingName(path, null, IShellItem, out shellItem);
if (Failed(hr))
throw new ECOMException(hr);
return shellItem;
}
不同的事物具有不同的显示名称
我可以在shell namespace 中找到一些众所周知的位置,并查看它们的绝对解析(
SIGDN_DESKTOPABSOLUTEPARSING
)和编辑(SIGDN_DESKTOPABSOLUTEEDITING
)显示名称:| Path | Editing | Parsing |
|-------------------|-----------------------|-------------------------------------------------------------------|
| C:\ | "C:\" | "C:\" |
| C:\Windows | "C:\Windows" | "C:\Windows" |
| Desktop | "Desktop" | "C:\Users\Ian\Desktop" |
| Computer | "This PC" | "::{20D04FE0-3AEA-1069-A2D8-08002B30309D}" |
| Recycle Bin | "Recycle Bin" | "::{645FF040-5081-101B-9F08-00AA002F954E}" |
| Documents Library | "Libraries\Documents" | "::{031E4825-7B94-4DC3-B131-E946B44C8DD5}\Documents.library-ms" " |
| Startup | "C:\Users\Ian\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup" | "C:\Users\Ian\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup" |
用户输入时如何解析它们?
我可以使用
IFileOpenDialog
让用户选择这些文件夹之一。但我真的希望用户能够输入并能够将其解析为
IShellItem
。问题是某些路径不是
SHCreateItemFromParsingName
解析的而不是:SHCreateItemFromParsingName("C:\")
:解析SHCreateItemFromParsingName("C:\Windows")
:解析SHCreateItemFromParsingName("")
:解析(但变为“此PC ”)SHCreateItemFromParsingName("This PC")
:失败SHCreateItemFromParsingName("Recycle Bin")
:SHCreateItemFromParsingName("Libraries")
:失败SHCreateItemFromParsingName("OneDrive")
:失败SHCreateItemFromParsingName("Libraries\Documents")
:失败SHCreateItemFromParsingName("Network")
:失败SHCreateItemFromParsingName("Startup")
:失败同时,我的程序使用的IFileOpenDialog控件可以很好地解析它们:
我如何解析用户可以键入的各种特殊Shell名称位置(Windows资源管理器和IFileOpen对话框可以解析)到该文件夹的
IShellItem
中?真正的问题是,我希望用户能够拥有一个包含以下内容的近期MRU列表:
并可以稍后解析它们。
最佳答案
调试Explorer并查看其工作方式将很有趣。
我的建议是;如果初始解析失败,则将shell:
放在路径字符串之前,然后尝试使用SHParseDisplayName
再次对其进行解析。如果在绑定(bind)上下文中设置 STR_PARSE_SHELL_PROTOCOL_TO_FILE_OBJECTS
,则还可以绑定(bind)到特殊文件。 shell:协议(protocol)能够解析特殊/已知文件夹的内部/规范名称,但是我不知道它是否还会检查显示名称。
编辑:
我现在有机会玩了一下,而shell:前缀并不是一个很大的改进,因为它只检查已知文件夹的规范名称:
PCWSTR paths[] = {
TEXT("C:\\"),
TEXT("C:\\Windows"),
TEXT(""),
TEXT("This PC"),
TEXT("MyComputerFolder"), // Canonical KF name
TEXT("Recycle Bin"),
TEXT("RecycleBinFolder"), // Canonical KF name
TEXT("Libraries"),
TEXT("OneDrive"),
TEXT("Libraries\\Documents"),
TEXT("Network"),
TEXT("NetworkPlacesFolder"), // Canonical KF name
TEXT("Startup"),
};
OleInitialize(0);
INT pad = 0, fill, i;
for (i = 0; i < ARRAYSIZE(paths); ++i) pad = max(pad, lstrlen(paths[i]));
for (i = 1, fill = printf("%-*s | Original | shell: |\n", pad, ""); i < fill; ++i) printf("-"); printf("\n");
for (i = 0; i < ARRAYSIZE(paths); ++i)
{
WCHAR buf[MAX_PATH], *p1 = NULL, *p2 = NULL;
IShellItem*pSI;
HRESULT hr = SHCreateItemFromParsingName(paths[i], NULL, IID_IShellItem, (void**) &pSI);
if (SUCCEEDED(hr)) pSI->GetDisplayName(SIGDN_DESKTOPABSOLUTEPARSING, &p1), pSI->Release();
wsprintf(buf, L"shell:%s", paths[i]);
HRESULT hr2 = SHCreateItemFromParsingName(buf, NULL, IID_IShellItem, (void**) &pSI);
if (SUCCEEDED(hr2)) pSI->GetDisplayName(SIGDN_DESKTOPABSOLUTEPARSING, &p2), pSI->Release();
wprintf(L"%-*s | %.8x | %.8x | %s\n", pad, paths[i], hr, hr2, p2 && *p2 ? p2 : p1 ? p1 : L"");
CoTaskMemFree(p1), CoTaskMemFree(p2);
}
给我这个输出:
| Original | shell: |
-------------------------------------------
C:\ | 00000000 | 80070003 | C:\
C:\Windows | 00000000 | 80070003 | C:\Windows
| 00000000 | 80070003 | ::{20D04FE0-3AEA-1069-A2D8-08002B30309D}
This PC | 80070002 | 80070003 |
MyComputerFolder | 80070002 | 00000000 | ::{20D04FE0-3AEA-1069-A2D8-08002B30309D}
Recycle Bin | 80070002 | 80070003 |
RecycleBinFolder | 80070002 | 00000000 | ::{645FF040-5081-101B-9F08-00AA002F954E}
Libraries | 80070002 | 00000000 | ::{031E4825-7B94-4DC3-B131-E946B44C8DD5}
OneDrive | 80070002 | 80070003 |
Libraries\Documents | 80070002 | 80070002 |
Network | 80070002 | 80070003 |
NetworkPlacesFolder | 80070002 | 00000000 | ::{F02C1A0D-BE21-4350-88B0-7367FC96EF3C}
Startup | 80070002 | 00000000 | C:\Users\Anders\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup
在Windows 8上,
SHCreateItemFromParsingName
调用SHParseDisplayName
(带有STR_PARSE_AND_CREATE_ITEM
和STR_PARSE_TRANSLATE_ALIASES
),因此即使Microsoft也无法在其API中分离解析和显示名称。如果要远离未公开的接口(interface),则必须添加第三遍,以检查已知的文件夹显示名称。或者,如雷蒙德·陈(Raymond Chen)在评论中建议的那样;根据该
IShellFolder
中的项目显示名称手动解析每个路径组件。