5#include "private/qgesturemanager_p.h"
6#include "private/qstandardgestures_p.h"
7#include "private/qwidget_p.h"
8#include "private/qgesture_p.h"
9#if QT_CONFIG(graphicsview)
10#include "private/qgraphicsitem_p.h"
11#include "qgraphicsitem.h"
13#include "private/qevent_p.h"
14#include "private/qapplication_p.h"
15#include "private/qwidgetwindow_p.h"
20#include "qohosgesturerecognizer_p.h"
24#include "qmacgesturerecognizer_p.h"
28#include <QtCore/QLoggingCategory>
29#include <QtCore/QVarLengthArray>
35Q_STATIC_LOGGING_CATEGORY(lcGestureManager,
"qt.widgets.gestures")
37#if !defined(Q_OS_MACOS)
38static inline int panTouchPoints()
41 static const char panTouchPointVariable[] =
"QT_PAN_TOUCHPOINTS";
42 if (qEnvironmentVariableIsSet(panTouchPointVariable)) {
44 const int result = qEnvironmentVariableIntValue(panTouchPointVariable, &ok);
45 if (ok && result >= 1)
47 qWarning(
"Ignoring invalid value of %s", panTouchPointVariable);
57QGestureManager::QGestureManager(QObject *parent)
58 : QObject(parent), m_lastCustomGestureId(Qt::CustomGesture)
60 qRegisterMetaType<Qt::GestureState>();
62#if defined(Q_OS_MACOS)
63 registerGestureRecognizer(
new QMacSwipeGestureRecognizer);
64 registerGestureRecognizer(
new QMacPinchGestureRecognizer);
65 registerGestureRecognizer(
new QMacPanGestureRecognizer);
66#elif defined(Q_OS_OHOS)
67 registerGestureRecognizer(
new QPanGestureRecognizer(panTouchPoints()));
68 registerGestureRecognizer(makeQOhosPinchGestureRecognizer().release());
69 registerGestureRecognizer(
new QSwipeGestureRecognizer);
70 registerGestureRecognizer(
new QTapGestureRecognizer);
72 registerGestureRecognizer(
new QPanGestureRecognizer(panTouchPoints()));
73 registerGestureRecognizer(
new QPinchGestureRecognizer);
74 registerGestureRecognizer(
new QSwipeGestureRecognizer);
75 registerGestureRecognizer(
new QTapGestureRecognizer);
77 registerGestureRecognizer(
new QTapAndHoldGestureRecognizer);
80QGestureManager::~QGestureManager()
82 qDeleteAll(m_recognizers);
83 for (
auto it = m_obsoleteGestures.cbegin(), end = m_obsoleteGestures.cend(); it != end; ++it) {
84 qDeleteAll(it.value());
89Qt::GestureType QGestureManager::registerGestureRecognizer(QGestureRecognizer *recognizer)
91 const QScopedPointer<QGesture> dummy(recognizer->create(
nullptr));
92 if (Q_UNLIKELY(!dummy)) {
93 qWarning(
"QGestureManager::registerGestureRecognizer: "
94 "the recognizer fails to create a gesture object, skipping registration.");
95 return Qt::GestureType(0);
97 Qt::GestureType type = dummy->gestureType();
98 if (type == Qt::CustomGesture) {
100 ++m_lastCustomGestureId;
101 type = Qt::GestureType(m_lastCustomGestureId);
103 m_recognizers.insert(type, recognizer);
107void QGestureManager::unregisterGestureRecognizer(Qt::GestureType type)
109 QList<QGestureRecognizer *> list = m_recognizers.values(type);
110 m_recognizers.remove(type);
111 for (
const auto &[g, recognizer] : std::as_const(m_gestureToRecognizer).asKeyValueRange()) {
112 if (list.contains(recognizer)) {
113 m_deletedRecognizers.insert(g, recognizer);
117 for (
const auto &[objectGesture, gestures] : std::as_const(m_objectGestures).asKeyValueRange()) {
118 if (objectGesture.gesture == type) {
119 for (QGesture *g : gestures) {
120 auto it = m_gestureToRecognizer.constFind(g);
121 if (it != m_gestureToRecognizer.cend() && it.value()) {
122 QGestureRecognizer *recognizer = it.value();
123 m_gestureToRecognizer.erase(it);
124 m_obsoleteGestures[recognizer].insert(g);
130 for (QGestureRecognizer *recognizer : std::as_const(list)) {
131 const bool isObsolete = m_obsoleteGestures.contains(recognizer);
132 const bool isDeleted = m_deletedRecognizers.values().contains(recognizer);
134 if (!isObsolete && !isDeleted)
139void QGestureManager::cleanupCachedGestures(QObject *target, Qt::GestureType type)
141 const auto iter = m_objectGestures.find({target, type});
142 if (iter == m_objectGestures.end())
145 const QList<QGesture *> &gestures = iter.value();
146 for (
auto &e : m_obsoleteGestures) {
147 for (QGesture *g : gestures)
150 for (QGesture *g : gestures) {
151 m_deletedRecognizers.remove(g);
152 m_gestureToRecognizer.remove(g);
153 m_maybeGestures.remove(g);
154 m_activeGestures.remove(g);
155 m_gestureOwners.remove(g);
156 m_gestureTargets.remove(g);
157 m_gesturesToDelete.insert(g);
160 m_objectGestures.erase(iter);
164QGesture *QGestureManager::getState(QObject *object, QGestureRecognizer *recognizer, Qt::GestureType type)
169 if (object->isWidgetType()) {
170 if (
static_cast<QWidget *>(object)->d_func()->data.in_destructor)
172 }
else if (QGesture *g = qobject_cast<QGesture *>(object)) {
174#if QT_CONFIG(graphicsview)
176 Q_ASSERT(qobject_cast<QGraphicsObject *>(object));
177 QGraphicsObject *graphicsObject =
static_cast<QGraphicsObject *>(object);
178 if (graphicsObject->QGraphicsItem::d_func()->inDestructor)
184 const auto states = m_objectGestures.value(QGestureManager::ObjectGesture(object, type));
185 for (QGesture *state : states) {
186 if (m_gestureToRecognizer.value(state) == recognizer)
190 Q_ASSERT(recognizer);
191 QGesture *state = recognizer->create(object);
194 state->setParent(
this);
195 if (state->gestureType() == Qt::CustomGesture) {
198 state->d_func()->gestureType = type;
199 if (lcGestureManager().isDebugEnabled())
200 state->setObjectName(QString::number((
int)type));
202 m_objectGestures[QGestureManager::ObjectGesture(object, type)].append(state);
203 m_gestureToRecognizer[state] = recognizer;
204 m_gestureOwners[state] = object;
213 case QEvent::MouseButtonPress:
214 case QEvent::MouseButtonRelease:
215 case QEvent::MouseButtonDblClick:
216 case QEvent::MouseMove:
217 case QEvent::TouchBegin:
218 case QEvent::TouchUpdate:
219 case QEvent::TouchCancel:
220 case QEvent::TouchEnd:
221 case QEvent::TabletEnterProximity:
222 case QEvent::TabletLeaveProximity:
223 case QEvent::TabletMove:
224 case QEvent::TabletPress:
225 case QEvent::TabletRelease:
226 case QEvent::GraphicsSceneMouseDoubleClick:
227 case QEvent::GraphicsSceneMousePress:
228 case QEvent::GraphicsSceneMouseRelease:
229 case QEvent::GraphicsSceneMouseMove:
239bool QGestureManager::filterEventThroughContexts(
const QMultiMap<QObject *,
240 Qt::GestureType> &contexts,
243 QSet<QGesture *> triggeredGestures;
244 QSet<QGesture *> finishedGestures;
245 QSet<QGesture *> newMaybeGestures;
246 QSet<QGesture *> notGestures;
251 bool consumeEventHint =
false;
254 typedef QMultiMap<QObject *, Qt::GestureType>::const_iterator ContextIterator;
255 ContextIterator contextEnd = contexts.end();
256 for (ContextIterator context = contexts.begin(); context != contextEnd; ++context) {
257 Qt::GestureType gestureType = context.value();
258 const QMultiMap<Qt::GestureType, QGestureRecognizer *> &const_recognizers = m_recognizers;
259 QMultiMap<Qt::GestureType, QGestureRecognizer *>::const_iterator
260 typeToRecognizerIterator = const_recognizers.lowerBound(gestureType),
261 typeToRecognizerEnd = const_recognizers.upperBound(gestureType);
262 for (; typeToRecognizerIterator != typeToRecognizerEnd; ++typeToRecognizerIterator) {
263 QGestureRecognizer *recognizer = typeToRecognizerIterator.value();
264 QObject *target = context.key();
265 QGesture *state = getState(target, recognizer, gestureType);
268 QGestureRecognizer::Result recognizerResult = recognizer->recognize(state, target, event);
269 QGestureRecognizer::Result recognizerState = recognizerResult & QGestureRecognizer::ResultState_Mask;
270 QGestureRecognizer::Result resultHint = recognizerResult & QGestureRecognizer::ResultHint_Mask;
271 if (recognizerState == QGestureRecognizer::TriggerGesture) {
272 qCDebug(lcGestureManager) <<
"QGestureManager:Recognizer: gesture triggered: " << state << event;
273 triggeredGestures << state;
274 }
else if (recognizerState == QGestureRecognizer::FinishGesture) {
275 qCDebug(lcGestureManager) <<
"QGestureManager:Recognizer: gesture finished: " << state << event;
276 finishedGestures << state;
277 }
else if (recognizerState == QGestureRecognizer::MayBeGesture) {
278 qCDebug(lcGestureManager) <<
"QGestureManager:Recognizer: maybe gesture: " << state << event;
279 newMaybeGestures << state;
280 }
else if (recognizerState == QGestureRecognizer::CancelGesture) {
281 qCDebug(lcGestureManager) <<
"QGestureManager:Recognizer: not gesture: " << state << event;
282 notGestures << state;
283 }
else if (recognizerState == QGestureRecognizer::Ignore) {
284 if (logIgnoredEvent(event->type()))
285 qCDebug(lcGestureManager) <<
"QGestureManager:Recognizer: ignored the event: " << state << event;
287 if (logIgnoredEvent(event->type())) {
288 qCDebug(lcGestureManager) <<
"QGestureManager:Recognizer: hm, lets assume the recognizer"
289 <<
"ignored the event: " << state << event;
292 if (resultHint & QGestureRecognizer::ConsumeEventHint) {
293 qCDebug(lcGestureManager) <<
"QGestureManager: we were asked to consume the event: "
295 consumeEventHint =
true;
299 if (!triggeredGestures.isEmpty() || !finishedGestures.isEmpty()
300 || !newMaybeGestures.isEmpty() || !notGestures.isEmpty()) {
301 const QSet<QGesture *> startedGestures = triggeredGestures - m_activeGestures;
302 triggeredGestures &= m_activeGestures;
305 const QSet<QGesture *> activeToMaybeGestures = m_activeGestures & newMaybeGestures;
308 QSet<QGesture *> maybeToCanceledGestures = m_maybeGestures & notGestures;
312 const QSet<QGesture *> canceledGestures = m_activeGestures & notGestures;
315 m_maybeGestures += newMaybeGestures;
318 QSet<QGesture *> notMaybeGestures = (startedGestures | triggeredGestures
319 | finishedGestures | canceledGestures
321 m_maybeGestures -= notMaybeGestures;
323 Q_ASSERT((startedGestures & finishedGestures).isEmpty());
324 Q_ASSERT((startedGestures & newMaybeGestures).isEmpty());
325 Q_ASSERT((startedGestures & canceledGestures).isEmpty());
326 Q_ASSERT((finishedGestures & newMaybeGestures).isEmpty());
327 Q_ASSERT((finishedGestures & canceledGestures).isEmpty());
328 Q_ASSERT((canceledGestures & newMaybeGestures).isEmpty());
330 const QSet<QGesture *> notStarted = finishedGestures - m_activeGestures;
331 if (!notStarted.isEmpty()) {
334 for (QGesture *gesture : notStarted)
335 gesture->d_func()->state = Qt::GestureStarted;
336 QSet<QGesture *> undeliveredGestures;
337 deliverEvents(notStarted, &undeliveredGestures);
338 finishedGestures -= undeliveredGestures;
341 m_activeGestures += startedGestures;
343 Q_ASSERT((m_activeGestures & triggeredGestures).size() == triggeredGestures.size());
344 m_activeGestures -= finishedGestures;
345 m_activeGestures -= activeToMaybeGestures;
346 m_activeGestures -= canceledGestures;
349 for (QGesture *gesture : startedGestures)
350 gesture->d_func()->state = Qt::GestureStarted;
351 for (QGesture *gesture : std::as_const(triggeredGestures))
352 gesture->d_func()->state = Qt::GestureUpdated;
353 for (QGesture *gesture : std::as_const(finishedGestures))
354 gesture->d_func()->state = Qt::GestureFinished;
355 for (QGesture *gesture : canceledGestures)
356 gesture->d_func()->state = Qt::GestureCanceled;
357 for (QGesture *gesture : activeToMaybeGestures)
358 gesture->d_func()->state = Qt::GestureFinished;
360 if (!m_activeGestures.isEmpty() || !m_maybeGestures.isEmpty() ||
361 !startedGestures.isEmpty() || !triggeredGestures.isEmpty() ||
362 !finishedGestures.isEmpty() || !canceledGestures.isEmpty()) {
363 qCDebug(lcGestureManager) <<
"QGestureManager::filterEventThroughContexts:"
364 <<
"\n\tactiveGestures:" << m_activeGestures
365 <<
"\n\tmaybeGestures:" << m_maybeGestures
366 <<
"\n\tstarted:" << startedGestures
367 <<
"\n\ttriggered:" << triggeredGestures
368 <<
"\n\tfinished:" << finishedGestures
369 <<
"\n\tcanceled:" << canceledGestures
370 <<
"\n\tmaybe-canceled:" << maybeToCanceledGestures;
373 QSet<QGesture *> undeliveredGestures;
374 deliverEvents(startedGestures+triggeredGestures+finishedGestures+canceledGestures,
375 &undeliveredGestures);
377 for (QGesture *g : startedGestures) {
378 if (undeliveredGestures.contains(g))
380 if (g->gestureCancelPolicy() == QGesture::CancelAllInContext) {
381 qCDebug(lcGestureManager) <<
"lets try to cancel some";
383 cancelGesturesForChildren(g);
387 m_activeGestures -= undeliveredGestures;
390 const QSet<QGesture *> endedGestures =
391 finishedGestures + canceledGestures + undeliveredGestures + maybeToCanceledGestures;
392 for (QGesture *gesture : endedGestures) {
394 m_gestureTargets.remove(gesture);
398 qDeleteAll(m_gesturesToDelete);
399 m_gesturesToDelete.clear();
401 return consumeEventHint;
405void QGestureManager::cancelGesturesForChildren(QGesture *original)
408 QWidget *originatingWidget = m_gestureTargets.value(original);
409 Q_ASSERT(originatingWidget);
410 if (!originatingWidget)
417 QSet<QGesture*> cancelledGestures;
418 QSet<QGesture*>::Iterator iter = m_activeGestures.begin();
419 while (iter != m_activeGestures.end()) {
420 QWidget *widget = m_gestureTargets.value(*iter);
422 if (widget != originatingWidget && originatingWidget->isAncestorOf(widget)) {
423 qCDebug(lcGestureManager) <<
" found a gesture to cancel" << (*iter);
424 (*iter)->d_func()->state = Qt::GestureCanceled;
425 cancelledGestures << *iter;
426 iter = m_activeGestures.erase(iter);
435 QSet<QGesture *> almostCanceledGestures = cancelledGestures;
436 while (!almostCanceledGestures.isEmpty()) {
437 QWidget *target =
nullptr;
438 QSet<QGesture*> gestures;
439 iter = almostCanceledGestures.begin();
441 while (iter != almostCanceledGestures.end()) {
442 QWidget *widget = m_gestureTargets.value(*iter);
443 if (target ==
nullptr)
445 if (target == widget) {
447 iter = almostCanceledGestures.erase(iter);
454 QSet<QGesture*> undeliveredGestures;
455 deliverEvents(gestures, &undeliveredGestures);
458 for (iter = cancelledGestures.begin(); iter != cancelledGestures.end(); ++iter)
462void QGestureManager::cleanupGesturesForRemovedRecognizer(QGesture *gesture)
464 QGestureRecognizer *recognizer = m_deletedRecognizers.value(gesture);
467 m_deletedRecognizers.remove(gesture);
468 if (m_deletedRecognizers.keys(recognizer).isEmpty()) {
470 qDeleteAll(m_obsoleteGestures.value(recognizer));
471 m_obsoleteGestures.remove(recognizer);
477bool QGestureManager::filterEvent(QWidget *receiver, QEvent *event)
479 QVarLengthArray<Qt::GestureType, 16> types;
480 QMultiMap<QObject *, Qt::GestureType> contexts;
481 QWidget *w = receiver;
482 typedef QMap<Qt::GestureType, Qt::GestureFlags>::const_iterator ContextIterator;
483 if (!w->d_func()->gestureContext.isEmpty()) {
484 for(ContextIterator it = w->d_func()->gestureContext.constBegin(),
485 e = w->d_func()->gestureContext.constEnd(); it != e; ++it) {
486 types.push_back(it.key());
487 contexts.insert(w, it.key());
491 w = w->isWindow() ?
nullptr : w->parentWidget();
494 for (ContextIterator it = w->d_func()->gestureContext.constBegin(),
495 e = w->d_func()->gestureContext.constEnd(); it != e; ++it) {
496 if (!(it.value() & Qt::DontStartGestureOnChildren)) {
497 if (!types.contains(it.key())) {
498 types.push_back(it.key());
499 contexts.insert(w, it.key());
505 w = w->parentWidget();
507 return contexts.isEmpty() ?
false : filterEventThroughContexts(contexts, event);
510#if QT_CONFIG(graphicsview)
511bool QGestureManager::filterEvent(QGraphicsObject *receiver, QEvent *event)
513 QVarLengthArray<Qt::GestureType, 16> types;
514 QMultiMap<QObject *, Qt::GestureType> contexts;
515 QGraphicsObject *item = receiver;
516 if (!item->QGraphicsItem::d_func()->gestureContext.isEmpty()) {
517 typedef QMap<Qt::GestureType, Qt::GestureFlags>::const_iterator ContextIterator;
518 for(ContextIterator it = item->QGraphicsItem::d_func()->gestureContext.constBegin(),
519 e = item->QGraphicsItem::d_func()->gestureContext.constEnd(); it != e; ++it) {
520 types.push_back(it.key());
521 contexts.insert(item, it.key());
525 item = item->parentObject();
528 typedef QMap<Qt::GestureType, Qt::GestureFlags>::const_iterator ContextIterator;
529 for (ContextIterator it = item->QGraphicsItem::d_func()->gestureContext.constBegin(),
530 e = item->QGraphicsItem::d_func()->gestureContext.constEnd(); it != e; ++it) {
531 if (!(it.value() & Qt::DontStartGestureOnChildren)) {
532 if (!types.contains(it.key())) {
533 types.push_back(it.key());
534 contexts.insert(item, it.key());
538 item = item->parentObject();
540 return contexts.isEmpty() ?
false : filterEventThroughContexts(contexts, event);
544bool QGestureManager::filterEvent(QObject *receiver, QEvent *event)
548 QWidgetWindow *widgetWindow = qobject_cast<QWidgetWindow *>(receiver);
550 if (widgetWindow && widgetWindow->widget())
551 return filterEvent(widgetWindow->widget(), event);
553 QGesture *state = qobject_cast<QGesture *>(receiver);
554 if (!state || !m_gestureToRecognizer.contains(state))
556 QMultiMap<QObject *, Qt::GestureType> contexts;
557 contexts.insert(state, state->gestureType());
558 return filterEventThroughContexts(contexts, event);
561void QGestureManager::getGestureTargets(
const QSet<QGesture*> &gestures,
562 QHash<QWidget *, QList<QGesture *> > *conflicts,
563 QHash<QWidget *, QList<QGesture *> > *normal)
565 typedef QHash<Qt::GestureType, QHash<QWidget *, QGesture *> > GestureByTypes;
566 GestureByTypes gestureByTypes;
569 for (QGesture *gesture : gestures) {
570 QWidget *receiver = m_gestureTargets.value(gesture,
nullptr);
573 gestureByTypes[gesture->gestureType()].insert(receiver, gesture);
577 for (GestureByTypes::const_iterator git = gestureByTypes.cbegin(), gend = gestureByTypes.cend(); git != gend; ++git) {
578 const QHash<QWidget *, QGesture *> &gestures = git.value();
579 for (QHash<QWidget *, QGesture *>::const_iterator wit = gestures.cbegin(), wend = gestures.cend(); wit != wend; ++wit) {
580 QWidget *widget = wit.key();
581 QWidget *w = widget->parentWidget();
583 QMap<Qt::GestureType, Qt::GestureFlags>::const_iterator it
584 = w->d_func()->gestureContext.constFind(git.key());
585 if (it != w->d_func()->gestureContext.constEnd()) {
587 if (!(it.value() & Qt::DontStartGestureOnChildren) && w != widget) {
589 (*conflicts)[widget].append(wit.value());
597 w = w->parentWidget();
600 (*normal)[widget].append(wit.value());
605void QGestureManager::deliverEvents(
const QSet<QGesture *> &gestures,
606 QSet<QGesture *> *undeliveredGestures)
608 if (gestures.isEmpty())
611 typedef QHash<QWidget *, QList<QGesture *> > GesturesPerWidget;
612 GesturesPerWidget conflictedGestures;
613 GesturesPerWidget normalStartedGestures;
615 QSet<QGesture *> startedGestures;
617 for (QSet<QGesture *>::const_iterator it = gestures.begin(),
618 e = gestures.end(); it != e; ++it) {
619 QGesture *gesture = *it;
620 QWidget *target = m_gestureTargets.value(gesture,
nullptr);
623 Q_ASSERT(gesture->state() == Qt::GestureStarted);
624 if (gesture->hasHotSpot()) {
626 const QPointF pt = gesture->hotSpot();
627 qCDebug(lcGestureManager) <<
__FUNCTION__ << gesture
628 <<
"doesn't have a target yet."
629 <<
"Trying hotspot at" << pt;
630 if (QWidget *topLevel = QApplication::topLevelAt(pt.toPoint())) {
631 QWidget *child = topLevel->childAt(topLevel->mapFromGlobal(pt));
632 target = child ? child : topLevel;
636 QObject *context = m_gestureOwners.value(gesture, 0);
637 if (context->isWidgetType())
638 target =
static_cast<QWidget *>(context);
639 qCDebug(lcGestureManager) <<
__FUNCTION__ << gesture
640 <<
"doesn't have a target yet."
641 <<
"Trying context" << context;
644 m_gestureTargets.insert(gesture, target);
647 Qt::GestureType gestureType = gesture->gestureType();
648 Q_ASSERT(gestureType != Qt::CustomGesture);
649 Q_UNUSED(gestureType);
651 if (Q_UNLIKELY(!target)) {
652 qCDebug(lcGestureManager) <<
__FUNCTION__ <<
"could not find the target for gesture"
653 << gesture->gestureType();
654 qWarning(
"QGestureManager::deliverEvents: could not find the target for gesture");
655 undeliveredGestures->insert(gesture);
657 if (gesture->state() == Qt::GestureStarted) {
658 startedGestures.insert(gesture);
660 normalStartedGestures[target].append(gesture);
665 getGestureTargets(startedGestures, &conflictedGestures, &normalStartedGestures);
666 qCDebug(lcGestureManager) <<
__FUNCTION__
667 <<
"\nstarted: " << startedGestures
668 <<
"\nconflicted: " << conflictedGestures
669 <<
"\nnormal: " << normalStartedGestures
673 for (GesturesPerWidget::const_iterator it = conflictedGestures.constBegin(),
674 e = conflictedGestures.constEnd(); it != e; ++it) {
675 QWidget *receiver = it.key();
676 const QList<QGesture *> &gestures = it.value();
677 qCDebug(lcGestureManager) <<
__FUNCTION__ <<
"sending GestureOverride to"
679 <<
"gestures:" << gestures;
680 QGestureEvent event(gestures);
681 event.t = QEvent::GestureOverride;
684 for (QGesture *g : gestures)
685 event.setAccepted(g,
false);
687 QCoreApplication::sendEvent(receiver, &event);
688 bool eventAccepted = event.isAccepted();
689 const auto eventGestures = event.gestures();
690 for (QGesture *gesture : eventGestures) {
691 if (eventAccepted || event.isAccepted(gesture)) {
692 QWidget *w = event.m_targetWidgets.value(gesture->gestureType(), 0);
694 qCDebug(lcGestureManager) <<
"override event: gesture was accepted:" << gesture << w;
695 QList<QGesture *> &gestures = normalStartedGestures[w];
696 gestures.append(gesture);
698 m_gestureTargets[gesture] = w;
700 qCDebug(lcGestureManager) <<
"override event: gesture wasn't accepted. putting back:" << gesture;
701 QList<QGesture *> &gestures = normalStartedGestures[receiver];
702 gestures.append(gesture);
708 for (GesturesPerWidget::const_iterator it = normalStartedGestures.constBegin(),
709 e = normalStartedGestures.constEnd(); it != e; ++it) {
710 if (!it.value().isEmpty()) {
711 qCDebug(lcGestureManager) <<
__FUNCTION__ <<
"sending to" << it.key()
712 <<
"gestures:" << it.value();
713 QGestureEvent event(it.value());
714 QCoreApplication::sendEvent(it.key(), &event);
715 bool eventAccepted = event.isAccepted();
716 const auto eventGestures = event.gestures();
717 for (QGesture *gesture : eventGestures) {
718 if (gesture->state() == Qt::GestureStarted &&
719 (eventAccepted || event.isAccepted(gesture))) {
720 QWidget *w = event.m_targetWidgets.value(gesture->gestureType(), 0);
722 qCDebug(lcGestureManager) <<
"started gesture was delivered and accepted by" << w;
723 m_gestureTargets[gesture] = w;
730void QGestureManager::recycle(QGesture *gesture)
732 QGestureRecognizer *recognizer = m_gestureToRecognizer.value(gesture, 0);
734 gesture->setGestureCancelPolicy(QGesture::CancelNone);
735 recognizer->reset(gesture);
736 m_activeGestures.remove(gesture);
738 cleanupGesturesForRemovedRecognizer(gesture);
742bool QGestureManager::gesturePending(QObject *o)
744 const QGestureManager *gm = QGestureManager::instance(DontForceCreation);
745 return gm && gm->m_gestureOwners.key(o);
752#include "moc_qgesturemanager_p.cpp"
Combined button and popup list for selecting options.
static bool logIgnoredEvent(QEvent::Type t)