2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2017-03-18 10:00:30 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2017 Martin Gräßlin <mgraesslin@kde.org>
|
2017-03-18 10:00:30 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2017-03-18 10:00:30 +00:00
|
|
|
#include "gestures.h"
|
|
|
|
|
2022-03-23 10:13:38 +00:00
|
|
|
#include <QDebug>
|
2017-03-19 10:40:03 +00:00
|
|
|
#include <QRect>
|
2017-03-29 12:45:44 +00:00
|
|
|
#include <cmath>
|
2022-03-23 10:13:38 +00:00
|
|
|
#include <functional>
|
2017-03-18 10:00:30 +00:00
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
Gesture::Gesture(QObject *parent)
|
|
|
|
: QObject(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Gesture::~Gesture() = default;
|
|
|
|
|
|
|
|
SwipeGesture::SwipeGesture(QObject *parent)
|
|
|
|
: Gesture(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
SwipeGesture::~SwipeGesture() = default;
|
|
|
|
|
2017-03-19 10:40:03 +00:00
|
|
|
void SwipeGesture::setStartGeometry(const QRect &geometry)
|
|
|
|
{
|
|
|
|
setMinimumX(geometry.x());
|
|
|
|
setMinimumY(geometry.y());
|
|
|
|
setMaximumX(geometry.x() + geometry.width());
|
|
|
|
setMaximumY(geometry.y() + geometry.height());
|
2020-03-23 19:26:21 +00:00
|
|
|
|
|
|
|
Q_ASSERT(m_maximumX >= m_minimumX);
|
|
|
|
Q_ASSERT(m_maximumY >= m_minimumY);
|
2017-03-19 10:40:03 +00:00
|
|
|
}
|
|
|
|
|
2022-10-25 08:18:45 +00:00
|
|
|
qreal SwipeGesture::deltaToProgress(const QPointF &delta) const
|
2017-03-19 10:40:03 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
if (!m_minimumDeltaRelevant || m_minimumDelta.isNull()) {
|
2017-03-19 10:40:03 +00:00
|
|
|
return 1.0;
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
|
2022-09-11 10:14:11 +00:00
|
|
|
switch (m_direction) {
|
2023-01-25 16:18:06 +00:00
|
|
|
case SwipeDirection::Up:
|
|
|
|
case SwipeDirection::Down:
|
2022-10-25 08:18:45 +00:00
|
|
|
return std::min(std::abs(delta.y()) / std::abs(m_minimumDelta.y()), 1.0);
|
2023-01-25 16:18:06 +00:00
|
|
|
case SwipeDirection::Left:
|
|
|
|
case SwipeDirection::Right:
|
2022-10-25 08:18:45 +00:00
|
|
|
return std::min(std::abs(delta.x()) / std::abs(m_minimumDelta.x()), 1.0);
|
2022-09-11 10:14:11 +00:00
|
|
|
default:
|
|
|
|
Q_UNREACHABLE();
|
2017-03-19 10:40:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-25 08:18:45 +00:00
|
|
|
bool SwipeGesture::minimumDeltaReached(const QPointF &delta) const
|
2017-03-19 10:40:03 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
return deltaToProgress(delta) >= 1.0;
|
2017-03-19 10:40:03 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
PinchGesture::PinchGesture(QObject *parent)
|
|
|
|
: Gesture(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PinchGesture::~PinchGesture() = default;
|
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
qreal PinchGesture::scaleDeltaToProgress(const qreal &scaleDelta) const
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
return std::clamp(std::abs(scaleDelta - 1) / minimumScaleDelta(), 0.0, 1.0);
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
bool PinchGesture::minimumScaleDeltaReached(const qreal &scaleDelta) const
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
return scaleDeltaToProgress(scaleDelta) >= 1.0;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 10:00:30 +00:00
|
|
|
GestureRecognizer::GestureRecognizer(QObject *parent)
|
|
|
|
: QObject(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
GestureRecognizer::~GestureRecognizer() = default;
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
void GestureRecognizer::registerSwipeGesture(KWin::SwipeGesture *gesture)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2022-02-02 21:51:39 +00:00
|
|
|
Q_ASSERT(!m_swipeGestures.contains(gesture));
|
|
|
|
auto connection = connect(gesture, &QObject::destroyed, this, std::bind(&GestureRecognizer::unregisterSwipeGesture, this, gesture));
|
2017-03-18 10:00:30 +00:00
|
|
|
m_destroyConnections.insert(gesture, connection);
|
2022-02-02 21:51:39 +00:00
|
|
|
m_swipeGestures << gesture;
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
void GestureRecognizer::unregisterSwipeGesture(KWin::SwipeGesture *gesture)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
|
|
|
auto it = m_destroyConnections.find(gesture);
|
|
|
|
if (it != m_destroyConnections.end()) {
|
|
|
|
disconnect(it.value());
|
|
|
|
m_destroyConnections.erase(it);
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
m_swipeGestures.removeAll(gesture);
|
2017-03-18 10:00:30 +00:00
|
|
|
if (m_activeSwipeGestures.removeOne(gesture)) {
|
2021-06-08 07:02:14 +00:00
|
|
|
Q_EMIT gesture->cancelled();
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
void GestureRecognizer::registerPinchGesture(KWin::PinchGesture *gesture)
|
|
|
|
{
|
|
|
|
Q_ASSERT(!m_pinchGestures.contains(gesture));
|
|
|
|
auto connection = connect(gesture, &QObject::destroyed, this, std::bind(&GestureRecognizer::unregisterPinchGesture, this, gesture));
|
|
|
|
m_destroyConnections.insert(gesture, connection);
|
|
|
|
m_pinchGestures << gesture;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GestureRecognizer::unregisterPinchGesture(KWin::PinchGesture *gesture)
|
|
|
|
{
|
|
|
|
auto it = m_destroyConnections.find(gesture);
|
|
|
|
if (it != m_destroyConnections.end()) {
|
|
|
|
disconnect(it.value());
|
|
|
|
m_destroyConnections.erase(it);
|
|
|
|
}
|
|
|
|
m_pinchGestures.removeAll(gesture);
|
|
|
|
if (m_activePinchGestures.removeOne(gesture)) {
|
|
|
|
Q_EMIT gesture->cancelled();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-19 10:40:03 +00:00
|
|
|
int GestureRecognizer::startSwipeGesture(uint fingerCount, const QPointF &startPos, StartPositionBehavior startPosBehavior)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2022-02-02 21:51:39 +00:00
|
|
|
m_currentFingerCount = fingerCount;
|
|
|
|
if (!m_activeSwipeGestures.isEmpty() || !m_activePinchGestures.isEmpty()) {
|
|
|
|
return 0;
|
|
|
|
}
|
2017-03-19 10:40:03 +00:00
|
|
|
int count = 0;
|
2022-10-31 19:02:23 +00:00
|
|
|
for (SwipeGesture *gesture : std::as_const(m_swipeGestures)) {
|
2022-09-11 10:14:11 +00:00
|
|
|
if (gesture->minimumFingerCountIsRelevant()) {
|
|
|
|
if (gesture->minimumFingerCount() > fingerCount) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (gesture->maximumFingerCountIsRelevant()) {
|
|
|
|
if (gesture->maximumFingerCount() < fingerCount) {
|
|
|
|
continue;
|
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
2017-03-19 10:40:03 +00:00
|
|
|
if (startPosBehavior == StartPositionBehavior::Relevant) {
|
2022-02-02 21:51:39 +00:00
|
|
|
if (gesture->minimumXIsRelevant()) {
|
|
|
|
if (gesture->minimumX() > startPos.x()) {
|
2017-03-19 10:40:03 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
if (gesture->maximumXIsRelevant()) {
|
|
|
|
if (gesture->maximumX() < startPos.x()) {
|
2017-03-19 10:40:03 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
if (gesture->minimumYIsRelevant()) {
|
|
|
|
if (gesture->minimumY() > startPos.y()) {
|
2017-03-19 10:40:03 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
if (gesture->maximumYIsRelevant()) {
|
|
|
|
if (gesture->maximumY() < startPos.y()) {
|
2017-03-19 10:40:03 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
|
|
|
|
// Only add gestures who's direction aligns with current swipe axis
|
2022-09-11 10:14:11 +00:00
|
|
|
switch (gesture->direction()) {
|
2023-01-25 16:18:06 +00:00
|
|
|
case SwipeDirection::Up:
|
|
|
|
case SwipeDirection::Down:
|
2022-09-11 10:14:02 +00:00
|
|
|
if (m_currentSwipeAxis == Axis::Horizontal) {
|
2022-02-02 21:51:39 +00:00
|
|
|
continue;
|
|
|
|
}
|
2022-09-11 10:14:11 +00:00
|
|
|
break;
|
2023-01-25 16:18:06 +00:00
|
|
|
case SwipeDirection::Left:
|
|
|
|
case SwipeDirection::Right:
|
2022-09-11 10:14:02 +00:00
|
|
|
if (m_currentSwipeAxis == Axis::Vertical) {
|
2022-02-02 21:51:39 +00:00
|
|
|
continue;
|
|
|
|
}
|
2022-09-11 10:14:11 +00:00
|
|
|
break;
|
2023-01-25 16:18:06 +00:00
|
|
|
case SwipeDirection::Invalid:
|
|
|
|
Q_UNREACHABLE();
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m_activeSwipeGestures << gesture;
|
2017-03-19 10:40:03 +00:00
|
|
|
count++;
|
2022-02-02 21:51:39 +00:00
|
|
|
Q_EMIT gesture->started();
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
2017-03-19 10:40:03 +00:00
|
|
|
return count;
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2022-10-25 08:18:45 +00:00
|
|
|
void GestureRecognizer::updateSwipeGesture(const QPointF &delta)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2021-06-25 23:00:56 +00:00
|
|
|
m_currentDelta += delta;
|
2022-02-02 21:51:39 +00:00
|
|
|
|
2023-01-25 16:18:06 +00:00
|
|
|
SwipeDirection direction; // Overall direction
|
2022-04-02 01:53:16 +00:00
|
|
|
Axis swipeAxis;
|
2022-02-02 21:51:39 +00:00
|
|
|
|
|
|
|
// Pick an axis for gestures so horizontal ones don't change to vertical ones without lifting fingers
|
|
|
|
if (m_currentSwipeAxis == Axis::None) {
|
2022-10-25 08:18:45 +00:00
|
|
|
if (std::abs(m_currentDelta.x()) >= std::abs(m_currentDelta.y())) {
|
2022-04-02 01:53:16 +00:00
|
|
|
swipeAxis = Axis::Horizontal;
|
2023-01-25 16:18:06 +00:00
|
|
|
direction = m_currentDelta.x() < 0 ? SwipeDirection::Left : SwipeDirection::Right;
|
2022-04-02 01:53:16 +00:00
|
|
|
} else {
|
|
|
|
swipeAxis = Axis::Vertical;
|
2023-01-25 16:18:06 +00:00
|
|
|
direction = m_currentDelta.y() < 0 ? SwipeDirection::Up : SwipeDirection::Down;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
2022-10-25 08:18:45 +00:00
|
|
|
if (std::abs(m_currentDelta.x()) >= 5 || std::abs(m_currentDelta.y()) >= 5) {
|
2022-04-02 01:53:16 +00:00
|
|
|
// only lock in a direction if the delta is big enough
|
|
|
|
// to prevent accidentally choosing the wrong direction
|
|
|
|
m_currentSwipeAxis = swipeAxis;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
2022-04-02 01:53:16 +00:00
|
|
|
} else {
|
|
|
|
swipeAxis = m_currentSwipeAxis;
|
2020-08-04 13:52:16 +00:00
|
|
|
}
|
2021-06-25 23:00:56 +00:00
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
// Find the current swipe direction
|
2022-04-02 01:53:16 +00:00
|
|
|
switch (swipeAxis) {
|
2022-02-02 21:51:39 +00:00
|
|
|
case Axis::Vertical:
|
2023-01-25 16:18:06 +00:00
|
|
|
direction = m_currentDelta.y() < 0 ? SwipeDirection::Up : SwipeDirection::Down;
|
2022-02-02 21:51:39 +00:00
|
|
|
break;
|
|
|
|
case Axis::Horizontal:
|
2023-01-25 16:18:06 +00:00
|
|
|
direction = m_currentDelta.x() < 0 ? SwipeDirection::Left : SwipeDirection::Right;
|
2022-02-02 21:51:39 +00:00
|
|
|
break;
|
|
|
|
default:
|
2022-04-02 01:53:16 +00:00
|
|
|
Q_UNREACHABLE();
|
2022-03-23 10:13:38 +00:00
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
|
|
|
|
// Eliminate wrong gestures (takes two iterations)
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
|
|
|
|
if (m_activeSwipeGestures.isEmpty()) {
|
|
|
|
startSwipeGesture(m_currentFingerCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto it = m_activeSwipeGestures.begin(); it != m_activeSwipeGestures.end();) {
|
|
|
|
auto g = static_cast<SwipeGesture *>(*it);
|
|
|
|
|
2022-09-11 10:14:11 +00:00
|
|
|
if (g->direction() != direction) {
|
2022-03-09 16:17:37 +00:00
|
|
|
// If a gesture was started from a touchscreen border never cancel it
|
|
|
|
if (!g->minimumXIsRelevant() || !g->maximumXIsRelevant() || !g->minimumYIsRelevant() || !g->maximumYIsRelevant()) {
|
|
|
|
Q_EMIT g->cancelled();
|
|
|
|
it = m_activeSwipeGestures.erase(it);
|
|
|
|
continue;
|
|
|
|
}
|
2017-03-19 10:40:03 +00:00
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
|
2017-03-18 10:00:30 +00:00
|
|
|
it++;
|
|
|
|
}
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
|
|
|
|
// Send progress update
|
2022-03-23 10:13:38 +00:00
|
|
|
for (SwipeGesture *g : std::as_const(m_activeSwipeGestures)) {
|
2022-09-11 10:14:08 +00:00
|
|
|
Q_EMIT g->progress(g->deltaToProgress(m_currentDelta));
|
|
|
|
Q_EMIT g->deltaProgress(m_currentDelta);
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
void GestureRecognizer::cancelActiveGestures()
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2022-10-31 19:02:23 +00:00
|
|
|
for (auto g : std::as_const(m_activeSwipeGestures)) {
|
2021-06-08 07:02:14 +00:00
|
|
|
Q_EMIT g->cancelled();
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
2022-10-31 19:02:23 +00:00
|
|
|
for (auto g : std::as_const(m_activePinchGestures)) {
|
2022-02-02 21:51:39 +00:00
|
|
|
Q_EMIT g->cancelled();
|
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
m_activeSwipeGestures.clear();
|
2022-02-02 21:51:39 +00:00
|
|
|
m_activePinchGestures.clear();
|
|
|
|
m_currentScale = 0;
|
2022-10-25 08:18:45 +00:00
|
|
|
m_currentDelta = QPointF(0, 0);
|
2022-02-02 21:51:39 +00:00
|
|
|
m_currentSwipeAxis = Axis::None;
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GestureRecognizer::cancelSwipeGesture()
|
|
|
|
{
|
2022-02-02 21:51:39 +00:00
|
|
|
cancelActiveGestures();
|
|
|
|
m_currentFingerCount = 0;
|
2022-10-25 08:18:45 +00:00
|
|
|
m_currentDelta = QPointF(0, 0);
|
2022-02-02 21:51:39 +00:00
|
|
|
m_currentSwipeAxis = Axis::None;
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GestureRecognizer::endSwipeGesture()
|
|
|
|
{
|
2022-10-25 08:18:45 +00:00
|
|
|
const QPointF delta = m_currentDelta;
|
2022-10-31 19:02:23 +00:00
|
|
|
for (auto g : std::as_const(m_activeSwipeGestures)) {
|
2022-09-11 10:14:08 +00:00
|
|
|
if (static_cast<SwipeGesture *>(g)->minimumDeltaReached(delta)) {
|
2021-06-08 07:02:14 +00:00
|
|
|
Q_EMIT g->triggered();
|
2017-03-19 10:40:03 +00:00
|
|
|
} else {
|
2021-06-08 07:02:14 +00:00
|
|
|
Q_EMIT g->cancelled();
|
2017-03-19 10:40:03 +00:00
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
m_activeSwipeGestures.clear();
|
2022-02-02 21:51:39 +00:00
|
|
|
m_currentFingerCount = 0;
|
2022-10-25 08:18:45 +00:00
|
|
|
m_currentDelta = QPointF(0, 0);
|
2022-02-02 21:51:39 +00:00
|
|
|
m_currentSwipeAxis = Axis::None;
|
|
|
|
}
|
|
|
|
|
|
|
|
int GestureRecognizer::startPinchGesture(uint fingerCount)
|
|
|
|
{
|
|
|
|
m_currentFingerCount = fingerCount;
|
|
|
|
int count = 0;
|
|
|
|
if (!m_activeSwipeGestures.isEmpty() || !m_activePinchGestures.isEmpty()) {
|
|
|
|
return 0;
|
|
|
|
}
|
2022-10-31 19:02:23 +00:00
|
|
|
for (PinchGesture *gesture : std::as_const(m_pinchGestures)) {
|
2022-09-11 10:14:11 +00:00
|
|
|
if (gesture->minimumFingerCountIsRelevant()) {
|
|
|
|
if (gesture->minimumFingerCount() > fingerCount) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (gesture->maximumFingerCountIsRelevant()) {
|
|
|
|
if (gesture->maximumFingerCount() < fingerCount) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// direction doesn't matter yet
|
|
|
|
m_activePinchGestures << gesture;
|
|
|
|
count++;
|
|
|
|
Q_EMIT gesture->started();
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2022-10-25 08:18:45 +00:00
|
|
|
void GestureRecognizer::updatePinchGesture(qreal scale, qreal angleDelta, const QPointF &posDelta)
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
|
|
|
m_currentScale = scale;
|
|
|
|
|
|
|
|
// Determine the direction of the swipe
|
2023-01-25 16:18:06 +00:00
|
|
|
PinchDirection direction;
|
2022-02-02 21:51:39 +00:00
|
|
|
if (scale < 1) {
|
2023-01-25 16:18:06 +00:00
|
|
|
direction = PinchDirection::Contracting;
|
2022-02-02 21:51:39 +00:00
|
|
|
} else {
|
2023-01-25 16:18:06 +00:00
|
|
|
direction = PinchDirection::Expanding;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Eliminate wrong gestures (takes two iterations)
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
if (m_activePinchGestures.isEmpty()) {
|
|
|
|
startPinchGesture(m_currentFingerCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto it = m_activePinchGestures.begin(); it != m_activePinchGestures.end();) {
|
|
|
|
auto g = static_cast<PinchGesture *>(*it);
|
|
|
|
|
2022-09-11 10:14:11 +00:00
|
|
|
if (g->direction() != direction) {
|
2022-02-02 21:51:39 +00:00
|
|
|
Q_EMIT g->cancelled();
|
|
|
|
it = m_activePinchGestures.erase(it);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
it++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (PinchGesture *g : std::as_const(m_activePinchGestures)) {
|
2022-09-11 10:14:08 +00:00
|
|
|
Q_EMIT g->progress(g->scaleDeltaToProgress(scale));
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GestureRecognizer::cancelPinchGesture()
|
|
|
|
{
|
|
|
|
cancelActiveGestures();
|
|
|
|
m_currentScale = 1;
|
|
|
|
m_currentFingerCount = 0;
|
|
|
|
m_currentSwipeAxis = Axis::None;
|
|
|
|
}
|
|
|
|
|
2022-03-23 10:13:38 +00:00
|
|
|
void GestureRecognizer::endPinchGesture() // because fingers up
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-10-31 19:02:23 +00:00
|
|
|
for (auto g : std::as_const(m_activePinchGestures)) {
|
2022-09-11 10:14:08 +00:00
|
|
|
if (g->minimumScaleDeltaReached(m_currentScale)) {
|
2022-02-02 21:51:39 +00:00
|
|
|
Q_EMIT g->triggered();
|
|
|
|
} else {
|
|
|
|
Q_EMIT g->cancelled();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_activeSwipeGestures.clear();
|
|
|
|
m_activePinchGestures.clear();
|
|
|
|
m_currentScale = 1;
|
|
|
|
m_currentFingerCount = 0;
|
|
|
|
m_currentSwipeAxis = Axis::None;
|
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:11 +00:00
|
|
|
bool SwipeGesture::maximumFingerCountIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_maximumFingerCountRelevant;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint SwipeGesture::minimumFingerCount() const
|
|
|
|
{
|
|
|
|
return m_minimumFingerCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SwipeGesture::setMinimumFingerCount(uint count)
|
|
|
|
{
|
|
|
|
m_minimumFingerCount = count;
|
|
|
|
m_minimumFingerCountRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SwipeGesture::minimumFingerCountIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_minimumFingerCountRelevant;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SwipeGesture::setMaximumFingerCount(uint count)
|
|
|
|
{
|
|
|
|
m_maximumFingerCount = count;
|
|
|
|
m_maximumFingerCountRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint SwipeGesture::maximumFingerCount() const
|
|
|
|
{
|
|
|
|
return m_maximumFingerCount;
|
|
|
|
}
|
|
|
|
|
2023-01-25 16:18:06 +00:00
|
|
|
SwipeDirection SwipeGesture::direction() const
|
2022-09-11 10:14:10 +00:00
|
|
|
{
|
|
|
|
return m_direction;
|
|
|
|
}
|
|
|
|
|
2023-01-25 16:18:06 +00:00
|
|
|
void SwipeGesture::setDirection(SwipeDirection direction)
|
2022-09-11 10:14:10 +00:00
|
|
|
{
|
|
|
|
m_direction = direction;
|
|
|
|
}
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
void SwipeGesture::setMinimumX(int x)
|
|
|
|
{
|
|
|
|
m_minimumX = x;
|
|
|
|
m_minimumXRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SwipeGesture::minimumX() const
|
|
|
|
{
|
|
|
|
return m_minimumX;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SwipeGesture::minimumXIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_minimumXRelevant;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SwipeGesture::setMinimumY(int y)
|
|
|
|
{
|
|
|
|
m_minimumY = y;
|
|
|
|
m_minimumYRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SwipeGesture::minimumY() const
|
|
|
|
{
|
|
|
|
return m_minimumY;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SwipeGesture::minimumYIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_minimumYRelevant;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SwipeGesture::setMaximumX(int x)
|
|
|
|
{
|
|
|
|
m_maximumX = x;
|
|
|
|
m_maximumXRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SwipeGesture::maximumX() const
|
|
|
|
{
|
|
|
|
return m_maximumX;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SwipeGesture::maximumXIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_maximumXRelevant;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SwipeGesture::setMaximumY(int y)
|
|
|
|
{
|
|
|
|
m_maximumY = y;
|
|
|
|
m_maximumYRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SwipeGesture::maximumY() const
|
|
|
|
{
|
|
|
|
return m_maximumY;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SwipeGesture::maximumYIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_maximumYRelevant;
|
|
|
|
}
|
|
|
|
|
2022-10-25 08:18:45 +00:00
|
|
|
QPointF SwipeGesture::minimumDelta() const
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
return m_minimumDelta;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-10-25 08:18:45 +00:00
|
|
|
void SwipeGesture::setMinimumDelta(const QPointF &delta)
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
m_minimumDelta = delta;
|
|
|
|
m_minimumDeltaRelevant = true;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
bool SwipeGesture::isMinimumDeltaRelevant() const
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
return m_minimumDeltaRelevant;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:11 +00:00
|
|
|
bool PinchGesture::minimumFingerCountIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_minimumFingerCountRelevant;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PinchGesture::setMinimumFingerCount(uint count)
|
|
|
|
{
|
|
|
|
m_minimumFingerCount = count;
|
|
|
|
m_minimumFingerCountRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint PinchGesture::minimumFingerCount() const
|
|
|
|
{
|
|
|
|
return m_minimumFingerCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PinchGesture::maximumFingerCountIsRelevant() const
|
|
|
|
{
|
|
|
|
return m_maximumFingerCountRelevant;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PinchGesture::setMaximumFingerCount(uint count)
|
|
|
|
{
|
|
|
|
m_maximumFingerCount = count;
|
|
|
|
m_maximumFingerCountRelevant = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint PinchGesture::maximumFingerCount() const
|
|
|
|
{
|
|
|
|
return m_maximumFingerCount;
|
|
|
|
}
|
|
|
|
|
2023-01-25 16:18:06 +00:00
|
|
|
PinchDirection PinchGesture::direction() const
|
2022-09-11 10:14:10 +00:00
|
|
|
{
|
|
|
|
return m_direction;
|
|
|
|
}
|
|
|
|
|
2023-01-25 16:18:06 +00:00
|
|
|
void PinchGesture::setDirection(PinchDirection direction)
|
2022-09-11 10:14:10 +00:00
|
|
|
{
|
|
|
|
m_direction = direction;
|
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
qreal PinchGesture::minimumScaleDelta() const
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
return m_minimumScaleDelta;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
void PinchGesture::setMinimumScaleDelta(const qreal &scaleDelta)
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
m_minimumScaleDelta = scaleDelta;
|
|
|
|
m_minimumScaleDeltaRelevant = true;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
bool PinchGesture::isMinimumScaleDeltaRelevant() const
|
2022-02-02 21:51:39 +00:00
|
|
|
{
|
2022-09-11 10:14:08 +00:00
|
|
|
return m_minimumScaleDeltaRelevant;
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int GestureRecognizer::startSwipeGesture(uint fingerCount)
|
|
|
|
{
|
|
|
|
return startSwipeGesture(fingerCount, QPointF(), StartPositionBehavior::Irrelevant);
|
|
|
|
}
|
|
|
|
|
|
|
|
int GestureRecognizer::startSwipeGesture(const QPointF &startPos)
|
|
|
|
{
|
|
|
|
return startSwipeGesture(1, startPos, StartPositionBehavior::Relevant);
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|