Absolute File Name: | /home/qt/qt5_coco/qt5/qtbase/src/plugins/platforms/xcb/qxcbdrag.cpp |
Source code | Switch to Preprocessed file |
Line | Source | Count | ||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
1 | /**************************************************************************** | - | ||||||||||||||||||
2 | ** | - | ||||||||||||||||||
3 | ** Copyright (C) 2016 The Qt Company Ltd. | - | ||||||||||||||||||
4 | ** Contact: https://www.qt.io/licensing/ | - | ||||||||||||||||||
5 | ** | - | ||||||||||||||||||
6 | ** This file is part of the QtGui module of the Qt Toolkit. | - | ||||||||||||||||||
7 | ** | - | ||||||||||||||||||
8 | ** $QT_BEGIN_LICENSE:LGPL$ | - | ||||||||||||||||||
9 | ** Commercial License Usage | - | ||||||||||||||||||
10 | ** Licensees holding valid commercial Qt licenses may use this file in | - | ||||||||||||||||||
11 | ** accordance with the commercial license agreement provided with the | - | ||||||||||||||||||
12 | ** Software or, alternatively, in accordance with the terms contained in | - | ||||||||||||||||||
13 | ** a written agreement between you and The Qt Company. For licensing terms | - | ||||||||||||||||||
14 | ** and conditions see https://www.qt.io/terms-conditions. For further | - | ||||||||||||||||||
15 | ** information use the contact form at https://www.qt.io/contact-us. | - | ||||||||||||||||||
16 | ** | - | ||||||||||||||||||
17 | ** GNU Lesser General Public License Usage | - | ||||||||||||||||||
18 | ** Alternatively, this file may be used under the terms of the GNU Lesser | - | ||||||||||||||||||
19 | ** General Public License version 3 as published by the Free Software | - | ||||||||||||||||||
20 | ** Foundation and appearing in the file LICENSE.LGPL3 included in the | - | ||||||||||||||||||
21 | ** packaging of this file. Please review the following information to | - | ||||||||||||||||||
22 | ** ensure the GNU Lesser General Public License version 3 requirements | - | ||||||||||||||||||
23 | ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. | - | ||||||||||||||||||
24 | ** | - | ||||||||||||||||||
25 | ** GNU General Public License Usage | - | ||||||||||||||||||
26 | ** Alternatively, this file may be used under the terms of the GNU | - | ||||||||||||||||||
27 | ** General Public License version 2.0 or (at your option) the GNU General | - | ||||||||||||||||||
28 | ** Public license version 3 or any later version approved by the KDE Free | - | ||||||||||||||||||
29 | ** Qt Foundation. The licenses are as published by the Free Software | - | ||||||||||||||||||
30 | ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 | - | ||||||||||||||||||
31 | ** included in the packaging of this file. Please review the following | - | ||||||||||||||||||
32 | ** information to ensure the GNU General Public License requirements will | - | ||||||||||||||||||
33 | ** be met: https://www.gnu.org/licenses/gpl-2.0.html and | - | ||||||||||||||||||
34 | ** https://www.gnu.org/licenses/gpl-3.0.html. | - | ||||||||||||||||||
35 | ** | - | ||||||||||||||||||
36 | ** $QT_END_LICENSE$ | - | ||||||||||||||||||
37 | ** | - | ||||||||||||||||||
38 | ****************************************************************************/ | - | ||||||||||||||||||
39 | - | |||||||||||||||||||
40 | #include "qxcbdrag.h" | - | ||||||||||||||||||
41 | #include <xcb/xcb.h> | - | ||||||||||||||||||
42 | #include "qxcbconnection.h" | - | ||||||||||||||||||
43 | #include "qxcbclipboard.h" | - | ||||||||||||||||||
44 | #include "qxcbmime.h" | - | ||||||||||||||||||
45 | #include "qxcbwindow.h" | - | ||||||||||||||||||
46 | #include "qxcbscreen.h" | - | ||||||||||||||||||
47 | #include "qwindow.h" | - | ||||||||||||||||||
48 | #include "qxcbcursor.h" | - | ||||||||||||||||||
49 | #include <private/qdnd_p.h> | - | ||||||||||||||||||
50 | #include <qdebug.h> | - | ||||||||||||||||||
51 | #include <qevent.h> | - | ||||||||||||||||||
52 | #include <qguiapplication.h> | - | ||||||||||||||||||
53 | #include <qrect.h> | - | ||||||||||||||||||
54 | #include <qpainter.h> | - | ||||||||||||||||||
55 | #include <qtimer.h> | - | ||||||||||||||||||
56 | - | |||||||||||||||||||
57 | #include <qpa/qwindowsysteminterface.h> | - | ||||||||||||||||||
58 | - | |||||||||||||||||||
59 | #include <private/qguiapplication_p.h> | - | ||||||||||||||||||
60 | #include <private/qshapedpixmapdndwindow_p.h> | - | ||||||||||||||||||
61 | #include <private/qsimpledrag_p.h> | - | ||||||||||||||||||
62 | #include <private/qhighdpiscaling_p.h> | - | ||||||||||||||||||
63 | - | |||||||||||||||||||
64 | QT_BEGIN_NAMESPACE | - | ||||||||||||||||||
65 | - | |||||||||||||||||||
66 | #ifndef QT_NO_DRAGANDDROP | - | ||||||||||||||||||
67 | - | |||||||||||||||||||
68 | //#define DND_DEBUG | - | ||||||||||||||||||
69 | #ifdef DND_DEBUG | - | ||||||||||||||||||
70 | #define DEBUG qDebug | - | ||||||||||||||||||
71 | #else | - | ||||||||||||||||||
72 | #define DEBUG if(0) qDebug | - | ||||||||||||||||||
73 | #endif | - | ||||||||||||||||||
74 | - | |||||||||||||||||||
75 | #ifdef DND_DEBUG | - | ||||||||||||||||||
76 | #define DNDDEBUG qDebug() | - | ||||||||||||||||||
77 | #else | - | ||||||||||||||||||
78 | #define DNDDEBUG if(0) qDebug() | - | ||||||||||||||||||
79 | #endif | - | ||||||||||||||||||
80 | - | |||||||||||||||||||
81 | const int xdnd_version = 5; | - | ||||||||||||||||||
82 | - | |||||||||||||||||||
83 | static inline xcb_window_t xcb_window(QPlatformWindow *w) | - | ||||||||||||||||||
84 | { | - | ||||||||||||||||||
85 | return static_cast<QXcbWindow *>(w)->xcb_window(); never executed: return static_cast<QXcbWindow *>(w)->xcb_window(); | 0 | ||||||||||||||||||
86 | } | - | ||||||||||||||||||
87 | - | |||||||||||||||||||
88 | static inline xcb_window_t xcb_window(QWindow *w) | - | ||||||||||||||||||
89 | { | - | ||||||||||||||||||
90 | return static_cast<QXcbWindow *>(w->handle())->xcb_window(); never executed: return static_cast<QXcbWindow *>(w->handle())->xcb_window(); | 0 | ||||||||||||||||||
91 | } | - | ||||||||||||||||||
92 | - | |||||||||||||||||||
93 | static xcb_window_t xdndProxy(QXcbConnection *c, xcb_window_t w) | - | ||||||||||||||||||
94 | { | - | ||||||||||||||||||
95 | xcb_window_t proxy = XCB_NONE; | - | ||||||||||||||||||
96 | - | |||||||||||||||||||
97 | xcb_get_property_cookie_t cookie = Q_XCB_CALL2(xcb_get_property(c->xcb_connection(), false, w, c->atom(QXcbAtom::XdndProxy), | - | ||||||||||||||||||
98 | XCB_ATOM_WINDOW, 0, 1), c); | - | ||||||||||||||||||
99 | xcb_get_property_reply_t *reply = xcb_get_property_reply(c->xcb_connection(), cookie, 0); | - | ||||||||||||||||||
100 | - | |||||||||||||||||||
101 | if (reply && reply->type == XCB_ATOM_WINDOW)
| 0 | ||||||||||||||||||
102 | proxy = *((xcb_window_t *)xcb_get_property_value(reply)); never executed: proxy = *((xcb_window_t *)xcb_get_property_value(reply)); | 0 | ||||||||||||||||||
103 | free(reply); | - | ||||||||||||||||||
104 | - | |||||||||||||||||||
105 | if (proxy == XCB_NONE)
| 0 | ||||||||||||||||||
106 | return proxy; never executed: return proxy; | 0 | ||||||||||||||||||
107 | - | |||||||||||||||||||
108 | // exists and is real? | - | ||||||||||||||||||
109 | cookie = Q_XCB_CALL2(xcb_get_property(c->xcb_connection(), false, proxy, c->atom(QXcbAtom::XdndProxy), | - | ||||||||||||||||||
110 | XCB_ATOM_WINDOW, 0, 1), c); | - | ||||||||||||||||||
111 | reply = xcb_get_property_reply(c->xcb_connection(), cookie, 0); | - | ||||||||||||||||||
112 | - | |||||||||||||||||||
113 | if (reply && reply->type == XCB_ATOM_WINDOW) {
| 0 | ||||||||||||||||||
114 | xcb_window_t p = *((xcb_window_t *)xcb_get_property_value(reply)); | - | ||||||||||||||||||
115 | if (proxy != p)
| 0 | ||||||||||||||||||
116 | proxy = 0; never executed: proxy = 0; | 0 | ||||||||||||||||||
117 | } else { never executed: end of block | 0 | ||||||||||||||||||
118 | proxy = 0; | - | ||||||||||||||||||
119 | } never executed: end of block | 0 | ||||||||||||||||||
120 | - | |||||||||||||||||||
121 | free(reply); | - | ||||||||||||||||||
122 | - | |||||||||||||||||||
123 | return proxy; never executed: return proxy; | 0 | ||||||||||||||||||
124 | } | - | ||||||||||||||||||
125 | - | |||||||||||||||||||
126 | class QXcbDropData : public QXcbMime | - | ||||||||||||||||||
127 | { | - | ||||||||||||||||||
128 | public: | - | ||||||||||||||||||
129 | QXcbDropData(QXcbDrag *d); | - | ||||||||||||||||||
130 | ~QXcbDropData(); | - | ||||||||||||||||||
131 | - | |||||||||||||||||||
132 | protected: | - | ||||||||||||||||||
133 | bool hasFormat_sys(const QString &mimeType) const Q_DECL_OVERRIDE; | - | ||||||||||||||||||
134 | QStringList formats_sys() const Q_DECL_OVERRIDE; | - | ||||||||||||||||||
135 | QVariant retrieveData_sys(const QString &mimeType, QVariant::Type type) const Q_DECL_OVERRIDE; | - | ||||||||||||||||||
136 | - | |||||||||||||||||||
137 | QVariant xdndObtainData(const QByteArray &format, QVariant::Type requestedType) const; | - | ||||||||||||||||||
138 | - | |||||||||||||||||||
139 | QXcbDrag *drag; | - | ||||||||||||||||||
140 | }; | - | ||||||||||||||||||
141 | - | |||||||||||||||||||
142 | - | |||||||||||||||||||
143 | QXcbDrag::QXcbDrag(QXcbConnection *c) : QXcbObject(c) | - | ||||||||||||||||||
144 | { | - | ||||||||||||||||||
145 | dropData = new QXcbDropData(this); | - | ||||||||||||||||||
146 | - | |||||||||||||||||||
147 | init(); | - | ||||||||||||||||||
148 | cleanup_timer = -1; | - | ||||||||||||||||||
149 | } executed 135 times by 5 tests: end of block Executed by:
| 135 | ||||||||||||||||||
150 | - | |||||||||||||||||||
151 | QXcbDrag::~QXcbDrag() | - | ||||||||||||||||||
152 | { | - | ||||||||||||||||||
153 | delete dropData; | - | ||||||||||||||||||
154 | } executed 347 times by 219 tests: end of block Executed by:
| 347 | ||||||||||||||||||
155 | - | |||||||||||||||||||
156 | void QXcbDrag::init() | - | ||||||||||||||||||
157 | { | - | ||||||||||||||||||
158 | currentWindow.clear(); | - | ||||||||||||||||||
159 | - | |||||||||||||||||||
160 | accepted_drop_action = Qt::IgnoreAction; | - | ||||||||||||||||||
161 | - | |||||||||||||||||||
162 | xdnd_dragsource = XCB_NONE; | - | ||||||||||||||||||
163 | - | |||||||||||||||||||
164 | waiting_for_status = false; | - | ||||||||||||||||||
165 | current_target = XCB_NONE; | - | ||||||||||||||||||
166 | current_proxy_target = XCB_NONE; | - | ||||||||||||||||||
167 | - | |||||||||||||||||||
168 | source_time = XCB_CURRENT_TIME; | - | ||||||||||||||||||
169 | target_time = XCB_CURRENT_TIME; | - | ||||||||||||||||||
170 | - | |||||||||||||||||||
171 | QXcbCursor::queryPointer(connection(), ¤t_virtual_desktop, 0); | - | ||||||||||||||||||
172 | drag_types.clear(); | - | ||||||||||||||||||
173 | } executed 135 times by 5 tests: end of block Executed by:
| 135 | ||||||||||||||||||
174 | - | |||||||||||||||||||
175 | QMimeData *QXcbDrag::platformDropData() | - | ||||||||||||||||||
176 | { | - | ||||||||||||||||||
177 | return dropData; executed 9 times by 9 tests: return dropData; Executed by:
| 9 | ||||||||||||||||||
178 | } | - | ||||||||||||||||||
179 | - | |||||||||||||||||||
180 | bool QXcbDrag::eventFilter(QObject *o, QEvent *e) | - | ||||||||||||||||||
181 | { | - | ||||||||||||||||||
182 | /* We are setting a mouse grab on the QShapedPixmapWindow in order not to | - | ||||||||||||||||||
183 | * lose the grab when the virtual desktop changes, but | - | ||||||||||||||||||
184 | * QBasicDrag::eventFilter() expects the events to be coming from the | - | ||||||||||||||||||
185 | * window where the drag was started. */ | - | ||||||||||||||||||
186 | if (initiatorWindow && o == shapedPixmapWindow())
| 0 | ||||||||||||||||||
187 | o = initiatorWindow.data(); never executed: o = initiatorWindow.data(); | 0 | ||||||||||||||||||
188 | return QBasicDrag::eventFilter(o, e); never executed: return QBasicDrag::eventFilter(o, e); | 0 | ||||||||||||||||||
189 | } | - | ||||||||||||||||||
190 | - | |||||||||||||||||||
191 | void QXcbDrag::startDrag() | - | ||||||||||||||||||
192 | { | - | ||||||||||||||||||
193 | // #fixme enableEventFilter(); | - | ||||||||||||||||||
194 | - | |||||||||||||||||||
195 | init(); | - | ||||||||||||||||||
196 | - | |||||||||||||||||||
197 | xcb_set_selection_owner(xcb_connection(), connection()->clipboard()->owner(), | - | ||||||||||||||||||
198 | atom(QXcbAtom::XdndSelection), connection()->time()); | - | ||||||||||||||||||
199 | - | |||||||||||||||||||
200 | QStringList fmts = QXcbMime::formatsHelper(drag()->mimeData()); | - | ||||||||||||||||||
201 | for (int i = 0; i < fmts.size(); ++i) {
| 0 | ||||||||||||||||||
202 | QVector<xcb_atom_t> atoms = QXcbMime::mimeAtomsForFormat(connection(), fmts.at(i)); | - | ||||||||||||||||||
203 | for (int j = 0; j < atoms.size(); ++j) {
| 0 | ||||||||||||||||||
204 | if (!drag_types.contains(atoms.at(j)))
| 0 | ||||||||||||||||||
205 | drag_types.append(atoms.at(j)); never executed: drag_types.append(atoms.at(j)); | 0 | ||||||||||||||||||
206 | } never executed: end of block | 0 | ||||||||||||||||||
207 | } never executed: end of block | 0 | ||||||||||||||||||
208 | if (drag_types.size() > 3)
| 0 | ||||||||||||||||||
209 | xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, connection()->clipboard()->owner(), never executed: xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, connection()->clipboard()->owner(), atom(QXcbAtom::XdndTypelist), XCB_ATOM_ATOM, 32, drag_types.size(), (const void *)drag_types.constData()); | 0 | ||||||||||||||||||
210 | atom(QXcbAtom::XdndTypelist), never executed: xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, connection()->clipboard()->owner(), atom(QXcbAtom::XdndTypelist), XCB_ATOM_ATOM, 32, drag_types.size(), (const void *)drag_types.constData()); | 0 | ||||||||||||||||||
211 | XCB_ATOM_ATOM, 32, drag_types.size(), (const void *)drag_types.constData()); never executed: xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, connection()->clipboard()->owner(), atom(QXcbAtom::XdndTypelist), XCB_ATOM_ATOM, 32, drag_types.size(), (const void *)drag_types.constData()); | 0 | ||||||||||||||||||
212 | - | |||||||||||||||||||
213 | setUseCompositing(current_virtual_desktop->compositingActive()); | - | ||||||||||||||||||
214 | setScreen(current_virtual_desktop->screens().constFirst()->screen()); | - | ||||||||||||||||||
215 | initiatorWindow = QGuiApplicationPrivate::currentMouseWindow; | - | ||||||||||||||||||
216 | QBasicDrag::startDrag(); | - | ||||||||||||||||||
217 | if (connection()->mouseGrabber() == Q_NULLPTR)
| 0 | ||||||||||||||||||
218 | shapedPixmapWindow()->setMouseGrabEnabled(true); never executed: shapedPixmapWindow()->setMouseGrabEnabled(true); | 0 | ||||||||||||||||||
219 | } never executed: end of block | 0 | ||||||||||||||||||
220 | - | |||||||||||||||||||
221 | void QXcbDrag::endDrag() | - | ||||||||||||||||||
222 | { | - | ||||||||||||||||||
223 | QBasicDrag::endDrag(); | - | ||||||||||||||||||
224 | initiatorWindow.clear(); | - | ||||||||||||||||||
225 | } never executed: end of block | 0 | ||||||||||||||||||
226 | - | |||||||||||||||||||
227 | static xcb_translate_coordinates_reply_t * | - | ||||||||||||||||||
228 | translateCoordinates(QXcbConnection *c, xcb_window_t from, xcb_window_t to, int x, int y) | - | ||||||||||||||||||
229 | { | - | ||||||||||||||||||
230 | xcb_translate_coordinates_cookie_t cookie = | - | ||||||||||||||||||
231 | xcb_translate_coordinates(c->xcb_connection(), from, to, x, y); | - | ||||||||||||||||||
232 | return xcb_translate_coordinates_reply(c->xcb_connection(), cookie, 0); never executed: return xcb_translate_coordinates_reply(c->xcb_connection(), cookie, 0); | 0 | ||||||||||||||||||
233 | } | - | ||||||||||||||||||
234 | - | |||||||||||||||||||
235 | static | - | ||||||||||||||||||
236 | bool windowInteractsWithPosition(xcb_connection_t *connection, const QPoint & pos, xcb_window_t w, xcb_shape_sk_t shapeType) | - | ||||||||||||||||||
237 | { | - | ||||||||||||||||||
238 | bool interacts = false; | - | ||||||||||||||||||
239 | xcb_shape_get_rectangles_reply_t *reply = xcb_shape_get_rectangles_reply(connection, xcb_shape_get_rectangles(connection, w, shapeType), NULL); | - | ||||||||||||||||||
240 | if (reply) {
| 0 | ||||||||||||||||||
241 | xcb_rectangle_t *rectangles = xcb_shape_get_rectangles_rectangles(reply); | - | ||||||||||||||||||
242 | if (rectangles) {
| 0 | ||||||||||||||||||
243 | const int nRectangles = xcb_shape_get_rectangles_rectangles_length(reply); | - | ||||||||||||||||||
244 | for (int i = 0; !interacts && i < nRectangles; ++i) {
| 0 | ||||||||||||||||||
245 | interacts = QRect(rectangles[i].x, rectangles[i].y, rectangles[i].width, rectangles[i].height).contains(pos); | - | ||||||||||||||||||
246 | } never executed: end of block | 0 | ||||||||||||||||||
247 | } never executed: end of block | 0 | ||||||||||||||||||
248 | free(reply); | - | ||||||||||||||||||
249 | } never executed: end of block | 0 | ||||||||||||||||||
250 | - | |||||||||||||||||||
251 | return interacts; never executed: return interacts; | 0 | ||||||||||||||||||
252 | } | - | ||||||||||||||||||
253 | - | |||||||||||||||||||
254 | xcb_window_t QXcbDrag::findRealWindow(const QPoint & pos, xcb_window_t w, int md, bool ignoreNonXdndAwareWindows) | - | ||||||||||||||||||
255 | { | - | ||||||||||||||||||
256 | if (w == shapedPixmapWindow()->handle()->winId())
| 0 | ||||||||||||||||||
257 | return 0; never executed: return 0; | 0 | ||||||||||||||||||
258 | - | |||||||||||||||||||
259 | if (md) {
| 0 | ||||||||||||||||||
260 | xcb_get_window_attributes_cookie_t cookie = xcb_get_window_attributes(xcb_connection(), w); | - | ||||||||||||||||||
261 | xcb_get_window_attributes_reply_t *reply = xcb_get_window_attributes_reply(xcb_connection(), cookie, 0); | - | ||||||||||||||||||
262 | if (!reply)
| 0 | ||||||||||||||||||
263 | return 0; never executed: return 0; | 0 | ||||||||||||||||||
264 | - | |||||||||||||||||||
265 | if (reply->map_state != XCB_MAP_STATE_VIEWABLE)
| 0 | ||||||||||||||||||
266 | return 0; never executed: return 0; | 0 | ||||||||||||||||||
267 | - | |||||||||||||||||||
268 | free(reply); | - | ||||||||||||||||||
269 | - | |||||||||||||||||||
270 | xcb_get_geometry_cookie_t gcookie = xcb_get_geometry(xcb_connection(), w); | - | ||||||||||||||||||
271 | xcb_get_geometry_reply_t *greply = xcb_get_geometry_reply(xcb_connection(), gcookie, 0); | - | ||||||||||||||||||
272 | if (!greply)
| 0 | ||||||||||||||||||
273 | return 0; never executed: return 0; | 0 | ||||||||||||||||||
274 | - | |||||||||||||||||||
275 | QRect windowRect(greply->x, greply->y, greply->width, greply->height); | - | ||||||||||||||||||
276 | free(greply); | - | ||||||||||||||||||
277 | if (windowRect.contains(pos)) {
| 0 | ||||||||||||||||||
278 | bool windowContainsMouse = !ignoreNonXdndAwareWindows; | - | ||||||||||||||||||
279 | { | - | ||||||||||||||||||
280 | xcb_get_property_cookie_t cookie = | - | ||||||||||||||||||
281 | Q_XCB_CALL(xcb_get_property(xcb_connection(), false, w, connection()->atom(QXcbAtom::XdndAware), | - | ||||||||||||||||||
282 | XCB_GET_PROPERTY_TYPE_ANY, 0, 0)); | - | ||||||||||||||||||
283 | xcb_get_property_reply_t *reply = xcb_get_property_reply(xcb_connection(), cookie, 0); | - | ||||||||||||||||||
284 | - | |||||||||||||||||||
285 | bool isAware = reply && reply->type != XCB_NONE;
| 0 | ||||||||||||||||||
286 | free(reply); | - | ||||||||||||||||||
287 | if (isAware) {
| 0 | ||||||||||||||||||
288 | const QPoint relPos = pos - windowRect.topLeft(); | - | ||||||||||||||||||
289 | // When ShapeInput and ShapeBounding are not set they return a single rectangle with the geometry of the window, this is why we | - | ||||||||||||||||||
290 | // need to check both here so that in the case one is set and the other is not we still get the correct result. | - | ||||||||||||||||||
291 | if (connection()->hasInputShape())
| 0 | ||||||||||||||||||
292 | windowContainsMouse = windowInteractsWithPosition(xcb_connection(), relPos, w, XCB_SHAPE_SK_INPUT); never executed: windowContainsMouse = windowInteractsWithPosition(xcb_connection(), relPos, w, XCB_SHAPE_SK_INPUT); | 0 | ||||||||||||||||||
293 | if (windowContainsMouse && connection()->hasXShape())
| 0 | ||||||||||||||||||
294 | windowContainsMouse = windowInteractsWithPosition(xcb_connection(), relPos, w, XCB_SHAPE_SK_BOUNDING); never executed: windowContainsMouse = windowInteractsWithPosition(xcb_connection(), relPos, w, XCB_SHAPE_SK_BOUNDING); | 0 | ||||||||||||||||||
295 | if (!connection()->hasInputShape() && !connection()->hasXShape())
| 0 | ||||||||||||||||||
296 | windowContainsMouse = true; never executed: windowContainsMouse = true; | 0 | ||||||||||||||||||
297 | if (windowContainsMouse)
| 0 | ||||||||||||||||||
298 | return w; never executed: return w; | 0 | ||||||||||||||||||
299 | } never executed: end of block | 0 | ||||||||||||||||||
300 | } | - | ||||||||||||||||||
301 | - | |||||||||||||||||||
302 | xcb_query_tree_cookie_t cookie = xcb_query_tree (xcb_connection(), w); | - | ||||||||||||||||||
303 | xcb_query_tree_reply_t *reply = xcb_query_tree_reply(xcb_connection(), cookie, 0); | - | ||||||||||||||||||
304 | - | |||||||||||||||||||
305 | if (!reply)
| 0 | ||||||||||||||||||
306 | return 0; never executed: return 0; | 0 | ||||||||||||||||||
307 | int nc = xcb_query_tree_children_length(reply); | - | ||||||||||||||||||
308 | xcb_window_t *c = xcb_query_tree_children(reply); | - | ||||||||||||||||||
309 | - | |||||||||||||||||||
310 | xcb_window_t r = 0; | - | ||||||||||||||||||
311 | for (uint i = nc; !r && i--;)
| 0 | ||||||||||||||||||
312 | r = findRealWindow(pos - windowRect.topLeft(), c[i], md-1, ignoreNonXdndAwareWindows); never executed: r = findRealWindow(pos - windowRect.topLeft(), c[i], md-1, ignoreNonXdndAwareWindows); | 0 | ||||||||||||||||||
313 | - | |||||||||||||||||||
314 | free(reply); | - | ||||||||||||||||||
315 | if (r)
| 0 | ||||||||||||||||||
316 | return r; never executed: return r; | 0 | ||||||||||||||||||
317 | - | |||||||||||||||||||
318 | // We didn't find a client window! Just use the | - | ||||||||||||||||||
319 | // innermost window. | - | ||||||||||||||||||
320 | - | |||||||||||||||||||
321 | // No children! | - | ||||||||||||||||||
322 | if (!windowContainsMouse)
| 0 | ||||||||||||||||||
323 | return 0; never executed: return 0; | 0 | ||||||||||||||||||
324 | else | - | ||||||||||||||||||
325 | return w; never executed: return w; | 0 | ||||||||||||||||||
326 | } | - | ||||||||||||||||||
327 | } never executed: end of block | 0 | ||||||||||||||||||
328 | return 0; never executed: return 0; | 0 | ||||||||||||||||||
329 | } | - | ||||||||||||||||||
330 | - | |||||||||||||||||||
331 | void QXcbDrag::move(const QPoint &globalPos) | - | ||||||||||||||||||
332 | { | - | ||||||||||||||||||
333 | - | |||||||||||||||||||
334 | if (source_sameanswer.contains(globalPos) && source_sameanswer.isValid())
| 0 | ||||||||||||||||||
335 | return; never executed: return; | 0 | ||||||||||||||||||
336 | - | |||||||||||||||||||
337 | QXcbVirtualDesktop *virtualDesktop = Q_NULLPTR; | - | ||||||||||||||||||
338 | QPoint cursorPos; | - | ||||||||||||||||||
339 | QXcbCursor::queryPointer(connection(), &virtualDesktop, &cursorPos); | - | ||||||||||||||||||
340 | QXcbScreen *screen = virtualDesktop->screenAt(cursorPos); | - | ||||||||||||||||||
341 | QPoint deviceIndependentPos = QHighDpiScaling::mapPositionFromNative(globalPos, screen); | - | ||||||||||||||||||
342 | - | |||||||||||||||||||
343 | if (virtualDesktop != current_virtual_desktop) {
| 0 | ||||||||||||||||||
344 | setUseCompositing(virtualDesktop->compositingActive()); | - | ||||||||||||||||||
345 | recreateShapedPixmapWindow(static_cast<QPlatformScreen*>(screen)->screen(), deviceIndependentPos); | - | ||||||||||||||||||
346 | if (connection()->mouseGrabber() == Q_NULLPTR)
| 0 | ||||||||||||||||||
347 | shapedPixmapWindow()->setMouseGrabEnabled(true); never executed: shapedPixmapWindow()->setMouseGrabEnabled(true); | 0 | ||||||||||||||||||
348 | - | |||||||||||||||||||
349 | current_virtual_desktop = virtualDesktop; | - | ||||||||||||||||||
350 | } else { never executed: end of block | 0 | ||||||||||||||||||
351 | QBasicDrag::moveShapedPixmapWindow(deviceIndependentPos); | - | ||||||||||||||||||
352 | } never executed: end of block | 0 | ||||||||||||||||||
353 | - | |||||||||||||||||||
354 | xcb_window_t rootwin = current_virtual_desktop->root(); | - | ||||||||||||||||||
355 | xcb_translate_coordinates_reply_t *translate = | - | ||||||||||||||||||
356 | ::translateCoordinates(connection(), rootwin, rootwin, globalPos.x(), globalPos.y()); | - | ||||||||||||||||||
357 | if (!translate)
| 0 | ||||||||||||||||||
358 | return; never executed: return; | 0 | ||||||||||||||||||
359 | - | |||||||||||||||||||
360 | xcb_window_t target = translate->child; | - | ||||||||||||||||||
361 | int lx = translate->dst_x; | - | ||||||||||||||||||
362 | int ly = translate->dst_y; | - | ||||||||||||||||||
363 | free (translate); | - | ||||||||||||||||||
364 | - | |||||||||||||||||||
365 | if (target && target != rootwin) {
| 0 | ||||||||||||||||||
366 | xcb_window_t src = rootwin; | - | ||||||||||||||||||
367 | while (target != 0) {
| 0 | ||||||||||||||||||
368 | DNDDEBUG << "checking target for XdndAware" << target << lx << ly; dead code: QMessageLogger(__FILE__, 368, __PRETTY_FUNCTION__).debug() << "checking target for XdndAware" << target << lx << ly; | - | ||||||||||||||||||
369 | - | |||||||||||||||||||
370 | // translate coordinates | - | ||||||||||||||||||
371 | translate = ::translateCoordinates(connection(), src, target, lx, ly); | - | ||||||||||||||||||
372 | if (!translate) {
| 0 | ||||||||||||||||||
373 | target = 0; | - | ||||||||||||||||||
374 | break; never executed: break; | 0 | ||||||||||||||||||
375 | } | - | ||||||||||||||||||
376 | lx = translate->dst_x; | - | ||||||||||||||||||
377 | ly = translate->dst_y; | - | ||||||||||||||||||
378 | src = target; | - | ||||||||||||||||||
379 | xcb_window_t child = translate->child; | - | ||||||||||||||||||
380 | free(translate); | - | ||||||||||||||||||
381 | - | |||||||||||||||||||
382 | // check if it has XdndAware | - | ||||||||||||||||||
383 | xcb_get_property_cookie_t cookie = Q_XCB_CALL(xcb_get_property(xcb_connection(), false, target, | - | ||||||||||||||||||
384 | atom(QXcbAtom::XdndAware), XCB_GET_PROPERTY_TYPE_ANY, 0, 0)); | - | ||||||||||||||||||
385 | xcb_get_property_reply_t *reply = xcb_get_property_reply(xcb_connection(), cookie, 0); | - | ||||||||||||||||||
386 | bool aware = reply && reply->type != XCB_NONE;
| 0 | ||||||||||||||||||
387 | free(reply); | - | ||||||||||||||||||
388 | if (aware) {
| 0 | ||||||||||||||||||
389 | DNDDEBUG << "Found XdndAware on " << target; dead code: QMessageLogger(__FILE__, 389, __PRETTY_FUNCTION__).debug() << "Found XdndAware on " << target; | - | ||||||||||||||||||
390 | break; never executed: break; | 0 | ||||||||||||||||||
391 | } | - | ||||||||||||||||||
392 | - | |||||||||||||||||||
393 | target = child; | - | ||||||||||||||||||
394 | } never executed: end of block | 0 | ||||||||||||||||||
395 | - | |||||||||||||||||||
396 | if (!target || target == shapedPixmapWindow()->handle()->winId()) {
| 0 | ||||||||||||||||||
397 | DNDDEBUG << "need to find real window"; dead code: QMessageLogger(__FILE__, 397, __PRETTY_FUNCTION__).debug() << "need to find real window"; | - | ||||||||||||||||||
398 | target = findRealWindow(globalPos, rootwin, 6, true); | - | ||||||||||||||||||
399 | if (target == 0)
| 0 | ||||||||||||||||||
400 | target = findRealWindow(globalPos, rootwin, 6, false); never executed: target = findRealWindow(globalPos, rootwin, 6, false); | 0 | ||||||||||||||||||
401 | DNDDEBUG << "real window found" << target; dead code: QMessageLogger(__FILE__, 401, __PRETTY_FUNCTION__).debug() << "real window found" << target; | - | ||||||||||||||||||
402 | } never executed: end of block | 0 | ||||||||||||||||||
403 | } never executed: end of block | 0 | ||||||||||||||||||
404 | - | |||||||||||||||||||
405 | QXcbWindow *w = 0; | - | ||||||||||||||||||
406 | if (target) {
| 0 | ||||||||||||||||||
407 | w = connection()->platformWindowFromId(target); | - | ||||||||||||||||||
408 | if (w && (w->window()->type() == Qt::Desktop) /*&& !w->acceptDrops()*/)
| 0 | ||||||||||||||||||
409 | w = 0; never executed: w = 0; | 0 | ||||||||||||||||||
410 | } else { never executed: end of block | 0 | ||||||||||||||||||
411 | w = 0; | - | ||||||||||||||||||
412 | target = rootwin; | - | ||||||||||||||||||
413 | } never executed: end of block | 0 | ||||||||||||||||||
414 | - | |||||||||||||||||||
415 | xcb_window_t proxy_target = xdndProxy(connection(), target); | - | ||||||||||||||||||
416 | if (!proxy_target)
| 0 | ||||||||||||||||||
417 | proxy_target = target; never executed: proxy_target = target; | 0 | ||||||||||||||||||
418 | int target_version = 1; | - | ||||||||||||||||||
419 | - | |||||||||||||||||||
420 | if (proxy_target) {
| 0 | ||||||||||||||||||
421 | xcb_get_property_cookie_t cookie = xcb_get_property(xcb_connection(), false, proxy_target, | - | ||||||||||||||||||
422 | atom(QXcbAtom::XdndAware), XCB_GET_PROPERTY_TYPE_ANY, 0, 1); | - | ||||||||||||||||||
423 | xcb_get_property_reply_t *reply = xcb_get_property_reply(xcb_connection(), cookie, 0); | - | ||||||||||||||||||
424 | if (!reply || reply->type == XCB_NONE)
| 0 | ||||||||||||||||||
425 | target = 0; never executed: target = 0; | 0 | ||||||||||||||||||
426 | - | |||||||||||||||||||
427 | target_version = *(uint32_t *)xcb_get_property_value(reply); | - | ||||||||||||||||||
428 | target_version = qMin(xdnd_version, target_version ? target_version : 1); | - | ||||||||||||||||||
429 | - | |||||||||||||||||||
430 | free(reply); | - | ||||||||||||||||||
431 | } never executed: end of block | 0 | ||||||||||||||||||
432 | - | |||||||||||||||||||
433 | if (target != current_target) {
| 0 | ||||||||||||||||||
434 | if (current_target)
| 0 | ||||||||||||||||||
435 | send_leave(); never executed: send_leave(); | 0 | ||||||||||||||||||
436 | - | |||||||||||||||||||
437 | current_target = target; | - | ||||||||||||||||||
438 | current_proxy_target = proxy_target; | - | ||||||||||||||||||
439 | if (target) {
| 0 | ||||||||||||||||||
440 | int flags = target_version << 24; | - | ||||||||||||||||||
441 | if (drag_types.size() > 3)
| 0 | ||||||||||||||||||
442 | flags |= 0x0001; never executed: flags |= 0x0001; | 0 | ||||||||||||||||||
443 | - | |||||||||||||||||||
444 | xcb_client_message_event_t enter; | - | ||||||||||||||||||
445 | enter.response_type = XCB_CLIENT_MESSAGE; | - | ||||||||||||||||||
446 | enter.sequence = 0; | - | ||||||||||||||||||
447 | enter.window = target; | - | ||||||||||||||||||
448 | enter.format = 32; | - | ||||||||||||||||||
449 | enter.type = atom(QXcbAtom::XdndEnter); | - | ||||||||||||||||||
450 | enter.data.data32[0] = connection()->clipboard()->owner(); | - | ||||||||||||||||||
451 | enter.data.data32[1] = flags; | - | ||||||||||||||||||
452 | enter.data.data32[2] = drag_types.size()>0 ? drag_types.at(0) : 0;
| 0 | ||||||||||||||||||
453 | enter.data.data32[3] = drag_types.size()>1 ? drag_types.at(1) : 0;
| 0 | ||||||||||||||||||
454 | enter.data.data32[4] = drag_types.size()>2 ? drag_types.at(2) : 0;
| 0 | ||||||||||||||||||
455 | // provisionally set the rectangle to 5x5 pixels... | - | ||||||||||||||||||
456 | source_sameanswer = QRect(globalPos.x() - 2, globalPos.y() -2 , 5, 5); | - | ||||||||||||||||||
457 | - | |||||||||||||||||||
458 | DEBUG() << "sending Xdnd enter source=" << enter.data.data32[0]; dead code: QMessageLogger(__FILE__, 458, __PRETTY_FUNCTION__).debug() << "sending Xdnd enter source=" << enter.data.data32[0]; | - | ||||||||||||||||||
459 | if (w)
| 0 | ||||||||||||||||||
460 | handleEnter(w, &enter, current_proxy_target); never executed: handleEnter(w, &enter, current_proxy_target); | 0 | ||||||||||||||||||
461 | else if (target)
| 0 | ||||||||||||||||||
462 | xcb_send_event(xcb_connection(), false, proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&enter); never executed: xcb_send_event(xcb_connection(), false, proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&enter); | 0 | ||||||||||||||||||
463 | waiting_for_status = false; | - | ||||||||||||||||||
464 | } never executed: end of block | 0 | ||||||||||||||||||
465 | } never executed: end of block | 0 | ||||||||||||||||||
466 | - | |||||||||||||||||||
467 | if (waiting_for_status)
| 0 | ||||||||||||||||||
468 | return; never executed: return; | 0 | ||||||||||||||||||
469 | - | |||||||||||||||||||
470 | if (target) {
| 0 | ||||||||||||||||||
471 | waiting_for_status = true; | - | ||||||||||||||||||
472 | - | |||||||||||||||||||
473 | xcb_client_message_event_t move; | - | ||||||||||||||||||
474 | move.response_type = XCB_CLIENT_MESSAGE; | - | ||||||||||||||||||
475 | move.sequence = 0; | - | ||||||||||||||||||
476 | move.window = target; | - | ||||||||||||||||||
477 | move.format = 32; | - | ||||||||||||||||||
478 | move.type = atom(QXcbAtom::XdndPosition); | - | ||||||||||||||||||
479 | move.data.data32[0] = connection()->clipboard()->owner(); | - | ||||||||||||||||||
480 | move.data.data32[1] = 0; // flags | - | ||||||||||||||||||
481 | move.data.data32[2] = (globalPos.x() << 16) + globalPos.y(); | - | ||||||||||||||||||
482 | move.data.data32[3] = connection()->time(); | - | ||||||||||||||||||
483 | move.data.data32[4] = toXdndAction(defaultAction(currentDrag()->supportedActions(), QGuiApplication::keyboardModifiers())); | - | ||||||||||||||||||
484 | DEBUG() << "sending Xdnd position source=" << move.data.data32[0] << "target=" << move.window; dead code: QMessageLogger(__FILE__, 484, __PRETTY_FUNCTION__).debug() << "sending Xdnd position source=" << move.data.data32[0] << "target=" << move.window; | - | ||||||||||||||||||
485 | - | |||||||||||||||||||
486 | source_time = connection()->time(); | - | ||||||||||||||||||
487 | - | |||||||||||||||||||
488 | if (w)
| 0 | ||||||||||||||||||
489 | handle_xdnd_position(w, &move); never executed: handle_xdnd_position(w, &move); | 0 | ||||||||||||||||||
490 | else | - | ||||||||||||||||||
491 | xcb_send_event(xcb_connection(), false, proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&move); never executed: xcb_send_event(xcb_connection(), false, proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&move); | 0 | ||||||||||||||||||
492 | } | - | ||||||||||||||||||
493 | } never executed: end of block | 0 | ||||||||||||||||||
494 | - | |||||||||||||||||||
495 | void QXcbDrag::drop(const QPoint &globalPos) | - | ||||||||||||||||||
496 | { | - | ||||||||||||||||||
497 | QBasicDrag::drop(globalPos); | - | ||||||||||||||||||
498 | - | |||||||||||||||||||
499 | if (!current_target)
| 0 | ||||||||||||||||||
500 | return; never executed: return; | 0 | ||||||||||||||||||
501 | - | |||||||||||||||||||
502 | xcb_client_message_event_t drop; | - | ||||||||||||||||||
503 | drop.response_type = XCB_CLIENT_MESSAGE; | - | ||||||||||||||||||
504 | drop.sequence = 0; | - | ||||||||||||||||||
505 | drop.window = current_target; | - | ||||||||||||||||||
506 | drop.format = 32; | - | ||||||||||||||||||
507 | drop.type = atom(QXcbAtom::XdndDrop); | - | ||||||||||||||||||
508 | drop.data.data32[0] = connection()->clipboard()->owner(); | - | ||||||||||||||||||
509 | drop.data.data32[1] = 0; // flags | - | ||||||||||||||||||
510 | drop.data.data32[2] = connection()->time(); | - | ||||||||||||||||||
511 | - | |||||||||||||||||||
512 | drop.data.data32[3] = 0; | - | ||||||||||||||||||
513 | drop.data.data32[4] = currentDrag()->supportedActions(); | - | ||||||||||||||||||
514 | - | |||||||||||||||||||
515 | QXcbWindow *w = connection()->platformWindowFromId(current_proxy_target); | - | ||||||||||||||||||
516 | - | |||||||||||||||||||
517 | if (w && (w->window()->type() == Qt::Desktop) /*&& !w->acceptDrops()*/)
| 0 | ||||||||||||||||||
518 | w = 0; never executed: w = 0; | 0 | ||||||||||||||||||
519 | - | |||||||||||||||||||
520 | Transaction t = { | - | ||||||||||||||||||
521 | connection()->time(), | - | ||||||||||||||||||
522 | current_target, | - | ||||||||||||||||||
523 | current_proxy_target, | - | ||||||||||||||||||
524 | w, | - | ||||||||||||||||||
525 | // current_embeddig_widget, | - | ||||||||||||||||||
526 | currentDrag(), | - | ||||||||||||||||||
527 | QTime::currentTime() | - | ||||||||||||||||||
528 | }; | - | ||||||||||||||||||
529 | transactions.append(t); | - | ||||||||||||||||||
530 | - | |||||||||||||||||||
531 | // timer is needed only for drops that came from other processes. | - | ||||||||||||||||||
532 | if (!t.targetWindow && cleanup_timer == -1) {
| 0 | ||||||||||||||||||
533 | cleanup_timer = startTimer(XdndDropTransactionTimeout); | - | ||||||||||||||||||
534 | } never executed: end of block | 0 | ||||||||||||||||||
535 | - | |||||||||||||||||||
536 | if (w) {
| 0 | ||||||||||||||||||
537 | handleDrop(w, &drop); | - | ||||||||||||||||||
538 | } else { never executed: end of block | 0 | ||||||||||||||||||
539 | xcb_send_event(xcb_connection(), false, current_proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&drop); | - | ||||||||||||||||||
540 | } never executed: end of block | 0 | ||||||||||||||||||
541 | - | |||||||||||||||||||
542 | current_target = 0; | - | ||||||||||||||||||
543 | current_proxy_target = 0; | - | ||||||||||||||||||
544 | source_time = 0; | - | ||||||||||||||||||
545 | // current_embedding_widget = 0; | - | ||||||||||||||||||
546 | } never executed: end of block | 0 | ||||||||||||||||||
547 | - | |||||||||||||||||||
548 | Qt::DropAction QXcbDrag::toDropAction(xcb_atom_t a) const | - | ||||||||||||||||||
549 | { | - | ||||||||||||||||||
550 | if (a == atom(QXcbAtom::XdndActionCopy) || a == 0)
| 0 | ||||||||||||||||||
551 | return Qt::CopyAction; never executed: return Qt::CopyAction; | 0 | ||||||||||||||||||
552 | if (a == atom(QXcbAtom::XdndActionLink))
| 0 | ||||||||||||||||||
553 | return Qt::LinkAction; never executed: return Qt::LinkAction; | 0 | ||||||||||||||||||
554 | if (a == atom(QXcbAtom::XdndActionMove))
| 0 | ||||||||||||||||||
555 | return Qt::MoveAction; never executed: return Qt::MoveAction; | 0 | ||||||||||||||||||
556 | return Qt::CopyAction; never executed: return Qt::CopyAction; | 0 | ||||||||||||||||||
557 | } | - | ||||||||||||||||||
558 | - | |||||||||||||||||||
559 | xcb_atom_t QXcbDrag::toXdndAction(Qt::DropAction a) const | - | ||||||||||||||||||
560 | { | - | ||||||||||||||||||
561 | switch (a) { | - | ||||||||||||||||||
562 | case Qt::CopyAction: never executed: case Qt::CopyAction: | 0 | ||||||||||||||||||
563 | return atom(QXcbAtom::XdndActionCopy); never executed: return atom(QXcbAtom::XdndActionCopy); | 0 | ||||||||||||||||||
564 | case Qt::LinkAction: never executed: case Qt::LinkAction: | 0 | ||||||||||||||||||
565 | return atom(QXcbAtom::XdndActionLink); never executed: return atom(QXcbAtom::XdndActionLink); | 0 | ||||||||||||||||||
566 | case Qt::MoveAction: never executed: case Qt::MoveAction: | 0 | ||||||||||||||||||
567 | case Qt::TargetMoveAction: never executed: case Qt::TargetMoveAction: | 0 | ||||||||||||||||||
568 | return atom(QXcbAtom::XdndActionMove); never executed: return atom(QXcbAtom::XdndActionMove); | 0 | ||||||||||||||||||
569 | case Qt::IgnoreAction: never executed: case Qt::IgnoreAction: | 0 | ||||||||||||||||||
570 | return XCB_NONE; never executed: return 0L; | 0 | ||||||||||||||||||
571 | default: never executed: default: | 0 | ||||||||||||||||||
572 | return atom(QXcbAtom::XdndActionCopy); never executed: return atom(QXcbAtom::XdndActionCopy); | 0 | ||||||||||||||||||
573 | } | - | ||||||||||||||||||
574 | } | - | ||||||||||||||||||
575 | - | |||||||||||||||||||
576 | int QXcbDrag::findTransactionByWindow(xcb_window_t window) | - | ||||||||||||||||||
577 | { | - | ||||||||||||||||||
578 | int at = -1; | - | ||||||||||||||||||
579 | for (int i = 0; i < transactions.count(); ++i) {
| 0 | ||||||||||||||||||
580 | const Transaction &t = transactions.at(i); | - | ||||||||||||||||||
581 | if (t.target == window || t.proxy_target == window) {
| 0 | ||||||||||||||||||
582 | at = i; | - | ||||||||||||||||||
583 | break; never executed: break; | 0 | ||||||||||||||||||
584 | } | - | ||||||||||||||||||
585 | } never executed: end of block | 0 | ||||||||||||||||||
586 | return at; never executed: return at; | 0 | ||||||||||||||||||
587 | } | - | ||||||||||||||||||
588 | - | |||||||||||||||||||
589 | int QXcbDrag::findTransactionByTime(xcb_timestamp_t timestamp) | - | ||||||||||||||||||
590 | { | - | ||||||||||||||||||
591 | int at = -1; | - | ||||||||||||||||||
592 | for (int i = 0; i < transactions.count(); ++i) {
| 0 | ||||||||||||||||||
593 | const Transaction &t = transactions.at(i); | - | ||||||||||||||||||
594 | if (t.timestamp == timestamp) {
| 0 | ||||||||||||||||||
595 | at = i; | - | ||||||||||||||||||
596 | break; never executed: break; | 0 | ||||||||||||||||||
597 | } | - | ||||||||||||||||||
598 | } never executed: end of block | 0 | ||||||||||||||||||
599 | return at; never executed: return at; | 0 | ||||||||||||||||||
600 | } | - | ||||||||||||||||||
601 | - | |||||||||||||||||||
602 | #if 0 | - | ||||||||||||||||||
603 | - | |||||||||||||||||||
604 | // find an ancestor with XdndAware on it | - | ||||||||||||||||||
605 | static Window findXdndAwareParent(Window window) | - | ||||||||||||||||||
606 | { | - | ||||||||||||||||||
607 | Window target = 0; | - | ||||||||||||||||||
608 | forever { | - | ||||||||||||||||||
609 | // check if window has XdndAware | - | ||||||||||||||||||
610 | Atom type = 0; | - | ||||||||||||||||||
611 | int f; | - | ||||||||||||||||||
612 | unsigned long n, a; | - | ||||||||||||||||||
613 | unsigned char *data = 0; | - | ||||||||||||||||||
614 | if (XGetWindowProperty(X11->display, window, ATOM(XdndAware), 0, 0, False, | - | ||||||||||||||||||
615 | AnyPropertyType, &type, &f,&n,&a,&data) == Success) { | - | ||||||||||||||||||
616 | if (data) | - | ||||||||||||||||||
617 | XFree(data); | - | ||||||||||||||||||
618 | if (type) { | - | ||||||||||||||||||
619 | target = window; | - | ||||||||||||||||||
620 | break; | - | ||||||||||||||||||
621 | } | - | ||||||||||||||||||
622 | } | - | ||||||||||||||||||
623 | - | |||||||||||||||||||
624 | // try window's parent | - | ||||||||||||||||||
625 | Window root; | - | ||||||||||||||||||
626 | Window parent; | - | ||||||||||||||||||
627 | Window *children; | - | ||||||||||||||||||
628 | uint unused; | - | ||||||||||||||||||
629 | if (!XQueryTree(X11->display, window, &root, &parent, &children, &unused)) | - | ||||||||||||||||||
630 | break; | - | ||||||||||||||||||
631 | if (children) | - | ||||||||||||||||||
632 | XFree(children); | - | ||||||||||||||||||
633 | if (window == root) | - | ||||||||||||||||||
634 | break; | - | ||||||||||||||||||
635 | window = parent; | - | ||||||||||||||||||
636 | } | - | ||||||||||||||||||
637 | return target; | - | ||||||||||||||||||
638 | } | - | ||||||||||||||||||
639 | - | |||||||||||||||||||
640 | - | |||||||||||||||||||
641 | // for embedding only | - | ||||||||||||||||||
642 | static QWidget* current_embedding_widget = 0; | - | ||||||||||||||||||
643 | static xcb_client_message_event_t last_enter_event; | - | ||||||||||||||||||
644 | - | |||||||||||||||||||
645 | - | |||||||||||||||||||
646 | static bool checkEmbedded(QWidget* w, const XEvent* xe) | - | ||||||||||||||||||
647 | { | - | ||||||||||||||||||
648 | if (!w) | - | ||||||||||||||||||
649 | return false; | - | ||||||||||||||||||
650 | - | |||||||||||||||||||
651 | if (current_embedding_widget != 0 && current_embedding_widget != w) { | - | ||||||||||||||||||
652 | current_target = ((QExtraWidget*)current_embedding_widget)->extraData()->xDndProxy; | - | ||||||||||||||||||
653 | current_proxy_target = current_target; | - | ||||||||||||||||||
654 | qt_xdnd_send_leave(); | - | ||||||||||||||||||
655 | current_target = 0; | - | ||||||||||||||||||
656 | current_proxy_target = 0; | - | ||||||||||||||||||
657 | current_embedding_widget = 0; | - | ||||||||||||||||||
658 | } | - | ||||||||||||||||||
659 | - | |||||||||||||||||||
660 | QWExtra* extra = ((QExtraWidget*)w)->extraData(); | - | ||||||||||||||||||
661 | if (extra && extra->xDndProxy != 0) { | - | ||||||||||||||||||
662 | - | |||||||||||||||||||
663 | if (current_embedding_widget != w) { | - | ||||||||||||||||||
664 | - | |||||||||||||||||||
665 | last_enter_event.xany.window = extra->xDndProxy; | - | ||||||||||||||||||
666 | XSendEvent(X11->display, extra->xDndProxy, False, NoEventMask, &last_enter_event); | - | ||||||||||||||||||
667 | current_embedding_widget = w; | - | ||||||||||||||||||
668 | } | - | ||||||||||||||||||
669 | - | |||||||||||||||||||
670 | ((XEvent*)xe)->xany.window = extra->xDndProxy; | - | ||||||||||||||||||
671 | XSendEvent(X11->display, extra->xDndProxy, False, NoEventMask, (XEvent*)xe); | - | ||||||||||||||||||
672 | if (currentWindow != w) { | - | ||||||||||||||||||
673 | currentWindow = w; | - | ||||||||||||||||||
674 | } | - | ||||||||||||||||||
675 | return true; | - | ||||||||||||||||||
676 | } | - | ||||||||||||||||||
677 | current_embedding_widget = 0; | - | ||||||||||||||||||
678 | return false; | - | ||||||||||||||||||
679 | } | - | ||||||||||||||||||
680 | #endif | - | ||||||||||||||||||
681 | - | |||||||||||||||||||
682 | - | |||||||||||||||||||
683 | void QXcbDrag::handleEnter(QPlatformWindow *window, const xcb_client_message_event_t *event, xcb_window_t proxy) | - | ||||||||||||||||||
684 | { | - | ||||||||||||||||||
685 | Q_UNUSED(window); | - | ||||||||||||||||||
686 | DEBUG() << "handleEnter" << window; dead code: QMessageLogger(__FILE__, 686, __PRETTY_FUNCTION__).debug() << "handleEnter" << window; | - | ||||||||||||||||||
687 | - | |||||||||||||||||||
688 | xdnd_types.clear(); | - | ||||||||||||||||||
689 | - | |||||||||||||||||||
690 | int version = (int)(event->data.data32[1] >> 24); | - | ||||||||||||||||||
691 | if (version > xdnd_version)
| 0 | ||||||||||||||||||
692 | return; never executed: return; | 0 | ||||||||||||||||||
693 | - | |||||||||||||||||||
694 | xdnd_dragsource = event->data.data32[0]; | - | ||||||||||||||||||
695 | if (!proxy)
| 0 | ||||||||||||||||||
696 | proxy = xdndProxy(connection(), xdnd_dragsource); never executed: proxy = xdndProxy(connection(), xdnd_dragsource); | 0 | ||||||||||||||||||
697 | current_proxy_target = proxy ? proxy : xdnd_dragsource;
| 0 | ||||||||||||||||||
698 | - | |||||||||||||||||||
699 | if (event->data.data32[1] & 1) {
| 0 | ||||||||||||||||||
700 | // get the types from XdndTypeList | - | ||||||||||||||||||
701 | xcb_get_property_cookie_t cookie = xcb_get_property(xcb_connection(), false, xdnd_dragsource, | - | ||||||||||||||||||
702 | atom(QXcbAtom::XdndTypelist), XCB_ATOM_ATOM, | - | ||||||||||||||||||
703 | 0, xdnd_max_type); | - | ||||||||||||||||||
704 | xcb_get_property_reply_t *reply = xcb_get_property_reply(xcb_connection(), cookie, 0); | - | ||||||||||||||||||
705 | if (reply && reply->type != XCB_NONE && reply->format == 32) {
| 0 | ||||||||||||||||||
706 | int length = xcb_get_property_value_length(reply) / 4; | - | ||||||||||||||||||
707 | if (length > xdnd_max_type)
| 0 | ||||||||||||||||||
708 | length = xdnd_max_type; never executed: length = xdnd_max_type; | 0 | ||||||||||||||||||
709 | - | |||||||||||||||||||
710 | xcb_atom_t *atoms = (xcb_atom_t *)xcb_get_property_value(reply); | - | ||||||||||||||||||
711 | xdnd_types.reserve(length); | - | ||||||||||||||||||
712 | for (int i = 0; i < length; ++i)
| 0 | ||||||||||||||||||
713 | xdnd_types.append(atoms[i]); never executed: xdnd_types.append(atoms[i]); | 0 | ||||||||||||||||||
714 | } never executed: end of block | 0 | ||||||||||||||||||
715 | free(reply); | - | ||||||||||||||||||
716 | } else { never executed: end of block | 0 | ||||||||||||||||||
717 | // get the types from the message | - | ||||||||||||||||||
718 | for(int i = 2; i < 5; i++) {
| 0 | ||||||||||||||||||
719 | if (event->data.data32[i])
| 0 | ||||||||||||||||||
720 | xdnd_types.append(event->data.data32[i]); never executed: xdnd_types.append(event->data.data32[i]); | 0 | ||||||||||||||||||
721 | } never executed: end of block | 0 | ||||||||||||||||||
722 | } never executed: end of block | 0 | ||||||||||||||||||
723 | for(int i = 0; i < xdnd_types.length(); ++i)
| 0 | ||||||||||||||||||
724 | DEBUG() << " " << connection()->atomName(xdnd_types.at(i)); dead code: QMessageLogger(__FILE__, 724, __PRETTY_FUNCTION__).debug() << " " << connection()->atomName(xdnd_types.at(i)); | - | ||||||||||||||||||
725 | } never executed: end of block | 0 | ||||||||||||||||||
726 | - | |||||||||||||||||||
727 | void QXcbDrag::handle_xdnd_position(QPlatformWindow *w, const xcb_client_message_event_t *e) | - | ||||||||||||||||||
728 | { | - | ||||||||||||||||||
729 | QPoint p((e->data.data32[2] & 0xffff0000) >> 16, e->data.data32[2] & 0x0000ffff); | - | ||||||||||||||||||
730 | Q_ASSERT(w); | - | ||||||||||||||||||
731 | QRect geometry = w->geometry(); | - | ||||||||||||||||||
732 | p -= geometry.topLeft(); | - | ||||||||||||||||||
733 | - | |||||||||||||||||||
734 | if (!w || !w->window() || (w->window()->type() == Qt::Desktop))
| 0 | ||||||||||||||||||
735 | return; never executed: return; | 0 | ||||||||||||||||||
736 | - | |||||||||||||||||||
737 | if (e->data.data32[0] != xdnd_dragsource) {
| 0 | ||||||||||||||||||
738 | DEBUG("xdnd drag position from unexpected source (%x not %x)", e->data.data32[0], xdnd_dragsource); dead code: QMessageLogger(__FILE__, 738, __PRETTY_FUNCTION__).debug("xdnd drag position from unexpected source (%x not %x)", e->data.data32[0], xdnd_dragsource); | - | ||||||||||||||||||
739 | return; never executed: return; | 0 | ||||||||||||||||||
740 | } | - | ||||||||||||||||||
741 | - | |||||||||||||||||||
742 | currentPosition = p; | - | ||||||||||||||||||
743 | currentWindow = w->window(); | - | ||||||||||||||||||
744 | - | |||||||||||||||||||
745 | // timestamp from the source | - | ||||||||||||||||||
746 | if (e->data.data32[3] != XCB_NONE) {
| 0 | ||||||||||||||||||
747 | target_time = e->data.data32[3]; | - | ||||||||||||||||||
748 | } never executed: end of block | 0 | ||||||||||||||||||
749 | - | |||||||||||||||||||
750 | QMimeData *dropData = 0; | - | ||||||||||||||||||
751 | Qt::DropActions supported_actions = Qt::IgnoreAction; | - | ||||||||||||||||||
752 | if (currentDrag()) {
| 0 | ||||||||||||||||||
753 | dropData = currentDrag()->mimeData(); | - | ||||||||||||||||||
754 | supported_actions = currentDrag()->supportedActions(); | - | ||||||||||||||||||
755 | } else { never executed: end of block | 0 | ||||||||||||||||||
756 | dropData = platformDropData(); | - | ||||||||||||||||||
757 | supported_actions = Qt::DropActions(toDropAction(e->data.data32[4])); | - | ||||||||||||||||||
758 | } never executed: end of block | 0 | ||||||||||||||||||
759 | - | |||||||||||||||||||
760 | QPlatformDragQtResponse qt_response = QWindowSystemInterface::handleDrag(w->window(),dropData,p,supported_actions); | - | ||||||||||||||||||
761 | QRect answerRect(p + geometry.topLeft(), QSize(1,1)); | - | ||||||||||||||||||
762 | answerRect = qt_response.answerRect().translated(geometry.topLeft()).intersected(geometry); | - | ||||||||||||||||||
763 | - | |||||||||||||||||||
764 | xcb_client_message_event_t response; | - | ||||||||||||||||||
765 | response.response_type = XCB_CLIENT_MESSAGE; | - | ||||||||||||||||||
766 | response.sequence = 0; | - | ||||||||||||||||||
767 | response.window = xdnd_dragsource; | - | ||||||||||||||||||
768 | response.format = 32; | - | ||||||||||||||||||
769 | response.type = atom(QXcbAtom::XdndStatus); | - | ||||||||||||||||||
770 | response.data.data32[0] = xcb_window(w); | - | ||||||||||||||||||
771 | response.data.data32[1] = qt_response.isAccepted(); // flags | - | ||||||||||||||||||
772 | response.data.data32[2] = 0; // x, y | - | ||||||||||||||||||
773 | response.data.data32[3] = 0; // w, h | - | ||||||||||||||||||
774 | response.data.data32[4] = toXdndAction(qt_response.acceptedAction()); // action | - | ||||||||||||||||||
775 | - | |||||||||||||||||||
776 | accepted_drop_action = qt_response.acceptedAction(); | - | ||||||||||||||||||
777 | - | |||||||||||||||||||
778 | if (answerRect.left() < 0)
| 0 | ||||||||||||||||||
779 | answerRect.setLeft(0); never executed: answerRect.setLeft(0); | 0 | ||||||||||||||||||
780 | if (answerRect.right() > 4096)
| 0 | ||||||||||||||||||
781 | answerRect.setRight(4096); never executed: answerRect.setRight(4096); | 0 | ||||||||||||||||||
782 | if (answerRect.top() < 0)
| 0 | ||||||||||||||||||
783 | answerRect.setTop(0); never executed: answerRect.setTop(0); | 0 | ||||||||||||||||||
784 | if (answerRect.bottom() > 4096)
| 0 | ||||||||||||||||||
785 | answerRect.setBottom(4096); never executed: answerRect.setBottom(4096); | 0 | ||||||||||||||||||
786 | if (answerRect.width() < 0)
| 0 | ||||||||||||||||||
787 | answerRect.setWidth(0); never executed: answerRect.setWidth(0); | 0 | ||||||||||||||||||
788 | if (answerRect.height() < 0)
| 0 | ||||||||||||||||||
789 | answerRect.setHeight(0); never executed: answerRect.setHeight(0); | 0 | ||||||||||||||||||
790 | - | |||||||||||||||||||
791 | // reset | - | ||||||||||||||||||
792 | target_time = XCB_CURRENT_TIME; | - | ||||||||||||||||||
793 | - | |||||||||||||||||||
794 | if (xdnd_dragsource == connection()->clipboard()->owner())
| 0 | ||||||||||||||||||
795 | handle_xdnd_status(&response); never executed: handle_xdnd_status(&response); | 0 | ||||||||||||||||||
796 | else | - | ||||||||||||||||||
797 | Q_XCB_CALL(xcb_send_event(xcb_connection(), false, current_proxy_target, never executed: xcb_send_event(xcb_connection(), false, current_proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&response) ; | 0 | ||||||||||||||||||
798 | XCB_EVENT_MASK_NO_EVENT, (const char *)&response)); never executed: xcb_send_event(xcb_connection(), false, current_proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&response) ; | 0 | ||||||||||||||||||
799 | } | - | ||||||||||||||||||
800 | - | |||||||||||||||||||
801 | namespace | - | ||||||||||||||||||
802 | { | - | ||||||||||||||||||
803 | class ClientMessageScanner { | - | ||||||||||||||||||
804 | public: | - | ||||||||||||||||||
805 | ClientMessageScanner(xcb_atom_t a) : atom(a) {} never executed: end of block | 0 | ||||||||||||||||||
806 | xcb_atom_t atom; | - | ||||||||||||||||||
807 | bool checkEvent(xcb_generic_event_t *event) const { | - | ||||||||||||||||||
808 | if (!event)
| 0 | ||||||||||||||||||
809 | return false; never executed: return false; | 0 | ||||||||||||||||||
810 | if ((event->response_type & 0x7f) != XCB_CLIENT_MESSAGE)
| 0 | ||||||||||||||||||
811 | return false; never executed: return false; | 0 | ||||||||||||||||||
812 | return ((xcb_client_message_event_t *)event)->type == atom; never executed: return ((xcb_client_message_event_t *)event)->type == atom; | 0 | ||||||||||||||||||
813 | } | - | ||||||||||||||||||
814 | }; | - | ||||||||||||||||||
815 | } | - | ||||||||||||||||||
816 | - | |||||||||||||||||||
817 | void QXcbDrag::handlePosition(QPlatformWindow * w, const xcb_client_message_event_t *event) | - | ||||||||||||||||||
818 | { | - | ||||||||||||||||||
819 | xcb_client_message_event_t *lastEvent = const_cast<xcb_client_message_event_t *>(event); | - | ||||||||||||||||||
820 | xcb_generic_event_t *nextEvent; | - | ||||||||||||||||||
821 | ClientMessageScanner scanner(atom(QXcbAtom::XdndPosition)); | - | ||||||||||||||||||
822 | while ((nextEvent = connection()->checkEvent(scanner))) {
| 0 | ||||||||||||||||||
823 | if (lastEvent != event)
| 0 | ||||||||||||||||||
824 | free(lastEvent); never executed: free(lastEvent); | 0 | ||||||||||||||||||
825 | lastEvent = (xcb_client_message_event_t *)nextEvent; | - | ||||||||||||||||||
826 | } never executed: end of block | 0 | ||||||||||||||||||
827 | - | |||||||||||||||||||
828 | handle_xdnd_position(w, lastEvent); | - | ||||||||||||||||||
829 | if (lastEvent != event)
| 0 | ||||||||||||||||||
830 | free(lastEvent); never executed: free(lastEvent); | 0 | ||||||||||||||||||
831 | } never executed: end of block | 0 | ||||||||||||||||||
832 | - | |||||||||||||||||||
833 | void QXcbDrag::handle_xdnd_status(const xcb_client_message_event_t *event) | - | ||||||||||||||||||
834 | { | - | ||||||||||||||||||
835 | DEBUG("xdndHandleStatus"); dead code: QMessageLogger(__FILE__, 835, __PRETTY_FUNCTION__).debug("xdndHandleStatus"); | - | ||||||||||||||||||
836 | waiting_for_status = false; | - | ||||||||||||||||||
837 | // ignore late status messages | - | ||||||||||||||||||
838 | if (event->data.data32[0] && event->data.data32[0] != current_target)
| 0 | ||||||||||||||||||
839 | return; never executed: return; | 0 | ||||||||||||||||||
840 | - | |||||||||||||||||||
841 | const bool dropPossible = event->data.data32[1]; | - | ||||||||||||||||||
842 | setCanDrop(dropPossible); | - | ||||||||||||||||||
843 | - | |||||||||||||||||||
844 | if (dropPossible) {
| 0 | ||||||||||||||||||
845 | accepted_drop_action = toDropAction(event->data.data32[4]); | - | ||||||||||||||||||
846 | updateCursor(accepted_drop_action); | - | ||||||||||||||||||
847 | } else { never executed: end of block | 0 | ||||||||||||||||||
848 | updateCursor(Qt::IgnoreAction); | - | ||||||||||||||||||
849 | } never executed: end of block | 0 | ||||||||||||||||||
850 | - | |||||||||||||||||||
851 | if ((event->data.data32[1] & 2) == 0) {
| 0 | ||||||||||||||||||
852 | QPoint p((event->data.data32[2] & 0xffff0000) >> 16, event->data.data32[2] & 0x0000ffff); | - | ||||||||||||||||||
853 | QSize s((event->data.data32[3] & 0xffff0000) >> 16, event->data.data32[3] & 0x0000ffff); | - | ||||||||||||||||||
854 | source_sameanswer = QRect(p, s); | - | ||||||||||||||||||
855 | } else { never executed: end of block | 0 | ||||||||||||||||||
856 | source_sameanswer = QRect(); | - | ||||||||||||||||||
857 | } never executed: end of block | 0 | ||||||||||||||||||
858 | } | - | ||||||||||||||||||
859 | - | |||||||||||||||||||
860 | void QXcbDrag::handleStatus(const xcb_client_message_event_t *event) | - | ||||||||||||||||||
861 | { | - | ||||||||||||||||||
862 | if (event->window != connection()->clipboard()->owner() || !drag())
| 0 | ||||||||||||||||||
863 | return; never executed: return; | 0 | ||||||||||||||||||
864 | - | |||||||||||||||||||
865 | xcb_client_message_event_t *lastEvent = const_cast<xcb_client_message_event_t *>(event); | - | ||||||||||||||||||
866 | xcb_generic_event_t *nextEvent; | - | ||||||||||||||||||
867 | ClientMessageScanner scanner(atom(QXcbAtom::XdndStatus)); | - | ||||||||||||||||||
868 | while ((nextEvent = connection()->checkEvent(scanner))) {
| 0 | ||||||||||||||||||
869 | if (lastEvent != event)
| 0 | ||||||||||||||||||
870 | free(lastEvent); never executed: free(lastEvent); | 0 | ||||||||||||||||||
871 | lastEvent = (xcb_client_message_event_t *)nextEvent; | - | ||||||||||||||||||
872 | } never executed: end of block | 0 | ||||||||||||||||||
873 | - | |||||||||||||||||||
874 | handle_xdnd_status(lastEvent); | - | ||||||||||||||||||
875 | if (lastEvent != event)
| 0 | ||||||||||||||||||
876 | free(lastEvent); never executed: free(lastEvent); | 0 | ||||||||||||||||||
877 | DEBUG("xdndHandleStatus end"); dead code: QMessageLogger(__FILE__, 877, __PRETTY_FUNCTION__).debug("xdndHandleStatus end"); | - | ||||||||||||||||||
878 | } never executed: end of block | 0 | ||||||||||||||||||
879 | - | |||||||||||||||||||
880 | void QXcbDrag::handleLeave(QPlatformWindow *w, const xcb_client_message_event_t *event) | - | ||||||||||||||||||
881 | { | - | ||||||||||||||||||
882 | DEBUG("xdnd leave"); dead code: QMessageLogger(__FILE__, 882, __PRETTY_FUNCTION__).debug("xdnd leave"); | - | ||||||||||||||||||
883 | if (!currentWindow || w != currentWindow.data()->handle())
| 0 | ||||||||||||||||||
884 | return; // sanity never executed: return; | 0 | ||||||||||||||||||
885 | - | |||||||||||||||||||
886 | // ### | - | ||||||||||||||||||
887 | // if (checkEmbedded(current_embedding_widget, event)) { | - | ||||||||||||||||||
888 | // current_embedding_widget = 0; | - | ||||||||||||||||||
889 | // currentWindow.clear(); | - | ||||||||||||||||||
890 | // return; | - | ||||||||||||||||||
891 | // } | - | ||||||||||||||||||
892 | - | |||||||||||||||||||
893 | if (event->data.data32[0] != xdnd_dragsource) {
| 0 | ||||||||||||||||||
894 | // This often happens - leave other-process window quickly | - | ||||||||||||||||||
895 | DEBUG("xdnd drag leave from unexpected source (%x not %x", event->data.data32[0], xdnd_dragsource); dead code: QMessageLogger(__FILE__, 895, __PRETTY_FUNCTION__).debug("xdnd drag leave from unexpected source (%x not %x", event->data.data32[0], xdnd_dragsource); | - | ||||||||||||||||||
896 | } never executed: end of block | 0 | ||||||||||||||||||
897 | - | |||||||||||||||||||
898 | QWindowSystemInterface::handleDrag(w->window(),0,QPoint(),Qt::IgnoreAction); | - | ||||||||||||||||||
899 | - | |||||||||||||||||||
900 | xdnd_dragsource = 0; | - | ||||||||||||||||||
901 | xdnd_types.clear(); | - | ||||||||||||||||||
902 | currentWindow.clear(); | - | ||||||||||||||||||
903 | } never executed: end of block | 0 | ||||||||||||||||||
904 | - | |||||||||||||||||||
905 | void QXcbDrag::send_leave() | - | ||||||||||||||||||
906 | { | - | ||||||||||||||||||
907 | if (!current_target)
| 0 | ||||||||||||||||||
908 | return; never executed: return; | 0 | ||||||||||||||||||
909 | - | |||||||||||||||||||
910 | - | |||||||||||||||||||
911 | xcb_client_message_event_t leave; | - | ||||||||||||||||||
912 | leave.response_type = XCB_CLIENT_MESSAGE; | - | ||||||||||||||||||
913 | leave.sequence = 0; | - | ||||||||||||||||||
914 | leave.window = current_target; | - | ||||||||||||||||||
915 | leave.format = 32; | - | ||||||||||||||||||
916 | leave.type = atom(QXcbAtom::XdndLeave); | - | ||||||||||||||||||
917 | leave.data.data32[0] = connection()->clipboard()->owner(); | - | ||||||||||||||||||
918 | leave.data.data32[1] = 0; // flags | - | ||||||||||||||||||
919 | leave.data.data32[2] = 0; // x, y | - | ||||||||||||||||||
920 | leave.data.data32[3] = 0; // w, h | - | ||||||||||||||||||
921 | leave.data.data32[4] = 0; // just null | - | ||||||||||||||||||
922 | - | |||||||||||||||||||
923 | QXcbWindow *w = connection()->platformWindowFromId(current_proxy_target); | - | ||||||||||||||||||
924 | - | |||||||||||||||||||
925 | if (w && (w->window()->type() == Qt::Desktop) /*&& !w->acceptDrops()*/)
| 0 | ||||||||||||||||||
926 | w = 0; never executed: w = 0; | 0 | ||||||||||||||||||
927 | - | |||||||||||||||||||
928 | if (w)
| 0 | ||||||||||||||||||
929 | handleLeave(w, (const xcb_client_message_event_t *)&leave); never executed: handleLeave(w, (const xcb_client_message_event_t *)&leave); | 0 | ||||||||||||||||||
930 | else | - | ||||||||||||||||||
931 | xcb_send_event(xcb_connection(), false,current_proxy_target, never executed: xcb_send_event(xcb_connection(), false,current_proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&leave); | 0 | ||||||||||||||||||
932 | XCB_EVENT_MASK_NO_EVENT, (const char *)&leave); never executed: xcb_send_event(xcb_connection(), false,current_proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)&leave); | 0 | ||||||||||||||||||
933 | - | |||||||||||||||||||
934 | current_target = 0; | - | ||||||||||||||||||
935 | current_proxy_target = 0; | - | ||||||||||||||||||
936 | source_time = XCB_CURRENT_TIME; | - | ||||||||||||||||||
937 | waiting_for_status = false; | - | ||||||||||||||||||
938 | } never executed: end of block | 0 | ||||||||||||||||||
939 | - | |||||||||||||||||||
940 | void QXcbDrag::handleDrop(QPlatformWindow *, const xcb_client_message_event_t *event) | - | ||||||||||||||||||
941 | { | - | ||||||||||||||||||
942 | DEBUG("xdndHandleDrop"); dead code: QMessageLogger(__FILE__, 942, __PRETTY_FUNCTION__).debug("xdndHandleDrop"); | - | ||||||||||||||||||
943 | if (!currentWindow) {
| 0 | ||||||||||||||||||
944 | xdnd_dragsource = 0; | - | ||||||||||||||||||
945 | return; // sanity never executed: return; | 0 | ||||||||||||||||||
946 | } | - | ||||||||||||||||||
947 | - | |||||||||||||||||||
948 | const uint32_t *l = event->data.data32; | - | ||||||||||||||||||
949 | - | |||||||||||||||||||
950 | DEBUG("xdnd drop"); dead code: QMessageLogger(__FILE__, 950, __PRETTY_FUNCTION__).debug("xdnd drop"); | - | ||||||||||||||||||
951 | - | |||||||||||||||||||
952 | if (l[0] != xdnd_dragsource) {
| 0 | ||||||||||||||||||
953 | DEBUG("xdnd drop from unexpected source (%x not %x", l[0], xdnd_dragsource); dead code: QMessageLogger(__FILE__, 953, __PRETTY_FUNCTION__).debug("xdnd drop from unexpected source (%x not %x", l[0], xdnd_dragsource); | - | ||||||||||||||||||
954 | return; never executed: return; | 0 | ||||||||||||||||||
955 | } | - | ||||||||||||||||||
956 | - | |||||||||||||||||||
957 | // update the "user time" from the timestamp in the event. | - | ||||||||||||||||||
958 | if (l[2] != 0)
| 0 | ||||||||||||||||||
959 | target_time = /*X11->userTime =*/ l[2]; never executed: target_time = l[2]; | 0 | ||||||||||||||||||
960 | - | |||||||||||||||||||
961 | Qt::DropActions supported_drop_actions; | - | ||||||||||||||||||
962 | QMimeData *dropData = 0; | - | ||||||||||||||||||
963 | if (currentDrag()) {
| 0 | ||||||||||||||||||
964 | dropData = currentDrag()->mimeData(); | - | ||||||||||||||||||
965 | supported_drop_actions = Qt::DropActions(l[4]); | - | ||||||||||||||||||
966 | } else { never executed: end of block | 0 | ||||||||||||||||||
967 | dropData = platformDropData(); | - | ||||||||||||||||||
968 | supported_drop_actions = accepted_drop_action; | - | ||||||||||||||||||
969 | - | |||||||||||||||||||
970 | // Drop coming from another app? Update keyboard modifiers. | - | ||||||||||||||||||
971 | QGuiApplicationPrivate::modifier_buttons = QGuiApplication::queryKeyboardModifiers(); | - | ||||||||||||||||||
972 | } never executed: end of block | 0 | ||||||||||||||||||
973 | - | |||||||||||||||||||
974 | if (!dropData)
| 0 | ||||||||||||||||||
975 | return; never executed: return; | 0 | ||||||||||||||||||
976 | // ### | - | ||||||||||||||||||
977 | // int at = findXdndDropTransactionByTime(target_time); | - | ||||||||||||||||||
978 | // if (at != -1) | - | ||||||||||||||||||
979 | // dropData = QDragManager::dragPrivate(X11->dndDropTransactions.at(at).object)->data; | - | ||||||||||||||||||
980 | // if we can't find it, then use the data in the drag manager | - | ||||||||||||||||||
981 | - | |||||||||||||||||||
982 | QPlatformDropQtResponse response = QWindowSystemInterface::handleDrop(currentWindow.data(),dropData,currentPosition,supported_drop_actions); | - | ||||||||||||||||||
983 | setExecutedDropAction(response.acceptedAction()); | - | ||||||||||||||||||
984 | - | |||||||||||||||||||
985 | xcb_client_message_event_t finished; | - | ||||||||||||||||||
986 | finished.response_type = XCB_CLIENT_MESSAGE; | - | ||||||||||||||||||
987 | finished.sequence = 0; | - | ||||||||||||||||||
988 | finished.window = xdnd_dragsource; | - | ||||||||||||||||||
989 | finished.format = 32; | - | ||||||||||||||||||
990 | finished.type = atom(QXcbAtom::XdndFinished); | - | ||||||||||||||||||
991 | finished.data.data32[0] = currentWindow ? xcb_window(currentWindow.data()) : XCB_NONE;
| 0 | ||||||||||||||||||
992 | finished.data.data32[1] = response.isAccepted(); // flags | - | ||||||||||||||||||
993 | finished.data.data32[2] = toXdndAction(response.acceptedAction()); | - | ||||||||||||||||||
994 | Q_XCB_CALL(xcb_send_event(xcb_connection(), false, current_proxy_target, | - | ||||||||||||||||||
995 | XCB_EVENT_MASK_NO_EVENT, (char *)&finished)); | - | ||||||||||||||||||
996 | - | |||||||||||||||||||
997 | xdnd_dragsource = 0; | - | ||||||||||||||||||
998 | currentWindow.clear(); | - | ||||||||||||||||||
999 | waiting_for_status = false; | - | ||||||||||||||||||
1000 | - | |||||||||||||||||||
1001 | // reset | - | ||||||||||||||||||
1002 | target_time = XCB_CURRENT_TIME; | - | ||||||||||||||||||
1003 | } never executed: end of block | 0 | ||||||||||||||||||
1004 | - | |||||||||||||||||||
1005 | - | |||||||||||||||||||
1006 | void QXcbDrag::handleFinished(const xcb_client_message_event_t *event) | - | ||||||||||||||||||
1007 | { | - | ||||||||||||||||||
1008 | DEBUG("xdndHandleFinished"); dead code: QMessageLogger(__FILE__, 1008, __PRETTY_FUNCTION__).debug("xdndHandleFinished"); | - | ||||||||||||||||||
1009 | if (event->window != connection()->clipboard()->owner())
| 0 | ||||||||||||||||||
1010 | return; never executed: return; | 0 | ||||||||||||||||||
1011 | - | |||||||||||||||||||
1012 | const unsigned long *l = (const unsigned long *)event->data.data32; | - | ||||||||||||||||||
1013 | - | |||||||||||||||||||
1014 | DNDDEBUG << "xdndHandleFinished, l[0]" << l[0] dead code: QMessageLogger(__FILE__, 1014, __PRETTY_FUNCTION__).debug() << "xdndHandleFinished, l[0]" << l[0] << "current_target" << current_target << "qt_xdnd_current_proxy_targe" << current_proxy_target; | - | ||||||||||||||||||
1015 | << "current_target" << current_target dead code: QMessageLogger(__FILE__, 1014, __PRETTY_FUNCTION__).debug() << "xdndHandleFinished, l[0]" << l[0] << "current_target" << current_target << "qt_xdnd_current_proxy_targe" << current_proxy_target; | - | ||||||||||||||||||
1016 | << "qt_xdnd_current_proxy_targe" << current_proxy_target; dead code: QMessageLogger(__FILE__, 1014, __PRETTY_FUNCTION__).debug() << "xdndHandleFinished, l[0]" << l[0] << "current_target" << current_target << "qt_xdnd_current_proxy_targe" << current_proxy_target; | - | ||||||||||||||||||
1017 | - | |||||||||||||||||||
1018 | if (l[0]) {
| 0 | ||||||||||||||||||
1019 | int at = findTransactionByWindow(l[0]); | - | ||||||||||||||||||
1020 | if (at != -1) {
| 0 | ||||||||||||||||||
1021 | - | |||||||||||||||||||
1022 | Transaction t = transactions.takeAt(at); | - | ||||||||||||||||||
1023 | if (t.drag)
| 0 | ||||||||||||||||||
1024 | t.drag->deleteLater(); never executed: t.drag->deleteLater(); | 0 | ||||||||||||||||||
1025 | // QDragManager *manager = QDragManager::self(); | - | ||||||||||||||||||
1026 | - | |||||||||||||||||||
1027 | // Window target = current_target; | - | ||||||||||||||||||
1028 | // Window proxy_target = current_proxy_target; | - | ||||||||||||||||||
1029 | // QWidget *embedding_widget = current_embedding_widget; | - | ||||||||||||||||||
1030 | // QDrag *currentObject = manager->object; | - | ||||||||||||||||||
1031 | - | |||||||||||||||||||
1032 | // current_target = t.target; | - | ||||||||||||||||||
1033 | // current_proxy_target = t.proxy_target; | - | ||||||||||||||||||
1034 | // current_embedding_widget = t.embedding_widget; | - | ||||||||||||||||||
1035 | // manager->object = t.object; | - | ||||||||||||||||||
1036 | - | |||||||||||||||||||
1037 | // if (!passive) | - | ||||||||||||||||||
1038 | // (void) checkEmbedded(currentWindow, xe); | - | ||||||||||||||||||
1039 | - | |||||||||||||||||||
1040 | // current_embedding_widget = 0; | - | ||||||||||||||||||
1041 | // current_target = 0; | - | ||||||||||||||||||
1042 | // current_proxy_target = 0; | - | ||||||||||||||||||
1043 | - | |||||||||||||||||||
1044 | // current_target = target; | - | ||||||||||||||||||
1045 | // current_proxy_target = proxy_target; | - | ||||||||||||||||||
1046 | // current_embedding_widget = embedding_widget; | - | ||||||||||||||||||
1047 | // manager->object = currentObject; | - | ||||||||||||||||||
1048 | } else { never executed: end of block | 0 | ||||||||||||||||||
1049 | qWarning("QXcbDrag::handleFinished - drop data has expired"); | - | ||||||||||||||||||
1050 | } never executed: end of block | 0 | ||||||||||||||||||
1051 | } | - | ||||||||||||||||||
1052 | waiting_for_status = false; | - | ||||||||||||||||||
1053 | } never executed: end of block | 0 | ||||||||||||||||||
1054 | - | |||||||||||||||||||
1055 | void QXcbDrag::timerEvent(QTimerEvent* e) | - | ||||||||||||||||||
1056 | { | - | ||||||||||||||||||
1057 | if (e->timerId() == cleanup_timer) {
| 0 | ||||||||||||||||||
1058 | bool stopTimer = true; | - | ||||||||||||||||||
1059 | for (int i = 0; i < transactions.count(); ++i) {
| 0 | ||||||||||||||||||
1060 | const Transaction &t = transactions.at(i); | - | ||||||||||||||||||
1061 | if (t.targetWindow) {
| 0 | ||||||||||||||||||
1062 | // dnd within the same process, don't delete, these are taken care of | - | ||||||||||||||||||
1063 | // in handleFinished() | - | ||||||||||||||||||
1064 | continue; never executed: continue; | 0 | ||||||||||||||||||
1065 | } | - | ||||||||||||||||||
1066 | QTime currentTime = QTime::currentTime(); | - | ||||||||||||||||||
1067 | int delta = t.time.msecsTo(currentTime); | - | ||||||||||||||||||
1068 | if (delta > XdndDropTransactionTimeout) {
| 0 | ||||||||||||||||||
1069 | /* delete transactions which are older than XdndDropTransactionTimeout. It could mean | - | ||||||||||||||||||
1070 | one of these: | - | ||||||||||||||||||
1071 | - client has crashed and as a result we have never received XdndFinished | - | ||||||||||||||||||
1072 | - showing dialog box on drop event where user's response takes more time than XdndDropTransactionTimeout (QTBUG-14493) | - | ||||||||||||||||||
1073 | - dnd takes unusually long time to process data | - | ||||||||||||||||||
1074 | */ | - | ||||||||||||||||||
1075 | if (t.drag)
| 0 | ||||||||||||||||||
1076 | t.drag->deleteLater(); never executed: t.drag->deleteLater(); | 0 | ||||||||||||||||||
1077 | transactions.removeAt(i--); | - | ||||||||||||||||||
1078 | } else { never executed: end of block | 0 | ||||||||||||||||||
1079 | stopTimer = false; | - | ||||||||||||||||||
1080 | } never executed: end of block | 0 | ||||||||||||||||||
1081 | - | |||||||||||||||||||
1082 | } | - | ||||||||||||||||||
1083 | if (stopTimer && cleanup_timer != -1) {
| 0 | ||||||||||||||||||
1084 | killTimer(cleanup_timer); | - | ||||||||||||||||||
1085 | cleanup_timer = -1; | - | ||||||||||||||||||
1086 | } never executed: end of block | 0 | ||||||||||||||||||
1087 | } never executed: end of block | 0 | ||||||||||||||||||
1088 | } never executed: end of block | 0 | ||||||||||||||||||
1089 | - | |||||||||||||||||||
1090 | void QXcbDrag::cancel() | - | ||||||||||||||||||
1091 | { | - | ||||||||||||||||||
1092 | DEBUG("QXcbDrag::cancel"); dead code: QMessageLogger(__FILE__, 1092, __PRETTY_FUNCTION__).debug("QXcbDrag::cancel"); | - | ||||||||||||||||||
1093 | QBasicDrag::cancel(); | - | ||||||||||||||||||
1094 | if (current_target)
| 0 | ||||||||||||||||||
1095 | send_leave(); never executed: send_leave(); | 0 | ||||||||||||||||||
1096 | } never executed: end of block | 0 | ||||||||||||||||||
1097 | - | |||||||||||||||||||
1098 | // find an ancestor with XdndAware on it | - | ||||||||||||||||||
1099 | static xcb_window_t findXdndAwareParent(QXcbConnection *c, xcb_window_t window) | - | ||||||||||||||||||
1100 | { | - | ||||||||||||||||||
1101 | xcb_window_t target = 0; | - | ||||||||||||||||||
1102 | forever { | - | ||||||||||||||||||
1103 | // check if window has XdndAware | - | ||||||||||||||||||
1104 | xcb_get_property_cookie_t gpCookie = Q_XCB_CALL( | - | ||||||||||||||||||
1105 | xcb_get_property(c->xcb_connection(), false, window, | - | ||||||||||||||||||
1106 | c->atom(QXcbAtom::XdndAware), XCB_GET_PROPERTY_TYPE_ANY, 0, 0)); | - | ||||||||||||||||||
1107 | xcb_get_property_reply_t *gpReply = xcb_get_property_reply( | - | ||||||||||||||||||
1108 | c->xcb_connection(), gpCookie, 0); | - | ||||||||||||||||||
1109 | bool aware = gpReply && gpReply->type != XCB_NONE;
| 0 | ||||||||||||||||||
1110 | free(gpReply); | - | ||||||||||||||||||
1111 | if (aware) {
| 0 | ||||||||||||||||||
1112 | target = window; | - | ||||||||||||||||||
1113 | break; never executed: break; | 0 | ||||||||||||||||||
1114 | } | - | ||||||||||||||||||
1115 | - | |||||||||||||||||||
1116 | // try window's parent | - | ||||||||||||||||||
1117 | xcb_query_tree_cookie_t qtCookie = Q_XCB_CALL( | - | ||||||||||||||||||
1118 | xcb_query_tree_unchecked(c->xcb_connection(), window)); | - | ||||||||||||||||||
1119 | xcb_query_tree_reply_t *qtReply = xcb_query_tree_reply( | - | ||||||||||||||||||
1120 | c->xcb_connection(), qtCookie, NULL); | - | ||||||||||||||||||
1121 | if (!qtReply)
| 0 | ||||||||||||||||||
1122 | break; never executed: break; | 0 | ||||||||||||||||||
1123 | xcb_window_t root = qtReply->root; | - | ||||||||||||||||||
1124 | xcb_window_t parent = qtReply->parent; | - | ||||||||||||||||||
1125 | free(qtReply); | - | ||||||||||||||||||
1126 | if (window == root)
| 0 | ||||||||||||||||||
1127 | break; never executed: break; | 0 | ||||||||||||||||||
1128 | window = parent; | - | ||||||||||||||||||
1129 | } never executed: end of block | 0 | ||||||||||||||||||
1130 | return target; never executed: return target; | 0 | ||||||||||||||||||
1131 | } | - | ||||||||||||||||||
1132 | - | |||||||||||||||||||
1133 | void QXcbDrag::handleSelectionRequest(const xcb_selection_request_event_t *event) | - | ||||||||||||||||||
1134 | { | - | ||||||||||||||||||
1135 | xcb_selection_notify_event_t notify; | - | ||||||||||||||||||
1136 | notify.response_type = XCB_SELECTION_NOTIFY; | - | ||||||||||||||||||
1137 | notify.requestor = event->requestor; | - | ||||||||||||||||||
1138 | notify.selection = event->selection; | - | ||||||||||||||||||
1139 | notify.target = XCB_NONE; | - | ||||||||||||||||||
1140 | notify.property = XCB_NONE; | - | ||||||||||||||||||
1141 | notify.time = event->time; | - | ||||||||||||||||||
1142 | - | |||||||||||||||||||
1143 | // which transaction do we use? (note: -2 means use current currentDrag()) | - | ||||||||||||||||||
1144 | int at = -1; | - | ||||||||||||||||||
1145 | - | |||||||||||||||||||
1146 | // figure out which data the requestor is really interested in | - | ||||||||||||||||||
1147 | if (currentDrag() && event->time == source_time) {
| 0 | ||||||||||||||||||
1148 | // requestor wants the current drag data | - | ||||||||||||||||||
1149 | at = -2; | - | ||||||||||||||||||
1150 | } else { never executed: end of block | 0 | ||||||||||||||||||
1151 | // if someone has requested data in response to XdndDrop, find the corresponding transaction. the | - | ||||||||||||||||||
1152 | // spec says to call xcb_convert_selection() using the timestamp from the XdndDrop | - | ||||||||||||||||||
1153 | at = findTransactionByTime(event->time); | - | ||||||||||||||||||
1154 | if (at == -1) {
| 0 | ||||||||||||||||||
1155 | // no dice, perhaps the client was nice enough to use the same window id in | - | ||||||||||||||||||
1156 | // xcb_convert_selection() that we sent the XdndDrop event to. | - | ||||||||||||||||||
1157 | at = findTransactionByWindow(event->requestor); | - | ||||||||||||||||||
1158 | } never executed: end of block | 0 | ||||||||||||||||||
1159 | - | |||||||||||||||||||
1160 | if (at == -1) {
| 0 | ||||||||||||||||||
1161 | xcb_window_t target = findXdndAwareParent(connection(), event->requestor); | - | ||||||||||||||||||
1162 | if (target) {
| 0 | ||||||||||||||||||
1163 | if (event->time == XCB_CURRENT_TIME && current_target == target)
| 0 | ||||||||||||||||||
1164 | at = -2; never executed: at = -2; | 0 | ||||||||||||||||||
1165 | else | - | ||||||||||||||||||
1166 | at = findTransactionByWindow(target); never executed: at = findTransactionByWindow(target); | 0 | ||||||||||||||||||
1167 | } | - | ||||||||||||||||||
1168 | } never executed: end of block | 0 | ||||||||||||||||||
1169 | } never executed: end of block | 0 | ||||||||||||||||||
1170 | - | |||||||||||||||||||
1171 | QDrag *transactionDrag = 0; | - | ||||||||||||||||||
1172 | if (at >= 0) {
| 0 | ||||||||||||||||||
1173 | transactionDrag = transactions.at(at).drag; | - | ||||||||||||||||||
1174 | } else if (at == -2) { never executed: end of block
| 0 | ||||||||||||||||||
1175 | transactionDrag = currentDrag(); | - | ||||||||||||||||||
1176 | } never executed: end of block | 0 | ||||||||||||||||||
1177 | - | |||||||||||||||||||
1178 | if (transactionDrag) {
| 0 | ||||||||||||||||||
1179 | xcb_atom_t atomFormat = event->target; | - | ||||||||||||||||||
1180 | int dataFormat = 0; | - | ||||||||||||||||||
1181 | QByteArray data; | - | ||||||||||||||||||
1182 | if (QXcbMime::mimeDataForAtom(connection(), event->target, transactionDrag->mimeData(),
| 0 | ||||||||||||||||||
1183 | &data, &atomFormat, &dataFormat)) {
| 0 | ||||||||||||||||||
1184 | int dataSize = data.size() / (dataFormat / 8); | - | ||||||||||||||||||
1185 | xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, event->requestor, event->property, | - | ||||||||||||||||||
1186 | atomFormat, dataFormat, dataSize, (const void *)data.constData()); | - | ||||||||||||||||||
1187 | notify.property = event->property; | - | ||||||||||||||||||
1188 | notify.target = atomFormat; | - | ||||||||||||||||||
1189 | } never executed: end of block | 0 | ||||||||||||||||||
1190 | } never executed: end of block | 0 | ||||||||||||||||||
1191 | - | |||||||||||||||||||
1192 | xcb_window_t proxy_target = xdndProxy(connection(), event->requestor); | - | ||||||||||||||||||
1193 | if (!proxy_target)
| 0 | ||||||||||||||||||
1194 | proxy_target = event->requestor; never executed: proxy_target = event->requestor; | 0 | ||||||||||||||||||
1195 | - | |||||||||||||||||||
1196 | xcb_send_event(xcb_connection(), false, proxy_target, XCB_EVENT_MASK_NO_EVENT, (const char *)¬ify); | - | ||||||||||||||||||
1197 | } never executed: end of block | 0 | ||||||||||||||||||
1198 | - | |||||||||||||||||||
1199 | - | |||||||||||||||||||
1200 | bool QXcbDrag::dndEnable(QXcbWindow *w, bool on) | - | ||||||||||||||||||
1201 | { | - | ||||||||||||||||||
1202 | DNDDEBUG << "xdndEnable" << w << on; dead code: QMessageLogger(__FILE__, 1202, __PRETTY_FUNCTION__).debug() << "xdndEnable" << w << on; | - | ||||||||||||||||||
1203 | if (on) {
| 0-4107 | ||||||||||||||||||
1204 | QXcbWindow *xdnd_widget = 0; | - | ||||||||||||||||||
1205 | if ((w->window()->type() == Qt::Desktop)) {
| 0-4107 | ||||||||||||||||||
1206 | if (desktop_proxy) // *WE* already have one.
| 0 | ||||||||||||||||||
1207 | return false; never executed: return false; | 0 | ||||||||||||||||||
1208 | - | |||||||||||||||||||
1209 | QXcbConnectionGrabber grabber(connection()); | - | ||||||||||||||||||
1210 | - | |||||||||||||||||||
1211 | // As per Xdnd4, use XdndProxy | - | ||||||||||||||||||
1212 | xcb_window_t proxy_id = xdndProxy(connection(), w->xcb_window()); | - | ||||||||||||||||||
1213 | - | |||||||||||||||||||
1214 | if (!proxy_id) {
| 0 | ||||||||||||||||||
1215 | desktop_proxy = new QWindow; | - | ||||||||||||||||||
1216 | xdnd_widget = static_cast<QXcbWindow *>(desktop_proxy->handle()); | - | ||||||||||||||||||
1217 | proxy_id = xdnd_widget->xcb_window(); | - | ||||||||||||||||||
1218 | xcb_atom_t xdnd_proxy = atom(QXcbAtom::XdndProxy); | - | ||||||||||||||||||
1219 | xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, w->xcb_window(), xdnd_proxy, | - | ||||||||||||||||||
1220 | XCB_ATOM_WINDOW, 32, 1, &proxy_id); | - | ||||||||||||||||||
1221 | xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, proxy_id, xdnd_proxy, | - | ||||||||||||||||||
1222 | XCB_ATOM_WINDOW, 32, 1, &proxy_id); | - | ||||||||||||||||||
1223 | } never executed: end of block | 0 | ||||||||||||||||||
1224 | - | |||||||||||||||||||
1225 | } else { never executed: end of block | 0 | ||||||||||||||||||
1226 | xdnd_widget = w; | - | ||||||||||||||||||
1227 | } executed 4107 times by 125 tests: end of block Executed by:
| 4107 | ||||||||||||||||||
1228 | if (xdnd_widget) {
| 0-4107 | ||||||||||||||||||
1229 | DNDDEBUG << "setting XdndAware for" << xdnd_widget << xdnd_widget->xcb_window(); dead code: QMessageLogger(__FILE__, 1229, __PRETTY_FUNCTION__).debug() << "setting XdndAware for" << xdnd_widget << xdnd_widget->xcb_window(); | - | ||||||||||||||||||
1230 | xcb_atom_t atm = xdnd_version; | - | ||||||||||||||||||
1231 | xcb_change_property(xcb_connection(), XCB_PROP_MODE_REPLACE, xdnd_widget->xcb_window(), | - | ||||||||||||||||||
1232 | atom(QXcbAtom::XdndAware), XCB_ATOM_ATOM, 32, 1, &atm); | - | ||||||||||||||||||
1233 | return true; executed 4107 times by 125 tests: return true; Executed by:
| 4107 | ||||||||||||||||||
1234 | } else { | - | ||||||||||||||||||
1235 | return false; never executed: return false; | 0 | ||||||||||||||||||
1236 | } | - | ||||||||||||||||||
1237 | } else { | - | ||||||||||||||||||
1238 | if ((w->window()->type() == Qt::Desktop)) {
| 0 | ||||||||||||||||||
1239 | xcb_delete_property(xcb_connection(), w->xcb_window(), atom(QXcbAtom::XdndProxy)); | - | ||||||||||||||||||
1240 | delete desktop_proxy; | - | ||||||||||||||||||
1241 | desktop_proxy = 0; | - | ||||||||||||||||||
1242 | } else { never executed: end of block | 0 | ||||||||||||||||||
1243 | DNDDEBUG << "not deleting XDndAware"; dead code: QMessageLogger(__FILE__, 1243, __PRETTY_FUNCTION__).debug() << "not deleting XDndAware"; | - | ||||||||||||||||||
1244 | } never executed: end of block | 0 | ||||||||||||||||||
1245 | return true; never executed: return true; | 0 | ||||||||||||||||||
1246 | } | - | ||||||||||||||||||
1247 | } | - | ||||||||||||||||||
1248 | - | |||||||||||||||||||
1249 | bool QXcbDrag::ownsDragObject() const | - | ||||||||||||||||||
1250 | { | - | ||||||||||||||||||
1251 | return true; never executed: return true; | 0 | ||||||||||||||||||
1252 | } | - | ||||||||||||||||||
1253 | - | |||||||||||||||||||
1254 | QXcbDropData::QXcbDropData(QXcbDrag *d) | - | ||||||||||||||||||
1255 | : QXcbMime(), | - | ||||||||||||||||||
1256 | drag(d) | - | ||||||||||||||||||
1257 | { | - | ||||||||||||||||||
1258 | } executed 135 times by 5 tests: end of block Executed by:
| 135 | ||||||||||||||||||
1259 | - | |||||||||||||||||||
1260 | QXcbDropData::~QXcbDropData() | - | ||||||||||||||||||
1261 | { | - | ||||||||||||||||||
1262 | } | - | ||||||||||||||||||
1263 | - | |||||||||||||||||||
1264 | QVariant QXcbDropData::retrieveData_sys(const QString &mimetype, QVariant::Type requestedType) const | - | ||||||||||||||||||
1265 | { | - | ||||||||||||||||||
1266 | QByteArray mime = mimetype.toLatin1(); | - | ||||||||||||||||||
1267 | QVariant data = xdndObtainData(mime, requestedType); | - | ||||||||||||||||||
1268 | return data; never executed: return data; | 0 | ||||||||||||||||||
1269 | } | - | ||||||||||||||||||
1270 | - | |||||||||||||||||||
1271 | QVariant QXcbDropData::xdndObtainData(const QByteArray &format, QVariant::Type requestedType) const | - | ||||||||||||||||||
1272 | { | - | ||||||||||||||||||
1273 | QByteArray result; | - | ||||||||||||||||||
1274 | - | |||||||||||||||||||
1275 | QXcbConnection *c = drag->connection(); | - | ||||||||||||||||||
1276 | QXcbWindow *xcb_window = c->platformWindowFromId(drag->xdnd_dragsource); | - | ||||||||||||||||||
1277 | if (xcb_window && drag->currentDrag() && xcb_window->window()->type() != Qt::Desktop) {
| 0 | ||||||||||||||||||
1278 | QMimeData *data = drag->currentDrag()->mimeData(); | - | ||||||||||||||||||
1279 | if (data->hasFormat(QLatin1String(format)))
| 0 | ||||||||||||||||||
1280 | result = data->data(QLatin1String(format)); never executed: result = data->data(QLatin1String(format)); | 0 | ||||||||||||||||||
1281 | return result; never executed: return result; | 0 | ||||||||||||||||||
1282 | } | - | ||||||||||||||||||
1283 | - | |||||||||||||||||||
1284 | QVector<xcb_atom_t> atoms = drag->xdnd_types; | - | ||||||||||||||||||
1285 | QByteArray encoding; | - | ||||||||||||||||||
1286 | xcb_atom_t a = mimeAtomForFormat(c, QLatin1String(format), requestedType, atoms, &encoding); | - | ||||||||||||||||||
1287 | if (a == XCB_NONE)
| 0 | ||||||||||||||||||
1288 | return result; never executed: return result; | 0 | ||||||||||||||||||
1289 | - | |||||||||||||||||||
1290 | if (c->clipboard()->getSelectionOwner(drag->atom(QXcbAtom::XdndSelection)) == XCB_NONE)
| 0 | ||||||||||||||||||
1291 | return result; // should never happen? never executed: return result; | 0 | ||||||||||||||||||
1292 | - | |||||||||||||||||||
1293 | xcb_atom_t xdnd_selection = c->atom(QXcbAtom::XdndSelection); | - | ||||||||||||||||||
1294 | result = c->clipboard()->getSelection(xdnd_selection, a, xdnd_selection, drag->targetTime()); | - | ||||||||||||||||||
1295 | - | |||||||||||||||||||
1296 | return mimeConvertToFormat(c, a, result, QLatin1String(format), requestedType, encoding); never executed: return mimeConvertToFormat(c, a, result, QLatin1String(format), requestedType, encoding); | 0 | ||||||||||||||||||
1297 | } | - | ||||||||||||||||||
1298 | - | |||||||||||||||||||
1299 | - | |||||||||||||||||||
1300 | bool QXcbDropData::hasFormat_sys(const QString &format) const | - | ||||||||||||||||||
1301 | { | - | ||||||||||||||||||
1302 | return formats().contains(format); never executed: return formats().contains(format); | 0 | ||||||||||||||||||
1303 | } | - | ||||||||||||||||||
1304 | - | |||||||||||||||||||
1305 | QStringList QXcbDropData::formats_sys() const | - | ||||||||||||||||||
1306 | { | - | ||||||||||||||||||
1307 | QStringList formats; | - | ||||||||||||||||||
1308 | for (int i = 0; i < drag->xdnd_types.size(); ++i) {
| 0 | ||||||||||||||||||
1309 | QString f = mimeAtomToString(drag->connection(), drag->xdnd_types.at(i)); | - | ||||||||||||||||||
1310 | if (!formats.contains(f))
| 0 | ||||||||||||||||||
1311 | formats.append(f); never executed: formats.append(f); | 0 | ||||||||||||||||||
1312 | } never executed: end of block | 0 | ||||||||||||||||||
1313 | return formats; never executed: return formats; | 0 | ||||||||||||||||||
1314 | } | - | ||||||||||||||||||
1315 | - | |||||||||||||||||||
1316 | #endif // QT_NO_DRAGANDDROP | - | ||||||||||||||||||
1317 | - | |||||||||||||||||||
1318 | QT_END_NAMESPACE | - | ||||||||||||||||||
Source code | Switch to Preprocessed file |