Как создать ярлык для виртуальной папки в 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.com/en-us/library/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 когда я это использую? Или я не должен использовать это решение? Любой пример кода для этого?

0

Решение

Чтобы создать ярлык, вы можете использовать официальная документация. Вот пример кода, который создает ярлык для детей «Этот компьютер» (он же: 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, вам не нужно ничего перечислять, это только для демонстрационных целей.

1

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]