ShellExecuteEx guid路径

时间:2017-07-27 17:47:19

标签: c++ winapi guid

我有一个表格的路径

::{26EE0668-A00A-44D7-9371-BEB064C98683}\0\Fonts

如果我将其输入资源管理器地址栏,则会正确打开控制面板 - 字体。

如何使用ShellExecuteEx或类似的东西(包含guids的路径)执行相同的操作?

1 个答案:

答案 0 :(得分:3)

要打开此类GUID路径,您必须将shell:协议添加到路径中:

shell:::{26EE0668-A00A-44D7-9371-BEB064C98683}\0\Fonts

请注意,GUID路径经常在Windows版本之间进行更改,例如上述路径在我的Windows 10计算机上无法正常工作。

更好的方法是使用 known folder API 获取指向该文件夹的PIDL并使用ShellExecuteEx打开该PIDL:

#include <iostream>
#include <windows.h>
#include <shellapi.h>
#include <atlbase.h>  // CComHeapPtr

struct CComInit
{
    HRESULT hr = E_FAIL;
    CComInit() { hr = ::CoInitialize(nullptr); }
    ~CComInit() { if( SUCCEEDED(hr) ) ::CoUninitialize(); }
};

int main() 
{
    CComInit com;
    if( FAILED( com.hr ) )
    {
        std::cout << "Failed to initialize COM, error: 0x" << std::hex << com.hr << '\n';
        return static_cast<int>( com.hr );
    }

    CComHeapPtr<ITEMIDLIST> pidl;
    HRESULT hr = ::SHGetKnownFolderIDList( FOLDERID_Fonts, KF_FLAG_DEFAULT, nullptr, &pidl );
    if( FAILED( hr ) )
    {
        std::cout << "SHGetKnownFolderIDList failed with error 0x" << std::hex << hr << '\n';
        return static_cast<int>( hr );
    }

    SHELLEXECUTEINFOW si{ sizeof(si) };
    si.fMask = SEE_MASK_NOASYNC | SEE_MASK_IDLIST;
    si.nShow = SW_SHOWDEFAULT;
    si.lpVerb = L"open";
    si.lpIDList = pidl;

    if( !::ShellExecuteExW( &si ) )
    {
        DWORD err = ::GetLastError();
        std::cout << "ShellExecuteExW failed with error " << err << '\n';
        return static_cast<int>( err );
    }

    return 0;
}

备注:

  • ATL类CComHeapPtr是一个智能指针,它在析构函数中自动调用CoTaskMemFree()来正确释放由shell分配的PIDL。如果您不想使用它,则必须手动调用CoTaskMemFree()或使用您自己的RAII包装器来执行此操作。
  • 使用标记SEE_MASK_NOASYNC,因为该示例是一个控制台程序,将在ShellExecuteEx调用后立即退出。