Как создать ярлык для виртуальной папки в C++ на Windows 7?

Я использую платформу Windows 7. Мне нужно создать ярлык для виртуальной папки в Windows 7. Я использую пример Windows 7 SDK для создания виртуальной папки в разделе "Компьютер":

Название примера проекта называется ExplorerDataProvider он определяет CLSID для класса IShellFolder:

// add classes supported by this module here
const CLASS_OBJECT_INIT c_rgClassObjectInit[] =
{
{ &CLSID_FolderViewImpl,            CFolderViewImplFolder_CreateInstance },
{ &CLSID_FolderViewImplContextMenu,CFolderViewImplContextMenu_CreateInstance },
};

Определение для CFolderViewImplFolder_CreateInstance является:

HRESULT CFolderViewImplFolder_CreateInstance(REFIID riid, void **ppv)
{
*ppv = NULL;
CFolderViewImplFolder* pFolderViewImplShellFolder = new (std::nothrow) CFolderViewImplFolder(0);
HRESULT hr = pFolderViewImplShellFolder ? S_OK : E_OUTOFMEMORY;
if (SUCCEEDED(hr))
{
    hr = pFolderViewImplShellFolder->QueryInterface(riid, ppv);
    pFolderViewImplShellFolder->Release();
}
return hr;
}

И CFolderViewImplFolder воплощать в жизнь IShellFolder2 драм IPersistFolder2, Я нашел аналогичный код, который используется для создания ярлыка для принтера здесь: https://www.codeproject.com/Articles/596642/Creating-a-shortcut-programmatically-in-Cplusplus и в https://msdn.microsoft. ком / EN-US / библиотека /aa969393.aspx#Shellink_Item_Identifiers

Получив идентификатор класса для IShellFolder, вы можете вызвать функцию CoCreateInstance для получения адреса интерфейса. Затем вы можете вызвать интерфейс, чтобы перечислить объекты в папке и получить адрес идентификатора элемента для объекта, который вы ищете. Наконец, вы можете использовать адрес в вызове функции-члена IShellLink::SetIDList, чтобы создать ярлык для объекта.

Я пересмотрел

hr = SHGetMalloc(&pMalloc);
hr = SHGetDesktopFolder( &pDesktopFolder );
hr = SHGetSpecialFolderLocation( NULL, CSIDL_PRINTERS, &netItemIdLst );
hr = pDesktopFolder->BindToObject( netItemIdLst, NULL, IID_IShellFolder, (void **)&pPrinterFolder );

в

// testFolder is the CLSID for the virtual folder implementation
hr = CoCreateInstance(testFolder, NULL, CLSCTX_INPROC_SERVER, IID_IShellFolder, (LPVOID*)&pVirtualFolder);

или же

hr = CoCreateInstance(testFolder, NULL, CLSCTX_INPROC_SERVER, IID_IShellFolder2, (LPVOID*)&pVirtualFolder);

Но pVirtualFolder все еще НЕДЕЙСТВИТЕЛЕН, и это печатает, что "не может найти соответствующий интерфейс".

Что-то не так с CoCreateInstance когда я это использую? Или я не должен использовать это решение? Любой пример кода для этого?

1 ответ

Решение

Для создания ярлыка вы можете использовать официальную документацию. Вот пример кода, который создает ярлык для детей "This PC" (он же ComputerFolder)

int main()
{
    CoInitialize(NULL);
    // I've used my Apple's iCloud as an example (name is in french)
    // it's a virtual folder, a shell namespace extension
    HRESULT hr = CreateComputerChildShortCut(L"Photos iCloud", L"c:\\temp\\my icloud");
    printf("hr:0x%08X\n", hr);
    CoUninitialize();
    return 0;
}

HRESULT CreateComputerChildShortCut(LPWSTR childDisplayName, LPWSTR path)
{
    // get My Computer folder's ShellItem (there are other ways for this...)
    CComPtr<IShellItem> folder;
    HRESULT hr = SHCreateItemInKnownFolder(FOLDERID_ComputerFolder, 0, NULL, IID_PPV_ARGS(&folder));
    if (FAILED(hr)) return hr;

    // enumerate children
    CComPtr<IEnumShellItems> items;
    hr = folder->BindToHandler(NULL, BHID_EnumItems, IID_PPV_ARGS(&items));
    if (FAILED(hr)) return hr;

    for (CComPtr<IShellItem> item; items->Next(1, &item, NULL) == S_OK; item.Release())
    {
        // get parsing path (if's often useful)
        CComHeapPtr<wchar_t> parsingPath;
        item->GetDisplayName(SIGDN_DESKTOPABSOLUTEPARSING, &parsingPath);
        wprintf(L"Path: %s\n", parsingPath);

        // get display name
        CComHeapPtr<wchar_t> displayName;
        item->GetDisplayName(SIGDN_NORMALDISPLAY, &displayName);
        wprintf(L" Name: %s\n", displayName);

        if (!lstrcmpi(childDisplayName, displayName))
        {
            // get pidl
            // it's the unambiguous way of referencing a shell thing
            CComHeapPtr<ITEMIDLIST> pidl;
            hr = SHGetIDListFromObject(item, &pidl);
            if (FAILED(hr)) return hr;

            // create an instance of the standard Shell's folder shortcut creator
            CComPtr<IShellLink> link;
            hr = link.CoCreateInstance(CLSID_FolderShortcut);
            if (FAILED(hr)) return hr;

            // just use the pidl
            hr = link->SetIDList(pidl);
            if (FAILED(hr)) return hr;

            CComPtr<IPersistFile> file;
            hr = link->QueryInterface(&file);
            if (FAILED(hr)) return hr;

            // save the shortcut (we could also change other IShellLink parameters)
            hr = file->Save(path, FALSE);
            if (FAILED(hr)) return hr;
            break;
        }
    }
    return S_OK;
}

Конечно, если вы знаете абсолютный путь синтаксического анализа или абсолютный pidl, вам не нужно ничего перечислять, это только для демонстрационных целей.

Другие вопросы по тегам