菜单项在macOS菜单栏App中禁用

时间:2016-12-31 12:41:05

标签: swift macos swift3

我正在尝试在macOS上构建一个菜单栏应用程序。

我似乎无法弄清楚为什么某些菜单项被禁用...截图:

Screenshot of the app

如您所见,退出菜单项已启用,并在单击时退出应用。但是,首选项项已被禁用。

我的代码

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和相同的结构,所以我对这种不一致感到有些困惑。这种情况发生的原因是什么?如何解决这个问题?

1 个答案:

答案 0 :(得分:3)

您的退出菜单项无法正常工作"。它没有使用您实现的terminate(_:)方法。在那里加上print()声明,您就会看到它没有被调用。

菜单项要么指定了特定的目标对象,要么使用响应程序链来搜索适当的目标。您没有为菜单项指定目标,因此他们正在使用响应程序链。您的MenuBarActions 不属于响应者链。 (类通常不能。某些对象可以。)因此,菜单项永远不会以您的课程为目标。

退出菜单有效,因为应用程序对象位于响应程序链上,并且它具有terminate(_:)方法。事实上,如果被调用,那么terminate(_:)方法会调用它。但是菜单项实际上是直接调用它。

您应该创建一个实际控制器对象(不仅仅是类)来实现操作方法并设置菜单项' target属性。