我正在尝试在macOS上构建一个菜单栏应用程序。
我似乎无法弄清楚为什么某些菜单项被禁用...截图:
如您所见,退出菜单项已启用,并在单击时退出应用。但是,首选项项已被禁用。
AppDelegate.swift:
let menuBarItem = NSStatusBar.system().statusItem(withLength: NSSquareStatusItemLength)
func applicationDidFinishLaunching(_ aNotification: Notification) {
menuBarItem.button?.image = NSImage(named: "MenuBarIcon")
menuBarItem.menu = MenuBarMenu()
}
MenuBarMenu.swift:
class MenuBarMenu: NSMenu {
init() {
super.init(title: "Menu")
self.addItem(withTitle: "Preferences...", action: #selector(MenuBarActions.openPreferencesWindow(_:)), keyEquivalent: "")
self.addItem(NSMenuItem.separator())
self.addItem(withTitle: "Quit", action: #selector(MenuBarActions.terminate(_:)), keyEquivalent: "")
}
required init(coder decoder: NSCoder) {
fatalError("init(coder:) has not been impemented")
}
}
class MenuBarActions {
@objc static func terminate(_ sender: NSMenuItem) {
NSApp.terminate(sender)
}
@objc static func openPreferencesWindow(_ sender: NSMenuItem) {
print("preferences")
}
}
我使用完全相同的方式为选择器创建MenuBarItems和相同的结构,所以我对这种不一致感到有些困惑。这种情况发生的原因是什么?如何解决这个问题?
答案 0 :(得分:3)
您的退出菜单项无法正常工作"。它没有使用您实现的terminate(_:)
方法。在那里加上print()
声明,您就会看到它没有被调用。
菜单项要么指定了特定的目标对象,要么使用响应程序链来搜索适当的目标。您没有为菜单项指定目标,因此他们正在使用响应程序链。您的MenuBarActions
类不属于响应者链。 (类通常不能。某些对象可以。)因此,菜单项永远不会以您的课程为目标。
退出菜单有效,因为应用程序对象位于响应程序链上,并且它具有terminate(_:)
方法。事实上,如果被调用,那么terminate(_:)
方法会调用它。但是菜单项实际上是直接调用它。
您应该创建一个实际控制器对象(不仅仅是类)来实现操作方法并设置菜单项' target
属性。