macOS 和 Windows 原生 API 的 Node.js 封装,使用 Swift + Win32 API + Node-API (N-API) 实现。
- 剪贴板变动监控 - 实时监听剪贴板内容变化
- 窗口激活监控 - 实时监听窗口切换事件
- 获取当前窗口 - 获取当前激活窗口的应用名和标识符
- 设置激活窗口 - 根据标识符激活指定应用
- 键盘模拟 - 模拟键盘按键和快捷键(支持修饰键)
- 粘贴模拟 - 模拟 Cmd+V (macOS) / Ctrl+V (Windows)
- 区域截图 - 选区截图并自动保存到剪贴板(Windows)
- macOS 10.15+
- Node.js 16.0+
- Swift 5.0+
- Xcode Command Line Tools
- Windows 10+
- Node.js 16.0+
- Visual Studio Build Tools 或 Visual Studio 2019+
npm install
npm run buildconst { ClipboardMonitor, WindowMonitor, WindowManager } = require('ztools-native-api');
// 1. 剪贴板变动监控(跨平台一致)
const clipboardMonitor = new ClipboardMonitor();
clipboardMonitor.start(() => {
console.log('剪贴板变化了!');
});
// 停止监控
clipboardMonitor.stop();
// 2. 窗口激活监控(实时监听窗口切换)
const windowMonitor = new WindowMonitor();
windowMonitor.start((windowInfo) => {
console.log('窗口切换:', windowInfo);
// macOS => { appName: 'Safari', bundleId: 'com.apple.Safari' }
// Windows => { appName: 'chrome', processId: 12345 }
});
// 停止监控
windowMonitor.stop();
// 3. 获取当前激活窗口
const activeWindow = WindowManager.getActiveWindow();
console.log(activeWindow);
// macOS => { appName: '终端', bundleId: 'com.apple.Terminal' }
// Windows => { appName: 'chrome', processId: 12345 }
// 3. 激活指定窗口
// macOS: 使用 bundleId (string)
WindowManager.activateWindow('com.apple.Safari');
// Windows: 使用 processId (number)
WindowManager.activateWindow(12345);
// 5. 模拟键盘按键
// 基本用法:输入单个字母
WindowManager.simulateKeyboardTap('a');
// 使用修饰键:输入大写字母
WindowManager.simulateKeyboardTap('a', 'shift');
// 模拟快捷键:Cmd+C (macOS) / Ctrl+C (Windows)
const modifier = process.platform === 'darwin' ? 'meta' : 'ctrl';
WindowManager.simulateKeyboardTap('c', modifier);
// 使用多个修饰键:Cmd+Shift+S (macOS) / Ctrl+Shift+S (Windows)
WindowManager.simulateKeyboardTap('s', modifier, 'shift');
// 特殊键:Enter、Tab、方向键等
WindowManager.simulateKeyboardTap('return');
WindowManager.simulateKeyboardTap('tab');
WindowManager.simulateKeyboardTap('left');
// 6. 模拟粘贴操作
WindowManager.simulatePaste();
// 7. 区域截图(仅 Windows)
const { ScreenCapture } = require('ztools-native-api');
ScreenCapture.start((result) => {
if (result.success) {
console.log(`截图成功!尺寸: ${result.width} x ${result.height}`);
console.log('截图已保存到剪贴板,可按 Ctrl+V 粘贴');
} else {
console.log('截图已取消');
}
});
// 操作:拖拽选择区域后释放鼠标,或按 ESC 取消const { WindowManager } = require('ztools-native-api');
// 获取当前窗口
const current = WindowManager.getActiveWindow();
// 跨平台激活窗口
if (WindowManager.getPlatform() === 'darwin') {
// macOS
WindowManager.activateWindow('com.apple.Safari');
} else if (WindowManager.getPlatform() === 'win32') {
// Windows - 使用之前获取的 processId
WindowManager.activateWindow(current.processId);
}启动剪贴板监控
- 参数:
callback()- 剪贴板变化时的回调函数(无参数,只通知变化事件) - 跨平台: ✅ 一致
停止剪贴板监控
- 跨平台: ✅ 一致
只读属性,是否正在监控
- 跨平台: ✅ 一致
启动窗口激活监控
- 参数:
callback(windowInfo)- 窗口切换时的回调函数- macOS:
{appName: string, bundleId: string} - Windows:
{appName: string, processId: number}
- macOS:
- 跨平台: ✅ API一致,返回值字段不同
停止窗口监控
- 跨平台: ✅ 一致
只读属性,是否正在监控
- 跨平台: ✅ 一致
获取当前激活窗口
- 返回值:
- macOS:
{appName: string, bundleId: string} | null - Windows:
{appName: string, processId: number} | null
- macOS:
示例:
// macOS
{ appName: 'Safari', bundleId: 'com.apple.Safari' }
// Windows
{ appName: 'chrome', processId: 12345 }激活指定应用窗口
- 参数:
- macOS:
bundleId(string) - Bundle Identifier - Windows:
processId(number) - 进程 ID
- macOS:
- 返回:
boolean- 是否激活成功
示例:
// macOS
WindowManager.activateWindow('com.apple.Safari');
// Windows
WindowManager.activateWindow(12345);获取当前平台
- 返回:
'darwin' | 'win32'
模拟键盘按键
- 参数:
key(string) - 要按的键(如 'a', 'return', 'tab', 'left' 等)...modifiers(string[]) - 修饰键(可选),支持 'shift', 'ctrl', 'alt', 'meta'
- 返回:
boolean- 是否成功 - 跨平台: ✅ 一致('meta' 在 macOS 上是 Command,在 Windows 上是 Win 键)
支持的按键:
- 字母:
a-z - 数字:
0-9 - 功能键:
f1-f12 - 特殊键:
return/enter,tab,space,backspace,delete,escape/esc - 方向键:
left,right,up,down - 符号键:
-,=,[,],\,;,',,,.,/,`
支持的修饰键:
shift- Shift 键ctrl/control- Control 键alt- Alt 键(macOS 上是 Option)meta- Command 键(macOS)/ Windows 键(Windows)
示例:
// 输入字母
WindowManager.simulateKeyboardTap('a');
// 输入大写字母(Shift + A)
WindowManager.simulateKeyboardTap('a', 'shift');
// 复制(Cmd+C / Ctrl+C)
const mod = process.platform === 'darwin' ? 'meta' : 'ctrl';
WindowManager.simulateKeyboardTap('c', mod);
// 多个修饰键(Cmd+Shift+S)
WindowManager.simulateKeyboardTap('s', mod, 'shift');
// 特殊键
WindowManager.simulateKeyboardTap('return'); // Enter
WindowManager.simulateKeyboardTap('tab'); // Tab
WindowManager.simulateKeyboardTap('left'); // 左方向键注意事项:
- macOS: 需要授予"辅助功能"权限(首次调用时会提示)
- Windows: 无需特殊权限
- 建议在调用前确保目标窗口已激活
模拟粘贴操作(Cmd+V / Ctrl+V)
- 返回:
boolean- 是否成功 - 跨平台: ✅ 一致
示例:
WindowManager.simulatePaste();启动区域截图(仅 Windows)
- 参数:
callback(result)- 截图完成时的回调函数result.success(boolean) - 是否成功截图result.width(number) - 截图宽度(成功时)result.height(number) - 截图高度(成功时)
- 平台:
⚠️ 仅支持 Windows
功能说明:
- 调用后会创建全屏半透明黑色遮罩
- 鼠标变为十字光标
- 拖拽鼠标选择截图区域
- 释放鼠标后自动截图并保存到剪贴板
- 按 ESC 键可取消截图
示例:
ScreenCapture.start((result) => {
if (result.success) {
console.log(`截图成功!尺寸: ${result.width}x${result.height}`);
// 截图已在剪贴板中,可按 Ctrl+V 粘贴
} else {
console.log('截图已取消');
}
});npm test
# 或运行特定测试
node test/test-keyboard.js # 完整键盘测试
node test/test-keyboard-simple.js # 简单键盘测试| 特性 | macOS | Windows |
|---|---|---|
| 窗口标识符 | Bundle ID (稳定,如 com.apple.Safari) |
Process ID (动态变化,如 12345) |
| 激活限制 | 较宽松 | 严格(需要线程附加 hack) |
| 剪贴板监控 | 轮询 changeCount |
消息循环 + WM_CLIPBOARDUPDATE |
| 键盘模拟 | ✅ 需要辅助功能权限 | ✅ 无需特殊权限 |
| 区域截图 | ❌ 暂不支持 | ✅ 支持(分层窗口 + GDI) |
| 权限要求 | 辅助功能权限(键盘模拟) | 无特殊要求 |
- Bundle ID 是稳定的,应用重启后不变
- 推荐使用 Bundle ID 作为窗口标识
- 键盘模拟需要辅助功能权限:
- 系统偏好设置 → 隐私与安全性 → 辅助功能
- 将你的应用或终端添加到允许列表
- 首次调用会自动提示授权
- Process ID 每次启动都会变化,不适合持久化存储
- 激活窗口可能受到 Windows 安全限制
- 建议结合应用名称 (
appName) 进行窗口识别
MIT