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

标签 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;
}

Note: A IShellItem was introduced around 2006 to provide a handy wrapper around the Windows 95-era IShellFolder+pidl constructs. You can even ask a IShellItem to cough up it's underlying IShellFolder and pidl with the IParentAndItem.GetParentAndItem interface and method.

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

我可以获取 shell 命名空间中一些众所周知的位置,并查看它们的绝对解析 (SIGDN_DESKTOABSOLUTEPARSING) 和编辑 ( SIGDN_DESKTOPABSOLUTEEDITING) 显示名称:

<表类="s-表"> <头> 路径 编辑 解析 <正文> C:\ “C:” “C:” C:\Windows "C:\Windows" "C:\Windows" 桌面 “桌面” "C:\Users\Ian\Desktop" 计算机 “这台电脑” "::{20D04FE0-3AEA-1069-A2D8-08002B30309D}" 回收站 “回收站” "::{645FF040-5081-101B-9F08-00AA002F954E}" 文档库 "图书馆\文档" "::{031E4825-7B94-4DC3-B131-E946B44C8DD5}\Documents.library-ms"" 启动 "C:\Users\Ian\AppData\Roaming\Microsoft\Windows\开始菜单\程序\启动" "C:\Users\Ian\AppData\Roaming\Microsoft\Windows\开始菜单\程序\启动"

用户输入时如何解析?

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

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

并能够将其解析为 IShellItem

问题是一些路径SHCreateItemFromParsingName没有解析:

  • SHCreateItemFromParsingName("C:\"):解析
  • SHCreateItemFromParsingName("C:\Windows"):解析
  • SHCreateItemFromParsingName(""):解析(但变成“This PC”)
  • SHCreateItemFromParsingName("This PC"):失败
  • SHCreateItemFromParsingName("回收站"):失败
  • SHCreateItemFromParsingName("Libraries"):失败
  • SHCreateItemFromParsingName("OneDrive"):失败
  • SHCreateItemFromParsingName("Libraries\Documents"):失败
  • SHCreateItemFromParsingName("Network"):失败
  • SHCreateItemFromParsingName("Startup"):失败

同时,我的程序使用的IFileOpenDialog 控件可以很好地解析它们:

enter image description here

我如何将用户可能输入的各种特殊 shell 名称位置(Windows 资源管理器和 IFileOpen 对话框可以解析)解析为该文件夹的 IShellItem

真正的问题是我希望用户能够拥有一个最近的 MRU 列表,其中包含如下内容:

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

并能够稍后解析它们。

最佳答案

调试 Explorer 并查看它是如何工作的会很有趣。

我的建议是;如果初始解析失败,请将 shell: 添加到路径字符串并尝试使用 SHParseDisplayName 再次解析它。如果你设置 STR_PARSE_SHELL_PROTOCOL_TO_FILE_OBJECTS在绑定(bind)上下文中,您还可以绑定(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_ITEMSTR_PARSE_TRANSLATE_ALIASES),因此即使是 Microsoft 也难以分离解析并在其 API 中显示名称。

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

关于windows - 如何将 SHCreateItemFromParsingName 与 shell 命名空间中的名称一起使用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42966489/

相关文章:

windows - 是否有免费/开源的带有图形进度的类似 wget 的 Windows 程序?

windows - cmd/c 以及文件路径中的 & 符号

c++ - 从 Alt + Tab 获取所有窗口,不包括 Metro 窗口 - Windows 8/8.1

c++ - Qt:如何通过外部程序打开文件, "open with..."对话框

winforms - 在 Windows 中更改另一个进程的任务栏图标

c++ - CFSTR_FILEDESCRIPTOR/CFSTR_FILECONTENTS 'Copying Files' 对话框丢失

windows - 无法使用 Composer 在 Windows 上安装 Mailgun 库

php - 想通过 php 站点运行我的 Windows 服务器上的 exe 文件

c# - 键枚举中的感叹号

java - 将windows中的复制操作添加到java应用程序中