blob: 503eb270a7d1ef0975d71516611268940882e562 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
|
// Copyright (C) 2017 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
#include "qquickshortcutcontext_p_p.h"
#include "qquickoverlay_p_p.h"
#include "qquicktooltip_p.h"
#include <QtQmlModels/private/qtqmlmodels-config_p.h>
#if QT_CONFIG(qml_object_model)
#include "qquickmenu_p.h"
#include "qquickmenu_p_p.h"
#endif
#include "qquickpopup_p.h"
#include <QtCore/qloggingcategory.h>
#include <QtGui/qguiapplication.h>
#include <QtQuick/qquickrendercontrol.h>
QT_BEGIN_NAMESPACE
Q_LOGGING_CATEGORY(lcContextMatcher, "qt.quick.controls.shortcutcontext.matcher")
static bool isBlockedByPopup(QQuickItem *item)
{
if (!item || !item->window())
return false;
QQuickOverlay *overlay = QQuickOverlay::overlay(item->window());
const auto popups = QQuickOverlayPrivate::get(overlay)->stackingOrderPopups();
for (QQuickPopup *popup : popups) {
if (qobject_cast<QQuickToolTip *>(popup))
continue; // ignore tooltips (QTBUG-60492)
if (popup->isModal() || popup->closePolicy() & QQuickPopup::CloseOnEscape) {
qCDebug(lcContextMatcher) << popup << "is modal or has a CloseOnEscape policy;"
<< "if the following are both true," << item << "will be blocked by it:"
<< (item != popup->popupItem()) << !popup->popupItem()->isAncestorOf(item);
return item != popup->popupItem() && !popup->popupItem()->isAncestorOf(item);
}
}
return false;
}
bool QQuickShortcutContext::matcher(QObject *obj, Qt::ShortcutContext context)
{
QQuickItem *item = nullptr;
switch (context) {
case Qt::ApplicationShortcut:
return true;
case Qt::WindowShortcut:
while (obj && !obj->isWindowType()) {
item = qobject_cast<QQuickItem *>(obj);
if (item && item->window()) {
obj = item->window();
break;
} else if (QQuickPopup *popup = qobject_cast<QQuickPopup *>(obj)) {
obj = popup->window();
item = popup->popupItem();
#if QT_CONFIG(qml_object_model)
if (!obj) {
// The popup has no associated window (yet). However, sub-menus,
// unlike top-level menus, will not have an associated window
// until their parent menu is opened. So, check if this is a sub-menu
// so that actions within it can grab shortcuts.
if (auto *menu = qobject_cast<QQuickMenu *>(popup)) {
auto parentMenu = QQuickMenuPrivate::get(menu)->parentMenu;
while (!obj && parentMenu)
obj = parentMenu->window();
}
}
#endif
break;
}
obj = obj->parent();
}
if (QWindow *renderWindow = QQuickRenderControl::renderWindowFor(qobject_cast<QQuickWindow *>(obj)))
obj = renderWindow;
qCDebug(lcContextMatcher) << "obj" << obj << "item" << item << "focusWindow" << QGuiApplication::focusWindow()
<< "!isBlockedByPopup(item)" << !isBlockedByPopup(item);
return obj && qobject_cast<QWindow*>(obj)->isActive() && !isBlockedByPopup(item);
default:
return false;
}
}
QT_END_NAMESPACE
|