如何在远离系统栏的位置打开NSPopover?

时间:2018-02-03 05:21:06

标签: swift macos cocoa nspopover nsstatusbar

我在状态栏中以图标的操作打开NSPopover

myPopover.show(relativeTo: button.bounds, of: button, preferredEdge: NSRectEdge.minY)

除了与弹出窗口和系统栏的距离为零之外,这种方式正常:

enter image description here

我希望与Dropbox应用程序获得相同的结果,该应用程序将弹出窗口移动到离系统栏很小的距离:

enter image description here

我尝试使用不会影响popover位置的button.bounds.offsetBy(dx: 0.0, dy: 20.0)以及将popover置于系统栏上方的button.bounds.offsetBy(dx: 0.0, dy: -20.0)

enter image description here

那么如何将NSPopover定位在离系统栏一定距离的位置?

1 个答案:

答案 0 :(得分:4)

首先,button.bounds.offsetBy(dx: 0.0, dy: -20.0)不起作用的原因是因为那些坐标落在状态栏本身的状态栏项目的“窗口”之外。因此,除此之外的任何东西都被裁掉了。

我通过在这里和那里收集信息解决了这个问题:

  1. 创建一个不可见的窗口。
  2. 在状态栏项目的屏幕中找到坐标,并将不可见窗口置于其下。
  3. 显示与隐藏窗口相关的NSPopover,而不是状态栏项目。
  4. enter image description here

    红色的东西是不可见的窗口(用于演示目的)。

    Swift 4(Xcode 9.2)

    // Create a window
    let invisibleWindow = NSWindow(contentRect: NSMakeRect(0, 0, 20, 5), styleMask: .borderless, backing: .buffered, defer: false)
    invisibleWindow.backgroundColor = .red
    invisibleWindow.alphaValue = 0
    
    if let button = statusBarItem.button {
        // find the coordinates of the statusBarItem in screen space
        let buttonRect:NSRect = button.convert(button.bounds, to: nil)
        let screenRect:NSRect = button.window!.convertToScreen(buttonRect)
    
        // calculate the bottom center position (10 is the half of the window width)
        let posX = screenRect.origin.x + (screenRect.width / 2) - 10
        let posY = screenRect.origin.y
    
        // position and show the window
        invisibleWindow.setFrameOrigin(NSPoint(x: posX, y: posY))
        invisibleWindow.makeKeyAndOrderFront(self)
    
        // position and show the NSPopover
        mainPopover.show(relativeTo: invisibleWindow.contentView!.frame, of: invisibleWindow.contentView!, preferredEdge: NSRectEdge.minY)
        NSApp.activate(ignoringOtherApps: true)
    }
    

    我尝试使用show(relativeTo: invisibleWindow.frame ...)并且弹出窗口没有显示,因为NSWindow不是NSView。要显示弹出窗口,必须传递视图。