当前位置: 首页 > news >正文

Electron 如何调用 Windows 原生 API

做 Electron 桌面应用的时候,难免要和操作系统打打交道。在 Windows 上,这些需求说起来也不少:

  • 调用 Windows Store API 搞应用内购买
  • 处理 Windows Store 应用特有的文件系统虚拟化
  • 获取系统级别的权限和资源
  • 和 Windows Runtime (WinRT) 组件交互

Electron 说到底还是 Node.js 环境,而 Node.js 本来就不直接提供访问 Windows 原生 API 的能力。两者之间,需要一座桥。

这就像你想和不懂中文的朋友交流,中间总得有个翻译官。Electron 是用 JavaScript 写的,Windows API 是 C/C++ 写的,语言不通,得想办法搭个桥。代码世界的残酷就在这里,没什么人情的。

关于 HagiCode

本文分享的方案来自我们在 HagiCode 项目中的实践经验。HagiCode Desktop 需要调用 Microsoft Store API 来处理订阅购买和许可证管理,这便是我们摸索出一套技术方案的原因。毕竟有需求才有动力,这话一点不假。

技术方案对比

在 Electron 中调用 Windows 原生 API,有几种主流方案可以选择。每种方案都有其适用场景,就像工具箱里的不同工具,用对了地方才能发挥最大作用,用错了也只是徒增麻烦。

方案适用场景优点缺点
dynwinrtWinRT API (如 Store API)类型安全、自动生成绑定、现代 JavaScript 支持只支持 WinRT API、需要 Windows SDK
原生 Node.js 扩展高性能、任何 Windows API完全控制、性能最优需要 C++ 开发能力、跨平台复杂
child_process + PowerShell临时性、一次性调用简单快捷、无需编译性能差、错误处理复杂
edge.js/ffi-napi调用现有 DLL可复用现有库兼容性问题、维护成本高

HagiCode Desktop 采用了混合方案:使用 dynwinrt 来访问 Windows Store API,使用原生 Node.js 扩展来处理高性能的 Store 购买操作,同时用 Node.js 原生 fs 和 path 模块处理 Windows Store 应用特有的文件系统虚拟化。能简单就简单,这也是我们的原则。

方案一:使用 dynwinrt 调用 WinRT API

dynwinrt 是 Microsoft 提供的一个工具链,可以基于 Windows SDK 的 metadata 文件自动生成 JavaScript 绑定。它专门用于调用 WinRT API,比如 Windows Store API。

安装依赖:

{
"optionalDependencies": {
"@microsoft/dynwinrt": "0.1.0-preview.6",
"@microsoft/dynwinrt-codegen": "0.1.0-preview.6"
}
}

生成 WinRT 绑定:

// scripts/generate-store-bindings.js
const { execFileSync } = 'node:child_process';
function generateStoreNamespace(windowsWinmdPath) {
execFileSync('npx', [
'dynwinrt-codegen',
'generate',
'--winmd', windowsWinmdPath,
'--namespace', 'Windows.Services.Store',
'--output', 'src/main/subscription/generated-js',
'--lang', 'js',
]);
}

使用生成的绑定:

// 使用 dynwinrt 生成的 Store API 绑定
import { Windows } from '../subscription/generated-js/index.js';
async function queryStoreProduct(storeId: string) {
const storeContext = Windows.Services.Store.StoreContext.getDefault();
const result = await storeContext.getAssociatedStoreProductsAsync(['Subscription', 'Durable']);
if (result.extendedError !== 0) {
throw new Error(`Store API error: ${result.extendedError}`);
}
return result.products.get(storeId);
}

dynwinrt 的好处是类型安全,生成的代码和现代 JavaScript 习惯一致。但它只能处理 WinRT API,如果你需要调用传统的 Win32 API,就得用别的方案了。工具就是这样,各有所长。

方案二:原生 Node.js 扩展

当需要高性能或者 dynwinrt 不支持的功能时,原生 Node.js 扩展是最佳选择。这个方案需要用 C++ 写代码,然后用 node-gyp 编译成 .node 文件。

创建 binding.gyp:

{
"targets": [{
"target_name": "windows-store-addon",
"sources": ["src/windows-store-addon.cpp"],
"include_dirs": [
"<!(node -e \"require('nan')\")"
],
"defines": [
"WIN32_LEAN_AND_MEAN"
]
}]
}

C++ 原生模块示例:

// src/windows-store-addon.cpp
#include <nan.h>
#include <windows.h>
#include <wrl.h>
#include <windows.services.store.h>
using namespace v8;
using namespace Windows::Services::Store;
NAN_METHOD(QueryStoreStatus) {
auto async = new Nan::AsyncWorker(
[]() {
// 调用 Windows Store API
auto context = StoreContext::GetDefault();
auto products = context->GetAssociatedStoreProductsAsync(...)->GetResults();
// 处理结果
}
);
Nan::AsyncQueueWorker(async);
}
NAN_MODULE_INIT(InitModule) {
Nan::Set(target, Nan::New("queryStoreStatus").ToLocalChecked(),
Nan::GetFunction(Nan::New<FunctionTemplate>(QueryStoreStatus)).ToLocalChecked());
}
NODE_MODULE(windows_store_addon, InitModule)

