草庐IT

c++ - 如何在 Windows 8 和 10 中枚举已安装的 StoreApps 及其 ID

coder 2024-06-10 原文

我想要获取的是所有已安装的 StoreApp 应用程序的 AppUserModelId,以便我可以将其传递给 IApplicationActivationManager->ActivateApplication

在 Windows 8 中它存储在注册表中,但在 Windows 10 中它不再存在。

网上有很多关于这个的问题,但即使经过几天的搜索,我也找不到令人满意的解决方案。

到目前为止,我有以下内容:

  1. 我创建了一个 IPackageManager 的实例,
  2. 我用当前用户的 SID 调用 FindPackagesByUserSecurityId()
  3. 我遍历返回的集合
  4. 我得到一个IPackage接口(interface)
  5. 从中我得到一个IPackageId接口(interface),
  6. 然后我调用 IPackageId->get_FamilyName()

例如,我在 Windows 10 上为 Windows 计算器设置了字符串“Microsoft.WindowsCalculator_8wekyb3d8bbwe”。

当我向这个字符串附加一个“!App”时,我有完整的 AppUserModelId 来启动 Windows 计算器:“Microsoft.WindowsCalculator_8wekyb3d8bbwe!App/强>”

但并非所有应用程序都在 FamilyName 后面使用“!App”。 例如 Spartan 使用 AppUserModelId “Microsoft.Windows.Spartan_cw5n1h2txyewy!Microsoft.Spartan.Spartan”,不以“!App”结尾。当我将“!Microsoft.Spartan.Spartan”替换为“!App”时,它不会启动 ->“此应用不支持指定的契约(Contract)”。

所以我的问题是我从哪里得到最后缺失的部分?

我在互联网上发现了一个 PowerShell 代码 ( http://poshcode.org/5702 ),它似乎做的事情非常相似:

Get-AppXPackage $PackageName -pv Package |
    Get-AppxPackageManifest | % {
        foreach($Application in $_.Package.Applications.Application) {
            if($Application.Id -like $AppId) {
                if($Protocol -and !($Application.Extensions.Extension.Protocol.Name | ? { ($_ + "://") -match (($Protocol -replace '\*','.*') + "(://)?") })) {
                    continue
                }

                [PSCustomObject]@{
                    # Notice the secret magic property:
                    PSTypeName = "Microsoft.Windows.Appx.Application"
                    AppUserModelId = $Package.PackageFamilyName + "!" + $Application.Id
                }
            }
        }
    }

我真的不明白这些神秘的 PowerShell 东西,但有一行对我来说似乎很有趣:

foreach($Application in $_.Package.Applications.Application)

这似乎是在一个包中枚举应用程序。

同一 PowerShell 代码中的注释说:

# The full AppUserModelId is composed of the package name, 
the publisher id, and the app id, such as
Microsoft.ZuneMusic_8wekyb3d8bbwe!Microsoft.ZuneMusic

所以缺少的是 $Application.Id

如果我能以任何方式获得 IAppInfo 接口(interface),我就可以调用 IAppInfo->get_Id() 我就准备好了。

但我不知道如何从 C++ 中的 IPackage 中获取它。

最佳答案

令人难以置信的是,没有人有想法! 这表明微软是如何让我们生活艰难的。 像枚举已安装的 StoreApps 及其 AppUserModelId 这样的通用任务需要一个科学研究部门。

我终于找到了一个在 Windows 8 和 Windows 10 上完美运行的解决方案。但是需要大量代码。

Windows 似乎没有在内存中保存应用程序 ID,也没有 API 可以直接确定它们。我研究了 Windows 10 SDK 中的所有头文件,但找不到可用于该任务的相应接口(interface)。

但我找到了获取它们的方法。我在问题的 6 个步骤之后继续:

  1. 调用 IPackage->get_InstalledLocation() 返回一个 IStorageFolder
  2. IStorageItem 的查询接口(interface)
  3. 调用IStorageItem->get_Path()

现在您有了安装应用程序的路径。 Windows 10 使用两个基本文件夹:

  • C:\Program Files\WindowsApps
  • C:\Windows\SystemApps

还有其他几个类似的

  • C:\Windows\vpnplugins
  • C:\Windows\devicesflow
  • C:\Windows\MicracastView
  • C:\Windows\PrintDialog
  • C:\Windows\PrintDialog3D
  • C:\Windows\WinStore

在返回的文件夹路径中,您将找到一个文件“AppxManifest.xml”。 这个文件看起来像:

<?xml version="1.0" encoding="utf-8"?>
<Package xmlns=".....">
    ......
    ......
    <Applications>
        <Application Id="microsoft.windowslive.mail" Executable="HxMail.exe" EntryPoint="Executable">
        ......
        ......
        </Application>
        <Application Id="microsoft.windowslive.calendar" Executable="HxCalendarAppImm.exe" EntryPoint="Executable">
        ......
        ......
        </Application>
    </Applications>
</Package>

瞧,它们就在那里。此程序包有两个应用程序 ID:“microsoft.windowslive.mail”和“microsoft.windowslive.calendar”。

然后您从第 6 步中获取包的 FamilyName 附加一个“!”并附加此 ID,您就完成了。

此包可以使用 AppUserModelId 之一通过 IApplicationActivationManager->ActivateApplication() 启动:

  • microsoft.windowscommunicationsapps_8wekyb3d8bbwe!microsoft.windowslive.calendar
  • microsoft.windowscommunicationsapps_8wekyb3d8bbwe!microsoft.windowslive.mail

关于c++ - 如何在 Windows 8 和 10 中枚举已安装的 StoreApps 及其 ID,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32150759/

有关c++ - 如何在 Windows 8 和 10 中枚举已安装的 StoreApps 及其 ID的更多相关文章

  1. ruby - 如何在 Ruby 中顺序创建 PI - 2

    出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits

  2. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  3. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  4. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  5. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  6. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  7. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  8. ruby - 如何在 Ruby 中拆分参数字符串 Bash 样式? - 2

    我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"

  9. ruby - 如何在 Lion 上安装 Xcode 4.6,需要用 RVM 升级 ruby - 2

    我实际上是在尝试使用RVM在我的OSX10.7.5上更新ruby,并在输入以下命令后:rvminstallruby我得到了以下回复:Searchingforbinaryrubies,thismighttakesometime.Checkingrequirementsforosx.Installingrequirementsforosx.Updatingsystem.......Errorrunning'requirements_osx_brew_update_systemruby-2.0.0-p247',pleaseread/Users/username/.rvm/log/138121

  10. ruby-on-rails - 如何在 ruby​​ 交互式 shell 中有多行? - 2

    这可能是个愚蠢的问题。但是,我是一个新手......你怎么能在交互式ruby​​shell中有多行代码?好像你只能有一条长线。按回车键运行代码。无论如何我可以在不运行代码的情况下跳到下一行吗?再次抱歉,如果这是一个愚蠢的问题。谢谢。 最佳答案 这是一个例子:2.1.2:053>a=1=>12.1.2:054>b=2=>22.1.2:055>a+b=>32.1.2:056>ifa>b#Thecode‘if..."startsthedefinitionoftheconditionalstatement.2.1.2:057?>puts"f

随机推荐