C++: как создать ярлык в меню "Пуск" в Windows
Я хотел бы знать, как получить путь к папке меню "Пуск" в Windows, а затем создать ярлык для пути, который может содержать символы, отличные от ASCII.
3 ответа
Вот решение. Он использует Qt, но это также возможно без. Тогда просто используйте std::wstring
вместо QString
, Для объединения путей и имен файлов вам придется использовать строковые операции вместо QDir
,
#include <shlobj.h>
bool createStartMenuEntry(QString targetPath) {
targetPath = QDir::toNativeSeparators(targetPath);
WCHAR startMenuPath[MAX_PATH];
HRESULT result = SHGetFolderPathW(NULL, CSIDL_COMMON_PROGRAMS, NULL, 0, startMenuPath);
if (SUCCEEDED(result)) {
QString linkPath = QDir(QString::fromWCharArray(startMenuPath)).absoluteFilePath("Shortcut Name.lnk");
CoInitialize(NULL);
IShellLinkW* shellLink = NULL;
result = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_ALL, IID_IShellLinkW, (void**)&shellLink);
if (SUCCEEDED(result)) {
shellLink->SetPath(targetPath.toStdWString().c_str());
shellLink->SetDescription(L"Shortcut Description");
shellLink->SetIconLocation(targetPath.toStdWString().c_str(), 0);
IPersistFile* persistFile;
result = shellLink->QueryInterface(IID_IPersistFile, (void**)&persistFile);
if (SUCCEEDED(result)) {
result = persistFile->Save(linkPath.toStdWString().c_str(), TRUE);
persistFile->Release();
} else {
return false;
}
shellLink->Release();
} else {
return false;
}
} else {
return false;
}
return true;
}
Это часть, которая получает местоположение папки меню пуск:
WCHAR startMenuPath[MAX_PATH];
HRESULT result = SHGetFolderPathW(NULL, CSIDL_COMMON_PROGRAMS, NULL, 0, startMenuPath);
Остальное - создание ярлыка. Замените имя и описание ярлыка на нужные вам значения.
Та же идея, что и принятого ответа, но метод Visual Studio.
Использование:
CString sProgramsPath = getenv("PROGRAMDATA");
CString sShortcutPath = sProgramsPath += "\\Microsoft\\Windows\\Start Menu\\Programs\\SHORTCUT_NAME.lnk";
// (that's .LNK)
CreateLink("C:\\target_file_path\\target_file_name.exe",
"sShortcutPath",
"C:\\target_file_path\\,
"Shortcut Description");
Функция:
/*============================================================================*/
// CreateLink - Uses the Shell's IShellLink and IPersistFile interfaces
// to create and store a shortcut to the specified object.
//
// Returns the result of calling the member functions of the interfaces.
//
// Parameters:
// lpszPathObj - Address of a buffer that contains the path of the object,
// including the file name.
// lpszPathLink - Address of a buffer that contains the path where the
// Shell link is to be stored, including the file name.
// lpszPath - Working directory of target Obj file
// lpszDesc - Address of a buffer that contains a description of the
// Shell link, stored in the Comment field of the link
// properties.
HRESULT CreateLink(
LPCSTR lpszPathObj,
LPCSTR lpszPathLink,
LPCSTR lpszPath,
LPCSTR lpszDesc )
/*============================================================================*/
{
IShellLink* psl = NULL;
HRESULT hres = CoInitialize(NULL);
if (!SUCCEEDED(hres))
LOGASSERT(FALSE);
// Get a pointer to the IShellLink interface. It is assumed that CoInitialize
// has already been called.
hres = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, (LPVOID*)&psl);
if (SUCCEEDED(hres))
{
IPersistFile* ppf;
// Set the path to the shortcut target and add the description.
psl->SetPath(lpszPathObj);
psl->SetDescription(lpszDesc);
psl->SetWorkingDirectory(lpszPath);
// Query IShellLink for the IPersistFile interface, used for saving the
// shortcut in persistent storage.
hres = psl->QueryInterface(IID_IPersistFile, (LPVOID*)&ppf);
if (SUCCEEDED(hres))
{
WCHAR wsz[MAX_PATH];
// Ensure that the string is Unicode.
MultiByteToWideChar(CP_ACP, 0, lpszPathLink, -1, wsz, MAX_PATH);
// Add code here to check return value from MultiByteWideChar
// for success.
// Save the link by calling IPersistFile::Save.
hres = ppf->Save(wsz, TRUE);
if (!SUCCEEDED(hres))
LOGASSERT(FALSE);
ppf->Release();
}
psl->Release();
}
CoUninitialize();
return hres;
}
Ответ Enigma очень близок, но, похоже, не работает в Windows 10. Ошибка возникает.E_ACCESSDENIED
при звонке вSave
функция включенаIPersistFile
.
Мне удалось адаптировать их ответ для работы, следуя документации Windows . Ключ в том, чтобы использоватьSHGetKnownFolderPath
скорее, чемgetenv("PROGRAMDATA")
:
#define WIN32_LEAN_AND_MEAN
#include "windows.h"
#include "combaseapi.h"
#include "shlobj.h"
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <stdbool.h>
bool windowsInstall()
{
char programFilesPath[1024] = {0};
{
WCHAR* wideProgramFilesPath = NULL;
SHGetKnownFolderPath(FOLDERID_Programs, 0, NULL, (&wideProgramFilesPath));
wcstombs(programFilesPath, wideProgramFilesPath, sizeof(programFilesPath));
CoTaskMemFree(wideProgramFilesPath);
}
char shortcutPath[2048] = {0};
/*
* Fill these in with your application details!
*
* The text before .lnk in shortcutPath will show as the shortcut text to the user */
snprintf(shortcutPath, sizeof(shortcutPath), "%s\\My Program.lnk", programFilesPath);
const char* executableFilename = "C:\\Path\\To\\MyProgram\\My_Program.exe";
const char* executableWorkingDirectory = "C:\\Path\\To\\MyProgram";
const char* shortcutDescription = "This is My Program's description.";
HRESULT result = CoInitialize(NULL);
if (!(SUCCEEDED(result)))
{
return false;
}
IShellLink* link = NULL;
result= CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, ((LPVOID*)(&link)));
if (!(SUCCEEDED(result)))
{
CoUninitialize();
return false;
}
link->SetPath(executableFilename);
link->SetWorkingDirectory(executableWorkingDirectory);
link->SetDescription(shortcutDescription);
IPersistFile* persistFile = NULL;
result= link->QueryInterface(IID_IPersistFile, ((void**)(&persistFile)));
if (!(SUCCEEDED(result)))
{
link->Release();
CoUninitialize();
return false;
}
WCHAR wideShortcutPath[1024];
MultiByteToWideChar(CP_ACP, 0, shortcutPath, -1, wideShortcutPath, (sizeof(wideShortcutPath) / sizeof(wideShortcutPath[0])));
result= persistFile->Save(wideShortcutPath, TRUE);
if (!(SUCCEEDED(result)))
{
persistFile->Release();
link->Release();
CoUninitialize();
return false;
}
persistFile->Release();
link->Release();
CoUninitialize();
return true;
}