C++:如何在Windows的开始菜单中创建快捷方式

3

我想知道如何在Windows上获取开始菜单文件夹的路径,然后创建一个快捷方式到可能包含非ASCII字符的路径。

3个回答

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);

其余部分则是创建快捷方式。将快捷方式名称和描述更改为所需值。

1
同样的想法,与接受的答案相同,但使用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;
}

0
Enigma的答案非常接近,但似乎在Windows 10上不起作用。当调用IPersistFile上的Save函数时,它会出现E_ACCESSDENIED错误。
我成功地根据他们的答案进行了适应,按照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;
}

你应该使用IShellLinkW_snwprintf等等,这样就不需要将wchar_t字符串转换为char字符串或者相反了。 - Remy Lebeau
我刚刚注意到bweber确实在他的答案中使用了SHGetKnownFolderPath函数。我没有阅读他们的答案的主要原因是,我看到了"Qt"并立即跳过它,因为我不使用Qt,也不会使用Qt,所以我假设他们需要某些Qt必需的功能。在我看来,应该从那个答案中删除任何对Qt的依赖。我将保留我的答案作为一个完整的示例,它不需要任何依赖项。 - Macoy
它可能没有依赖关系,但是它有不必要的数据转换,可以并且应该避免以防止任何不必要的数据损失。 - Remy Lebeau

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接