1

我有一个必须一直运行的应用程序(如果用户同意这一点)。

当用户退出应用程序时,我将前台应用程序转换为 LSUIElement(应用程序只有一个菜单栏图标,停靠图标和菜单消失)。

我在菜单项中有一个选项可以正常工作并将 LSUIElement 转换为前台应用程序(我使用函数[NSApp setActivationPolicy:NSApplicationActivationPolicyRegular][NSApp activateIgnoringOtherApps:YES])。

当用户双击应用程序时出现我的问题。我[NSApp setActivationPolicy:NSApplicationActivationPolicyRegular]在委托方法中再次使用了applicationWillUnhide:(NSNotification *)notification,除了没有出现的菜单之外,一切都很好。如果我去另一个应用程序,然后我回来菜单就会出现。我尝试了不同的方法,但我找不到一个好的方法。

我想知道当用户双击应用程序时调用的委托方法,或者NSApplication在那一刻调用的函数是什么,因为我认为在函数中使用setActivationPolicy:applicationWillUnhide函数为时已晚。

4

2 回答 2

0

要将普通应用程序转换为 LSUIElement 我使用

ProcessSerialNumber psn = { 0, kCurrentProcess };
TransformProcessType(&psn, kProcessTransformToUIElementApplication);

并将其更改回前台:

ProcessSerialNumber psn = { 0, kCurrentProcess };
TransformProcessType(&psn, kProcessTransformToForegroundApplication);
于 2012-10-15T15:50:44.420 回答
0

这是答案。在找到这个问题之前,我已经完成了隐藏/显示。这个问题启发了我最终的答案。

以下代码的作用如下:

  1. 当应用程序启动时,应用程序显示在停靠栏中,并显示一个菜单栏项。
  2. 当用户单击菜单栏项时,应用程序会隐藏并从停靠栏中删除。
  3. 当用户再次单击时,应用程序会重新显示为停靠。
  4. 如果应用程序被隐藏并且用户通过双击或启动板再次打开该应用程序,该应用程序将再次显示在 Dock 中。
  5. 如果应用程序没有隐藏但被其他应用程序遮挡,则单击菜单栏项或重新启动它会将应用程序置于最前面。
  6. 当用户单击窗口上的关闭按钮时,应用程序会从 Dock 中删除。
  7. 当用户通过 cmd+q 或从文件菜单退出应用程序时,应用程序退出并且菜单栏项也退出。

我删除了其他不直接相关的代码。

您可能会注意到的其他事项:

  1. 对于我的 Info.plist,LSUIElement 未设置或设置为 NO。如果你想设置为是。您需要在情节提要中设置不启动视图控制器并自己从窗口控制器构造。
  2. 您还将通过鼠标左键单击菜单栏项来处理逻辑,因为您从一开始就没有窗口。

代码:

import Cocoa

@NSApplicationMain
class AppDelegate: NSObject, NSApplicationDelegate {
    private let statusItem = NSStatusBar.system.statusItem(withLength:NSStatusItem.squareLength)
    weak private var window:NSWindow? = nil

    func applicationDidFinishLaunching(_ aNotification: Notification) {        
        setupMenubarTray()

        self.window = NSApp.orderedWindows.first

        NotificationCenter.default.addObserver(self, selector: #selector(windowWillClose(_:)), name: NSWindow.willCloseNotification, object: self.window!)
    }

    func applicationShouldHandleReopen(_ sender: NSApplication, hasVisibleWindows flag: Bool) -> Bool {
        if !window!.isVisible {
            activeApp()
            return false
        }

        return true
    }
}

extension AppDelegate {
    @objc func windowWillClose(_ noti:Notification) {
        removeFromDock()
    }

    private func showInDock() {
        NSApp.setActivationPolicy(.regular)
    }

    private func removeFromDock() {
        NSApp.setActivationPolicy(.accessory)
    }
}

// MARK: - setup menubar button
extension AppDelegate {
    private func setupMenubarTray() {
        guard let button = statusItem.button else {
            fatalError()
        }

        setTrayIcon(for:button)
        button.action = #selector(mouseLeftButtonClicked)
    }

    private func setTrayIcon(for button:NSStatusBarButton) {
        let useMonochromeIcon = UserDefaults.standard.bool(forKey: DefaultsKey.useMonochromeIcon.key)
        button.image = NSImage(imageLiteralResourceName: useMonochromeIcon ? "MonochromeIcon" : "TrayIcon")
    }

    @objc private func mouseLeftButtonClicked() {
        if NSApp.isHidden || !window!.isKeyWindow {
            self.activeApp()
        } else {
            self.hide()
        }
    }

    private func activeApp() {
        showInDock()
        window?.makeKeyAndOrderFront(nil)
        NSApp.activate(ignoringOtherApps: true)

        checker.sendNotification()
    }

    private func hide() {
        removeFromDock()
        NSApp.hide(nil)
    }
}
于 2020-01-05T01:35:08.733 回答