QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskGestureRecognizer.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskGestureRecognizer.h"
7#include "QskEvent.h"
8#include "QskQuick.h"
9#include "QskInternalMacros.h"
10
11#include <qcoreapplication.h>
12#include <qquickitem.h>
13#include <qquickwindow.h>
14#include <qvector.h>
15
16#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
17 QSK_QT_PRIVATE_BEGIN
18 #include <private/qquickwindow_p.h>
19 QSK_QT_PRIVATE_END
20#endif
21
22static QMouseEvent* qskClonedMouseEvent( const QMouseEvent* event )
23{
24 QMouseEvent* clonedEvent;
25
26#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
27 clonedEvent = QQuickWindowPrivate::cloneMouseEvent(
28 const_cast< QMouseEvent* >( event ), nullptr );
29#else
30 clonedEvent = event->clone();
31#endif
32 clonedEvent->setAccepted( false );
33
34 return clonedEvent;
35}
36
37class QskGestureRecognizer::PrivateData
38{
39 public:
40 void startTimer( QskGestureRecognizer* recognizer )
41 {
42 if ( timeoutId >= 0 )
43 {
44 // warning
45 }
46
47 if ( timeout <= 0 )
48 {
49 // warning
50 }
51
52 timeoutId = recognizer->startTimer( timeout );
53 }
54
55 void stopTimer( QskGestureRecognizer* recognizer )
56 {
57 if ( timeoutId >= 0 )
58 {
59 recognizer->killTimer( timeoutId );
60 timeoutId = -1;
61 }
62 }
63
64 inline Qt::MouseButtons effectiveMouseButtons() const
65 {
66 if ( buttons != Qt::NoButton )
67 return buttons;
68
69 return watchedItem->acceptedMouseButtons();
70 }
71
72 QPointer< QQuickItem > watchedItem = nullptr;
73 QPointer< QQuickItem > targetItem = nullptr;
74
75 QVector< QMouseEvent* > pendingEvents;
76
77 quint64 timestampStarted = 0;
78 quint64 timestampProcessed = 0;
79
80 int timeoutId = -1;
81 int timeout = -1; // ms
82
83 Qt::MouseButtons buttons = Qt::NoButton;
84
85 unsigned char state = QskGestureRecognizer::Idle;
86 bool rejectOnTimeout = true;
87 bool expired = false;
88};
89
90QskGestureRecognizer::QskGestureRecognizer( QObject* parent )
91 : QObject( parent )
92 , m_data( new PrivateData() )
93{
94}
95
96QskGestureRecognizer::~QskGestureRecognizer()
97{
98 qDeleteAll( m_data->pendingEvents );
99}
100
101void QskGestureRecognizer::setWatchedItem( QQuickItem* item )
102{
103 if ( m_data->watchedItem )
104 {
105 m_data->watchedItem->removeEventFilter( this );
106 reset();
107 }
108
109 m_data->watchedItem = item;
110
111 if ( m_data->watchedItem )
112 m_data->watchedItem->installEventFilter( this );
113
114 /*
115 // doing those here: ???
116 m_data->watchedItem->setFiltersChildMouseEvents();
117 m_data->watchedItem->setAcceptedMouseButtons();
118 */
119}
120
121QQuickItem* QskGestureRecognizer::watchedItem() const
122{
123 return m_data->watchedItem;
124}
125
126void QskGestureRecognizer::setTargetItem( QQuickItem* item )
127{
128 m_data->targetItem = item;
129}
130
131QQuickItem* QskGestureRecognizer::targetItem() const
132{
133 return m_data->targetItem;
134}
135
136void QskGestureRecognizer::setAcceptedMouseButtons( Qt::MouseButtons buttons )
137{
138 m_data->buttons = buttons;
139}
140
141Qt::MouseButtons QskGestureRecognizer::acceptedMouseButtons() const
142{
143 return m_data->buttons;
144}
145
146bool QskGestureRecognizer::isAcceptedPos( const QPointF& pos ) const
147{
148 return m_data->watchedItem && m_data->watchedItem->contains( pos );
149}
150
151void QskGestureRecognizer::setRejectOnTimeout( bool on )
152{
153 m_data->rejectOnTimeout = on;
154}
155
156bool QskGestureRecognizer::rejectOnTimeout() const
157{
158 return m_data->rejectOnTimeout;
159}
160
161void QskGestureRecognizer::setTimeout( int ms )
162{
163 m_data->timeout = ms;
164}
165
166int QskGestureRecognizer::timeout() const
167{
168 return m_data->timeout;
169}
170
171void QskGestureRecognizer::timerEvent( QTimerEvent* event )
172{
173 if ( event->timerId() == m_data->timeoutId )
174 {
175 m_data->stopTimer( this );
176
177 if ( m_data->rejectOnTimeout )
178 {
179 reject();
180 m_data->expired = true;
181 }
182
183 return;
184 }
185
186 Inherited::timerEvent( event );
187}
188
189quint64 QskGestureRecognizer::timestampStarted() const
190{
191 return m_data->timestampStarted;
192}
193
194void QskGestureRecognizer::setState( State state )
195{
196 if ( state != m_data->state )
197 {
198 const auto oldState = static_cast< State >( m_data->state );
199 m_data->state = state;
200
201 Q_EMIT stateChanged( oldState, state );
202 }
203}
204
205QskGestureRecognizer::State QskGestureRecognizer::state() const
206{
207 return static_cast< QskGestureRecognizer::State >( m_data->state );
208}
209
210bool QskGestureRecognizer::eventFilter( QObject* object, QEvent* event)
211{
212 auto& watchedItem = m_data->watchedItem;
213
214 if ( ( object == watchedItem ) &&
215 ( static_cast< int >( event->type() ) == QskEvent::GestureFilter ) )
216 {
217 if ( !watchedItem->isEnabled() || !watchedItem->isVisible()
218 || watchedItem->window() == nullptr )
219 {
220 reset();
221 return false;
222 }
223
224 auto ev = static_cast< QskGestureFilterEvent* >( event );
225 ev->setMaybeGesture( maybeGesture( ev->item(), ev->event() ) );
226
227 return ev->maybeGesture();
228 }
229
230 return false;
231}
232
233bool QskGestureRecognizer::maybeGesture(
234 const QQuickItem* item, const QEvent* event )
235{
236 switch( static_cast< int >( event->type() ) )
237 {
238 case QEvent::UngrabMouse:
239 {
240 if ( m_data->state != Idle )
241 {
242 // someone took our grab away, we have to give up
243 reset();
244 return true;
245 }
246
247 return false;
248 }
249 case QEvent::MouseButtonPress:
250 {
251 if ( state() != Idle )
252 {
253 qWarning() << "QskGestureRecognizer: pressed, while not being idle";
254 return false;
255 }
256
257 const auto mouseEvent = static_cast< const QMouseEvent* >( event );
258
259 if ( mouseEvent->timestamp() > m_data->timestampProcessed )
260 {
261 m_data->timestampProcessed = mouseEvent->timestamp();
262 }
263 else
264 {
265 /*
266 A mouse event might appear several times:
267
268 - we ran into a timeout and reposted the events
269
270 - we rejected because the event sequence does
271 not match the acceptance criterions and reposted
272 the events
273
274 - another gesture recognizer for a child item
275 has reposted the events because of the reasons above
276
277 For most situations we can simply skip processing already
278 processed events with the exception of a timeout. Here we might
279 have to retry without timeout, when none of the items was
280 accepting the events. This specific situation is indicated by
281 item set to null.
282 */
283
284 if ( item || !m_data->expired )
285 return false;
286 }
287
288 return processMouseEvent( item, mouseEvent );
289 }
290 case QEvent::MouseMove:
291 case QEvent::MouseButtonRelease:
292 {
293 if ( state() <= Idle )
294 return false;
295
296 const auto mouseEvent = static_cast< const QMouseEvent* >( event );
297 return processMouseEvent( item, mouseEvent );
298 }
299 }
300
301 return false;
302}
303
304bool QskGestureRecognizer::processMouseEvent(
305 const QQuickItem* item, const QMouseEvent* event )
306{
307 auto& watchedItem = m_data->watchedItem;
308
309 const auto pos = watchedItem->mapFromScene( qskMouseScenePosition( event ) );
310 const auto timestamp = event->timestamp();
311
312 if ( event->type() == QEvent::MouseButtonPress )
313 {
314 if ( !isAcceptedPos( pos ) )
315 return false;
316
317 if ( m_data->state != Idle )
318 {
319 // should not happen, when using the recognizer correctly
320 qWarning() << "QskGestureRecognizer: pressed, while not being idle";
321 abort();
322 return false;
323 }
324
325 if ( !( m_data->effectiveMouseButtons() & event->button() ) )
326 return false;
327
328 /*
329 We try to grab the mouse for watchedItem and indicate, that we want
330 to keep it. Then all mouse events should end up at watchedItem.
331 */
332 if ( !qskGrabMouse( watchedItem ) )
333 return false;
334
335 m_data->timestampStarted = timestamp;
336
337 if ( m_data->timeout != 0 )
338 {
339 /*
340 We need to be able to replay the press in case we later find
341 out, that we don't want to handle the mouse event sequence,
342 */
343
344 m_data->stopTimer( this );
345
346 if ( m_data->timeout > 0 )
347 m_data->startTimer( this );
348
349 setState( Pending );
350 }
351 else
352 {
353 accept();
354 }
355 }
356
357 if ( m_data->state <= Idle )
358 return false;
359
360 if ( m_data->state == Pending )
361 m_data->pendingEvents += qskClonedMouseEvent( event );
362
363 switch( static_cast< int >( event->type() ) )
364 {
365 case QEvent::MouseButtonPress:
366 processPress( pos, timestamp, item == nullptr );
367 break;
368
369 case QEvent::MouseMove:
370 processMove( pos, timestamp );
371 break;
372
373 case QEvent::MouseButtonRelease:
374 {
375 if ( m_data->state == Pending )
376 {
377 reject();
378 }
379 else
380 {
381 processRelease( pos, timestamp );
382 reset();
383 }
384 break;
385 }
386 }
387
388 return true;
389}
390
391void QskGestureRecognizer::processPress(
392 const QPointF& pos, quint64 timestamp, bool isFinal )
393{
394 Q_UNUSED( pos );
395 Q_UNUSED( timestamp );
396 Q_UNUSED( isFinal );
397}
398
399void QskGestureRecognizer::processMove( const QPointF& pos, quint64 timestamp )
400{
401 Q_UNUSED( pos );
402 Q_UNUSED( timestamp );
403}
404
405void QskGestureRecognizer::processRelease( const QPointF& pos, quint64 timestamp )
406{
407 Q_UNUSED( pos );
408 Q_UNUSED( timestamp );
409}
410
411void QskGestureRecognizer::accept()
412{
413 m_data->stopTimer( this );
414
415 qDeleteAll( m_data->pendingEvents );
416 m_data->pendingEvents.clear();
417
418 setState( Accepted );
419}
420
421void QskGestureRecognizer::reject()
422{
423 /*
424 Moving the events to a local buffer, so that we can clear
425 m_data->pendingEvents before replaying them.
426 */
427 const auto events = m_data->pendingEvents;
428 m_data->pendingEvents.clear();
429
430 reset();
431
432 /*
433 Not 100% sure if we should send or post the events
434
435 Posting the events adds an extra round trip but we avoid
436 recursive event handler calls, that might not be expected
437 from the implementation of a control.
438 */
439
440 if ( !events.isEmpty() )
441 {
442 if ( auto watchedItem = m_data->watchedItem )
443 {
444 if ( const auto window = watchedItem->window() )
445 {
446 for ( auto event : events )
447 QCoreApplication::postEvent( window, event );
448 }
449 }
450
451#if 0
452 // when using QCoreApplication::sendEvent above
453 qDeleteAll( events );
454#endif
455 }
456}
457
458void QskGestureRecognizer::abort()
459{
460 reset();
461}
462
463void QskGestureRecognizer::reset()
464{
465 m_data->stopTimer( this );
466 qskUngrabMouse( m_data->watchedItem );
467
468 qDeleteAll( m_data->pendingEvents );
469 m_data->pendingEvents.clear();
470
471 m_data->timestampStarted = 0;
472 m_data->expired = false;
473
474 setState( Idle );
475}
476
477#include "moc_QskGestureRecognizer.cpp"