编译和使用:

node-gyp rebuild
import addon from './build/Release/windows-store-addon.node';
const result = addon.queryStoreStatus({
storeId: 'your-store-id',
productKinds: ['Subscription', 'Durable']
});

原生扩展的性能是最好的,但开发成本也高。需要懂 C++,还要处理跨平台兼容问题。如果你的团队有 C++ 经验,或者性能要求特别高,这个方案值得投入。只是这条路走起来,终究是辛苦一些。

方案三:处理 Windows Store 应用虚拟化

Windows Store 应用运行在虚拟化环境中,路径映射需要特殊处理。HagiCode Desktop 用下面的函数来处理这个问题:

// src/main/windows-store-path-display.ts
export function resolveWindowsStorePackageFamilyName(executablePath: string): string | null {
const WINDOWS_APPS_SEGMENT = '\\windowsapps\\';
const windowsPath = executablePath.replace(/\//g, '\\');
const markerIndex = windowsPath.toLowerCase().indexOf(WINDOWS_APPS_SEGMENT);
if (markerIndex < 0) return null;
const relativePath = windowsPath.slice(markerIndex + WINDOWS_APPS_SEGMENT.length);
const packageFullName = relativePath.split('\\', 1)[0]?.trim();
return packageFullName || null;
}
export function resolveWindowsStoreVirtualizedPhysicalPath(
logicalPath: string,
options: ResolveWindowsStorePathDisplayOptions = {}
): string | null {
const packageFamilyName = options.packageFamilyName
?? resolveWindowsStorePackageFamilyName(options.execPath ?? process.execPath);
if (!packageFamilyName) return null;
const packageStorageRoot = path.win32.join(
options.env.LOCALAPPDATA,
'Packages',
packageFamilyName
);
// 将虚拟化路径映射到物理路径
if (isPathWithinWindowsRoot(logicalPath, options.env.APPDATA)) {
return path.win32.join(
packageStorageRoot,
'LocalCache',
http://www.jsqmd.com/news/1091841/

相关文章:

  • Go 高性能网络服务:从 TCP 参数调优到连接池工程实践
  • 深入解析TSB41BA3D PHY-LLC状态传输机制:实时事件通知与串行总线协同设计
  • QEMU安全配置:虚拟机隔离、权限控制与安全最佳实践
  • 豆包LaTeX公式转Word全攻略:AI导出鸭助你一键搞定
  • 从IO 500双登顶出发,中国存储领跑AI新周期
  • 【共创季稿事节】鸿蒙 ArkTS 安全区布局完全指南:SafeArea、expandSafeArea 与 Web 适配实战
  • 02 如何解决粘包问题
  • Metasploit实战入门:从Auxiliary侦察到Meterpreter后渗透完整指南
  • 【机器学习300问】早停法(Early Stopping):从损失曲线到实战调参的防过拟合指南
  • 联想小新休眠黑屏无法唤醒?聊聊低温锡 CPU 虚焊故障现象
  • 2026年银行全员营销新变局:当任务完成率统计成为“硬指标”,哪套系统真正能落地?
  • TI TPIC7710评估板实战指南:从硬件解析到软件调试的汽车电机控制验证
  • 2026年排盘精准度与底层逻辑:哪家八字排盘app排盘最标准、操作简单、功能齐全且能保存命盘
  • AI视频生成神器Pixelle-Video:3分钟让普通人变身视频创作高手
  • 地产三维动画制作公司怎么选:从技术路线到交付保障的完整决策框架
  • 3步掌握CDS API:解锁全球气象数据的Python神器
  • Windows本地训练LoRA模型完全指南:从环境配置到效果调优
  • Pytest测试用例精准执行:从命令行筛选到CI/CD集成的完整指南
  • NoFences:终极Windows桌面分区工具,3分钟打造整洁高效工作空间
  • 如何在Windows、macOS和Linux上免费畅玩Switch游戏:Ryujinx模拟器完全指南
  • Cloud Agent 开发笔记(2):Agent 引擎与 Tool 体系
  • 从“想做一个 Craft”到 ArkBlocks:一次 AI 协作开发原生 Block Editor 的心路历程
  • 计算机毕业设计之电影购票推荐网站的设计与实现
  • 深入解析MSPM0 UNICOMM-I2C模块:从协议原理到驱动实战
  • 批量白底图工具:多水印功能详解
  • 第5章-与HTTP协作的Web服务器
  • 斗地主AI实战指南:3步掌握DouZero智能辅助系统
  • 【入门】一文搞懂 Flume+Kafka+ZooKeeper:概念关系与 CentOS 7 完整部署指南
  • 手把手教你:如何向NCBI GEO高效提交高通量测序数据
  • 做汽车部件、芯片、新能源、新材料的研发人,是不是有这种感觉:通用PLM用起来各种别扭[特殊字符]