macOS 应用程序 : handling key combinations bound to global keyboard shortcuts

标签 macos cocoa keyboard-shortcuts keyboard-events nsevent

在某些应用程序中,应用程序直接处理键盘快捷键是有意义的,否则这些快捷键将绑定(bind)到系统范围的组合。例如,⌘-Space(通常是 Spotlight)或 ⌘-Tab(通常是应用程序切换器)。这适用于各种 Mac 应用程序,例如 VMWare Fusion、Apple 自己的屏幕共享和远程桌面客户端(分别将事件转发到 VM 或服务器,而不是在本地处理它们),以及应用程序中的一些类似的第三方应用程序商店。

我们希望在我们正在开发的应用程序中实现这种模式,但很难弄清楚如何实现。我应该指出,有问题的应用程序是常规的前台应用程序,是沙盒的,任何解决方案都必须遵守应用程序商店规则。商店中的其他应用程序可以做到这一点这一事实意味着这一定是可能的。

需要明确的是,我们希望:

  • 检测并处理所有按键,包括绑定(bind)到全局快捷键的按键。
  • 防止全局快捷方式触发其全局绑定(bind)效果。

苹果的Event Architecture document表明前台应用程序应该已经接收到这些事件。 (它只讨论早期级别处理诸如电源和弹出按钮之类的事情,这很好。)它继续建议,并且 key events document also implies NSApplicationsendEvent: 方法根据修饰符标志检测潜在的快捷方式,将它们分派(dispatch)到窗口,如果失败,则分派(dispatch)到菜单栏。没有明确说明全局绑定(bind)的快捷方式会发生什么。

我尝试子类化 NSApplication 并覆盖 sendEvent:。无论我是否将所有事件传递给父类(super class)实现,或者如果我说过滤修饰键事件,当我按 ⌘-Space 时,我都会收到按下和释放命令 (⌘) 键的事件,但不会收到空格键的事件。 Spotlight UI 始终会弹出。

我还没有从 Apple 或其他地方找到太多关于 NSApplication 子类及其早期事件处理的信息。我似乎无法找出在什么级别检测和处理全局快捷方式。

有人可以给我指出正确的方向吗?

可能无效的解决方案:

我在其他 Stack Overflow 帖子中看到的建议,但不适用于我见过的执行此操作(并且会违反 App Store 规则)的其他应用程序:

  • Accessibilty API(需要特殊许可)
  • 事件点击/ Hook (需要以 root 身份运行)

无论如何,这两种方法都太过分了,因为它们可以让您在所有时间拦截所有事件,而不仅仅是当您的应用程序是前台应用程序时。

NSeventaddGlobalMonitorForEventsMatchingMask:handler: 同时不会阻止全局快捷方式处理程序为这些事件触发,所以我什至懒得尝试它。

最佳答案

好吧,Cocoa 事件方法和 Quartz 事件点击已经过时了,因为它们要么需要 root 访问权限,要么需要辅助访问权限,或者不会在 Dock 之前捕获事件。

Carbon 的 PushSymbolicHotKeyMode 已退出,因为根据文档,它需要辅助访问权限。

Carbon 的 RegisterEventHotKey 可能已经退出,因为苹果似乎不允许它(请参阅我对问题的评论中的链接)。然而,即便如此,我测试过,你不能用它来捕获 Command+Tab。

我对它的工作原理进行了快速概念验证,但是 YMMV:

  • 实现此 answer 中的 KeyboardWatcher 示例类。您需要链接 IOKit。
  • 添加硬件 - USB (com.apple.security.device.usb) 沙箱权利。这是必要的,因为 KeyboardWatcher 使用 HID 来捕获按键
  • Handle_DeviceEventCallback 将为您提供按下的按键。显然,您可以根据自己的需要进行修改
  • 使用SetSystemUIMode 阻止任务切换器和 Spotlight。您需要链接 Carbon。

SetSystemUIMode(kUIModeContentSuppressed, kUIOptionDisableProcessSwitch);

请注意,这仅在您的应用程序位于前台时有效(可能是您想要的)。我使用跟踪矩形在我的 View 上设置了此设置,因此只有当鼠标位于我的 View 上时它才会生效(就像在 Remotix 中一样):

- (void)viewDidLoad {
[super viewDidLoad];

NSTrackingArea* trackingArea = [[NSTrackingArea alloc] initWithRect:[self.view bounds] options: (NSTrackingMouseEnteredAndExited | 

NSTrackingActiveAlways) owner:self userInfo:nil];
    [self.view addTrackingArea:trackingArea];
}

- (void) mouseEntered:(NSEvent*)theEvent {
    SetSystemUIMode(kUIModeContentSuppressed, kUIOptionDisableProcessSwitch);
}

- (void) mouseExited:(NSEvent*)theEvent {
    SetSystemUIMode(kUIModeNormal, 0);
}

Remotix 似乎链接了 Carbon 和 IOKit,但我看不到它们是否有 USB 授权(我尝试了演示,而不是 App Store 版本)。他们有可能正在做这样的事情。

<小时/>

实现此目的的正常方法是安装 Quartz 事件分接头。但是,要接收针对其他应用程序的事件,您需要(如您所说)成为 root 用户,或者为您的应用程序启用辅助功能访问权限。

在当前的沙盒规则中似乎无法使用事件点击。这在 developer forum 中得到了证实。 。该链接仅供登录,但引用线程:

Is there are any chance to handle events that comming from media keys by prevents launch iTunes. Before sandbox it was possible by create CGEventTap but now sandbox deny using hid-controll.

No, this is not currently possible within App Sandbox.

我不确定还有其他方法可以做到这一点;我有兴趣知道 App Store 中的哪些应用程序可以?

VMWare Fusion 显然没有沙盒化,Apple 自己的应用程序不受这些规则的约束。请记住,沙盒仅对 2012 年推出后添加的新应用程序强制执行。该日期之前添加的应用程序不会强制执行沙盒。看这个answer .

关于macOS 应用程序 : handling key combinations bound to global keyboard shortcuts,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44472903/

相关文章:

iphone - 如何跟踪 AVAssetWriter 的写入进度?

objective-c - Mac 上使用 Objective-C 的图像处理资源

cocoa - HUD NSPanel 不太透明

c# - 如何将 Ctrl+,(控制加逗号)指定为 WPF 菜单项的键盘快捷键?

objective-c - 服务菜单启动错误的应用程序包

macos - Docker:OS X上不存在导出路径

objective-c - 带有 ARC 的 Interface Builder 中的单例

emacs - 在 EMACS org.mode 中覆盖 Ctrl-TAB

jQuery BlockUI : Is it possible to run a function after a user hits the "enter" key?

c - 在使用 lldb 调试 macOS 应用程序时,是否可以使用观察点观察寄存器的内容以进行更改?