如何将SHCreateItemFromParsingName与Shell命名空间中的名称一起使用?

Ian*_*oyd 5 windows winapi windows-shell

我正在使用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;
}
Run Code Online (Sandbox Code Playgroud)

注意:A IShellItem于2006年左右推出,旨在为Windows 95时代IShellFolder+ pidl构造提供方便的包装。你甚至可以问IShellItem咳嗽了它的基础IShellFolder,并pidlIParentAndItem.GetParentAndItem接口和方法。

不同的事物具有不同的显示名称

我可以在shell名称空间中找到一些众所周知的位置,并查看它们的绝对解析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" |
Run Code Online (Sandbox Code Playgroud)

用户输入时如何解析它们?

我可以IFileOpenDialog用来让用户选择这些文件夹之一。但我真的希望用户能够输入

  • “ C:\ Users”
  • “ C:\ Windows \ Fonts”
  • “这台电脑”
  • “回收站”
  • “图书馆”
  • “启动”
  • “字体”

并可以将它解析为一个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列表包含喜欢的东西:

  • C:\ Windows
  • 回收站
  • 这台电脑

并可以稍后解析它们。

And*_*ers 6

调试 Explorer 并看看它是如何做到的会很有趣。

我的建议是;如果初始解析失败,则添加shell:到路径字符串并尝试使用SHParseDisplayName. 如果您STR_PARSE_SHELL_PROTOCOL_TO_FILE_OBJECTS在绑定上下文中设置,您还可以绑定到特殊文件。shell: 协议能够解析特殊/已知文件夹的内部/规范名称,但我不知道它是否也检查显示名称。

编辑:

我现在有机会玩一下,shell: prefix 并不是一个巨大的改进,因为它只检查已知的文件夹规范名称:

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);
}
Run Code Online (Sandbox Code Playgroud)

给我这个输出:

                    | 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
Run Code Online (Sandbox Code Playgroud)

在 Windows 8 上SHCreateItemFromParsingName调用SHParseDisplayName(使用STR_PARSE_AND_CREATE_ITEMSTR_PARSE_TRANSLATE_ALIASES),因此即使是 Microsoft 也无法在其 API 中分离解析名称和显示名称。

如果您想远离未记录的界面,则必须添加第三遍,在其中检查已知文件夹显示名称。或者,正如 Raymond Chen 在评论中所建议的那样;根据项目显示名称手动解析每个路径组件IShellFolder