Обработка смежных областей отслеживания как одной смежной области

Я пытаюсь представить интерфейс окна без заголовка, когда мышь покидает заголовок окна или вид содержимого, но не при переходе от одного к другому; в сущности, две области отслеживания функционируют как одна (я прибегнул к этому, поскольку не мог понять, как создать одну область, когда заголовок скрыт):

override func mouseEntered(with theEvent: NSEvent) {
    let hideTitle = (doc?.settings.autoHideTitle.value == true)
    if theEvent.modifierFlags.contains(.shift) {
        NSApp.activate(ignoringOtherApps: true)
    }

    switch theEvent.trackingNumber {
    case closeTrackingTag:
        Swift.print("close entered")
        closeButton?.image = closeButtonImage
        break

    default:
        Swift.print(String(format: "%@ entered",
                           (theEvent.trackingNumber == titleTrackingTag
                            ? "title" : "view")))
        let lastMouseOver = mouseOver
        mouseOver = true
        updateTranslucency()

        //  view or title entered
        if hideTitle && (lastMouseOver != mouseOver) {
            updateTitleBar(didChange: !lastMouseOver)
        }
    }
}

override func mouseExited(with theEvent: NSEvent) {
    let hideTitle = (doc?.settings.autoHideTitle.value == true)

    switch theEvent.trackingNumber {
    case closeTrackingTag:
        Swift.print("close exited")
        closeButton?.image = nullImage
        break

    default:
        Swift.print(String(format: "%@ exited",
                           (theEvent.trackingNumber == titleTrackingTag
                            ? "title" : "view")))

        let lastMouseOver = mouseOver
        mouseOver = false
        updateTranslucency()

        if hideTitle && (lastMouseOver != mouseOver) {
            updateTitleBar(didChange: lastMouseOver)
        }
    }
}

Кроме того, на кнопке закрытия есть прямоугольник отслеживания, который отображается только после его завершения. Во всяком случае, из моего вывода трассировщика я вижу проблему - наведите курсор мыши на окно снизу вверх над его заголовком:

view entered
updateTitleBar
**view entered**
view exited
updateTitleBar
title entered
updateTitleBar
title exited

Заметьте, конечно, почему я получаю событие второго представления, введенное (представление введено), но перемещение из вида на смежный заголовок каждый вызывает вызов updateTilebar(), который визуально заикается - пока что не исправлено анимацией:

fileprivate func docIconToggle() {
    let docIconButton = panel.standardWindowButton(.documentIconButton)

    if settings.autoHideTitle.value == false || mouseOver {
        if let doc = self.document {
            docIconButton?.image = (doc as! Document).displayImage
        }
        else
        {
            docIconButton?.image = NSApp.applicationIconImage
        }
        docIconButton?.isHidden = false
        self.synchronizeWindowTitleWithDocumentName()
    }
    else
    {
        docIconButton?.isHidden = true
    }
}

@objc func updateTitleBar(didChange: Bool) {
    if didChange {
        Swift.print("updateTitleBar")
        if settings.autoHideTitle.value == true && !mouseOver {
            NSAnimationContext.runAnimationGroup({ (context) -> Void in
                context.duration = 1.0
                panel.animator().titleVisibility = NSWindowTitleVisibility.hidden
                panel.animator().titlebarAppearsTransparent = true
                panel.animator().styleMask.formUnion(.fullSizeContentView)
            }, completionHandler: {
                self.docIconToggle()
            })
        } else {
            NSAnimationContext.runAnimationGroup({ (context) -> Void in
                context.duration = 1.0
                panel.animator().titleVisibility = NSWindowTitleVisibility.visible
                panel.animator().titlebarAppearsTransparent = false
                panel.animator().styleMask.formSymmetricDifference(.fullSizeContentView)
            }, completionHandler: {
                self.docIconToggle()
            })
        }
    }
}

Поэтому мой вопрос о том, как отложить действия, когда области являются смежными.

Они (заголовок заголовка и контент) не являются братьями и сестрами друг друга, поэтому я не думал, что hitTest() выполним, но, по сути, если бы я мог сказать, двигаюсь ли я в соседнюю область отслеживания, я бы хотел, чтобы соч.

Любая помощь с тем, почему анимация не работает, будет плюсом.

1 ответ

Не верный ответ, но если вы знаете прямоугольник соседнего вида, вы можете использовать местоположение события, чтобы проверить, хотите ли вы игнорировать движения между соседними видами:

override func mouseExited(with theEvent: NSEvent) {
    let hideTitle = (doc?.settings.autoHideTitle.value == true)
    let location : NSPoint = theEvent.locationInWindow

    switch theEvent.trackingNumber {
    case closeTrackingTag:
        Swift.print("close exited")
        closeButton?.image = nullImage
        break

    default:
        let vSize = self.window?.contentView?.bounds.size

        //  If we exit to the title bar area we're still "inside"
        //  and visa-versa, leaving title to content view.
        if theEvent.trackingNumber == titleTrackingTag, let tSize = titleView?.bounds.size {
            if location.x >= 0.0 && location.x <= (vSize?.width)! && location.y < ((vSize?.height)! + tSize.height) {
                Swift.print("title -> view")
                return
            }
        }
        else
        if theEvent.trackingNumber == viewTrackingTag {
            if location.x >= 0.0 && location.x <= (vSize?.width)! && location.y > (vSize?.height)! {
                Swift.print("view -> title")
                return
            }
        }

        mouseOver = false
        updateTranslucency()

        if hideTitle {
            updateTitleBar(didChange: true)
        }

        Swift.print(String(format: "%@ exited",
                           (theEvent.trackingNumber == titleTrackingTag
                            ? "title" : "view")))
    }
}
Другие вопросы по тегам