129 lines
4.0 KiB
C++
Raw Normal View History

2024-06-29 11:52:32 +06:00
/*
Gwenview: an image viewer
Copyright 2019 Steffen Hartleib <steffenhartleib@t-online.de>
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Cambridge, MA 02110-1301, USA.
*/
// Self
#include "tapholdandmoving.h"
// Qt
#include <QGraphicsWidget>
#include <QTouchEvent>
// KF
// Local
#include "gwenview_lib_debug.h"
#include "lib/touch/touch_helper.h"
namespace Gwenview
{
struct TapHoldAndMovingRecognizerPrivate {
TapHoldAndMovingRecognizer *q = nullptr;
bool mTargetIsGrapicsWidget = false;
qint64 mTouchBeginnTimestamp;
bool mTouchPointStationary;
bool mGestureTriggered;
Qt::GestureState mLastGestureState = Qt::NoGesture;
};
TapHoldAndMovingRecognizer::TapHoldAndMovingRecognizer()
: QGestureRecognizer()
, d(new TapHoldAndMovingRecognizerPrivate)
{
d->q = this;
}
TapHoldAndMovingRecognizer::~TapHoldAndMovingRecognizer()
{
delete d;
}
QGesture *TapHoldAndMovingRecognizer::create(QObject *)
{
return static_cast<QGesture *>(new TapHoldAndMoving());
}
QGestureRecognizer::Result TapHoldAndMovingRecognizer::recognize(QGesture *state, QObject *watched, QEvent *event)
{
// Because of a bug in Qt in a gesture event in a graphicsview, all gestures are trigger twice
// https://bugreports.qt.io/browse/QTBUG-13103
if (qobject_cast<QGraphicsWidget *>(watched))
d->mTargetIsGrapicsWidget = true;
if (d->mTargetIsGrapicsWidget && watched->isWidgetType())
return Ignore;
switch (event->type()) {
case QEvent::TouchBegin: {
auto touchEvent = static_cast<QTouchEvent *>(event);
d->mTouchBeginnTimestamp = touchEvent->timestamp();
d->mGestureTriggered = false;
d->mTouchPointStationary = true;
state->setHotSpot(touchEvent->touchPoints().first().screenPos());
d->mLastGestureState = Qt::NoGesture;
return MayBeGesture;
}
case QEvent::TouchUpdate: {
auto touchEvent = static_cast<QTouchEvent *>(event);
const qint64 now = touchEvent->timestamp();
const QPoint pos = touchEvent->touchPoints().first().pos().toPoint();
state->setHotSpot(touchEvent->touchPoints().first().screenPos());
if (touchEvent->touchPoints().size() >> 1) {
d->mGestureTriggered = false;
d->mLastGestureState = Qt::GestureCanceled;
return CancelGesture;
}
if (touchEvent->touchPoints().size() == 1 && d->mLastGestureState != Qt::GestureCanceled) {
if (!d->mGestureTriggered && d->mTouchPointStationary && now - d->mTouchBeginnTimestamp >= Touch_Helper::Touch::durationForTapHold) {
d->mGestureTriggered = true;
}
}
d->mTouchPointStationary = Touch_Helper::touchStationary(event);
if (d->mGestureTriggered && d->mLastGestureState != Qt::GestureCanceled) {
state->setProperty("pos", pos);
d->mLastGestureState = Qt::GestureStarted;
return TriggerGesture;
}
break;
}
case QEvent::TouchEnd: {
auto touchEvent = static_cast<QTouchEvent *>(event);
state->setHotSpot(touchEvent->touchPoints().first().screenPos());
if (d->mGestureTriggered) {
d->mLastGestureState = Qt::GestureFinished;
return FinishGesture;
}
break;
}
default:
return Ignore;
}
return Ignore;
}
TapHoldAndMoving::TapHoldAndMoving(QObject *parent)
: QGesture(parent)
{
}
} // namespace