C++获取商店应用(msix应用)桌面快捷方式的安装目录

传统应用的快捷方式目标指向可执行文件的路径,但是对于商店应用(也叫msix打包应用),则指向一个奇怪的字符串,使用IShellLink::GetPath获取路径时,则得到的是空字符串,而我们的最终目的是要拿到应用的安装路径,那该怎么办呢?

首先解释一下,那个奇怪的字符串叫AUMID(App User Model Id),由应用包系列名称AppInfo.PackageFamilyName和应用标识符AppInfo.Id组成。

分3步获取安装目录

1,先获取快捷方式的PIDL

cpp 复制代码
HRESULT hr = S_OK;
IShellLinkW* psl = NULL;
IPersistFile* psf = NULL;
LPITEMIDLIST pidlLnk = NULL;
do
{
    hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLinkW, (LPVOID*)&psl);
    if (FAILED(hr)) {
        break;
    }
    hr = psl->QueryInterface(IID_IPersistFile, (LPVOID*)&psf);
    if (FAILED(hr)) {
        break;
    }
    //加载快捷方式
    hr = psf->Load(L"C:\\Users\\Xyy\\Desktop\\Microsoft Teams - 快捷方式.lnk", STGM_READ);
    if (FAILED(hr)) {
        break;
    }
    //获取快捷方式
    hr = psl->GetIDList(&pidlLnk);
    if (FAILED(hr)) {
        break;
    }
} while (false);
//释放资源
if (pidlLnk) ILFree(pidlLnk);

2,通过PIDL获取应用的AUMID

这里要注意,并非所有拿不到路径的快捷方式都是商店应用,因此要判断快捷方式的父目录是否是FOLDERID_AppsFolder,这是一个虚拟目录

cpp 复制代码
...
LPITEMIDLIST pidlAppsFolder = NULL;
PWSTR ppszName = NULL;
do
{
    ...
    //获取FOLDERID_AppsFolder的PIDL
    hr = SHGetKnownFolderIDList(FOLDERID_AppsFolder, 0, NULL, &pidlAppsFolder);
    if (FAILED(hr)) {
        break;
    }
    //判断当前快捷方式的父目录是否是FOLDERID_AppsFolder
    if (!ILIsParent(pidlAppsFolder, pidlLnk, FALSE)) {
        printf("此快捷方式不是商店应用");
        break;
    }
    //根据PIDL获取AUMID
    hr = SHGetNameFromIDList(pidlLnk, SIGDN_PARENTRELATIVEPARSING, &ppszName);
    if (FAILED(hr)) {
        break;
    }
} while (false);
//释放资源
...
if (pidlAppsFolder) ILFree(pidlAppsFolder);
if (ppszName) CoTaskMemFree(ppszName);

3,通过AUMID解析出packageFamily,再根据PackageManager解析出安装目录

PackageManagerWinRT的类型,如何在c++中使用WinRT,请参考C++/WinRT

以下代码需要管理员权限才能运行。

cpp 复制代码
//根据AUMID拿到packageFamily
std::wstring fullString(ppszName);
size_t pos = fullString.find(L'!');
if (pos == std::wstring::npos) {
    break;
}
std::wstring packageFamily = fullString.substr(0, pos);

std::wstring installPath = L"";
PackageManager packageManager;
//通过packageFamily查找所有包
auto packages = packageManager.FindPackages(packageFamily);
for (auto package : packages) {
    auto listEnties = package.GetAppListEntries();
    for (auto entry : listEnties) {
        if (entry.AppUserModelId() == ppszName) {
            installPath == package.InstalledPath();
            break;
        }
    }
    if (!installPath.empty()) {
        break;
    }
}
if (installPath.empty()) {
    break;
}
//找到安装目录
printf("找到安装目录:%ls", installPath.c_str());
相关推荐
guigu20126 分钟前
C++ 面向对象进阶:继承深化与多态详解
开发语言·c++
晚风予卿云月1 小时前
详解STL中stack_queue为什么选择deque作为默认容器
c++·stl·deque·stack_queue
charlie1145141912 小时前
精读C++20设计模式——结构型设计模式:代理模式
c++·学习·设计模式·代理模式·c++20·概论
序属秋秋秋4 小时前
《C++进阶之C++11》【可变参数模板 + emplace接口 + 新的类功能】
c++·笔记·学习·c++11·可变参数模板·emplace系列接口
Pocker_Spades_A4 小时前
C++程序设计上机作业(1)
开发语言·c++
Chen--Xing4 小时前
OpenMP并行化编程指南
c++·密码学·openmp
乱飞的秋天5 小时前
C++中的特殊成员函数
开发语言·c++
攻城狮7号5 小时前
【AI时代速通QT】第八节:Visual Studio与Qt-从项目迁移到多版本管理
c++·qt·跨平台·visual studio·qt vs tools
郝学胜-神的一滴5 小时前
QAxios研发笔记(一):在Qt环境下,构建Promise风格的Get请求接口
开发语言·c++·spring boot·qt·ajax·前端框架·软件工程
AA陈超5 小时前
虚幻引擎UE5专用服务器游戏开发-21 连招技能动画蒙太奇播放
c++·游戏·ue5·游戏引擎·虚幻