blob: 5d3184a44da454b12918d3112f358cb847cb3db5 (
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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
|
// 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 <QtGui/private/qguiapplication_p.h>
#include <QtQuick/qquickrendercontrol.h>
#include <QtQuickTemplates2/private/qquickpopupwindow_p_p.h>
QT_BEGIN_NAMESPACE
Q_STATIC_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());
auto popups = QQuickOverlayPrivate::get(overlay)->stackingOrderPopups();
for (QWindow *popupWindow : QGuiApplicationPrivate::popup_list) {
if (QQuickPopupWindow *quickPopupWindow = qobject_cast<QQuickPopupWindow *>(popupWindow);
quickPopupWindow && quickPopupWindow->popup())
popups += quickPopupWindow->popup();
}
for (QQuickPopup *popup : std::as_const(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 one of the following is 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)
{
if ((context != Qt::ApplicationShortcut) && (context != Qt::WindowShortcut))
return false;
QQuickItem *item = nullptr;
// look for the window contains embedded shortcut
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 (parentMenu) {
obj = parentMenu->window();
if (obj)
break;
parentMenu = QQuickMenuPrivate::get(parentMenu)->parentMenu;
}
}
}
#endif
break;
}
obj = obj->parent();
}
if (context == Qt::ApplicationShortcut) {
// the application shortcuts inside hidden/closed windows should not match
return obj && qobject_cast<QWindow*>(obj)->isVisible();
} else {
Q_ASSERT(context == Qt::WindowShortcut);
QQuickWindow *window = qobject_cast<QQuickWindow *>(obj);
if (QWindow *renderWindow = QQuickRenderControl::renderWindowFor(window))
obj = renderWindow;
qCDebug(lcContextMatcher) << "obj" << obj << "item" << item << "focusWindow"
<< QGuiApplication::focusWindow()
<< "!isBlockedByPopup(item)" << !isBlockedByPopup(item);
return obj && qobject_cast<QWindow*>(obj)->isActive() && !isBlockedByPopup(item);
}
}
QT_END_NAMESPACE
|