我目前正在开发一个 Safari App Extension,它由两部分组成:
- 仅主机状态栏应用程序
- Safari 扩展
此外,主机应用程序提供了一个全局快捷方式,可在状态栏中打开一个弹出窗口。但是,我想检查哪个应用程序的窗口当前处于活动状态,因为如果 Safari 窗口当前处于活动状态,我不想打开弹出框。有什么方法可以使用 Swift 找出当前处于活动状态的应用程序窗口吗?
谢谢您的帮助。
我目前正在开发一个 Safari App Extension,它由两部分组成:
此外,主机应用程序提供了一个全局快捷方式,可在状态栏中打开一个弹出窗口。但是,我想检查哪个应用程序的窗口当前处于活动状态,因为如果 Safari 窗口当前处于活动状态,我不想打开弹出框。有什么方法可以使用 Swift 找出当前处于活动状态的应用程序窗口吗?
谢谢您的帮助。
因此,根据 Alexander 和 Wileke 的评论,我想我找到了解决方案。
NSWorkspace.shared.frontmostApplication
您可以检查Safari当前是否处于活动状态。但正如 Wilke 所说,这并不意味着它有一个活动窗口。因此,我们使用CGWindowListCopyWindowInfo
首先获取所有窗口并通过比较 PID 来检查其中是否至少有一个属于 Safari。
这样,我们可以肯定地说 Safari 当前必须有一个活动窗口来接收关键事件。这一定是真的,因为现在有一种方式是 Safari 在最前面,没有任何窗口,或者 Safari 作为一个窗口但同时不是最前面。
好吧,除非我错过了什么。但现在它有效。
这是我想出的代码:
func safariIsActive() -> Bool {
// Get the app that currently has the focus.
let frontApp = NSWorkspace.shared.frontmostApplication!
// Check if the front most app is Safari
if frontApp.bundleIdentifier == "com.apple.Safari" {
// If it is Safari, it still does not mean, that is receiving key events
// (i.e., has a window at the front).
// But what we can safely say is, that if Safari is the front most app
// and it has at least one window, it has to be the window that
// crrently receives key events.
let safariPID = frontApp.processIdentifier
// With this procedure, we get all available windows.
let options = CGWindowListOption(arrayLiteral: CGWindowListOption.excludeDesktopElements, CGWindowListOption.optionOnScreenOnly)
let windowListInfo = CGWindowListCopyWindowInfo(options, CGWindowID(0))
let windowInfoList = windowListInfo as NSArray? as? [[String: AnyObject]]
// Now that we have all available windows, we are going to check if at least one of them
// is owned by Safari.
for info in windowInfoList! {
let windowPID = info["kCGWindowOwnerPID"] as! UInt32
if windowPID == safariPID {
return true
}
}
}
return false
}