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
|
|
|
#ifndef KWIN_GESTURES_H
|
|
|
|
#define KWIN_GESTURES_H
|
|
|
|
|
2017-03-22 20:03:04 +00:00
|
|
|
#include <kwin_export.h>
|
|
|
|
|
2017-03-18 10:00:30 +00:00
|
|
|
#include <QObject>
|
2017-03-19 10:40:03 +00:00
|
|
|
#include <QPointF>
|
|
|
|
#include <QSizeF>
|
2017-03-18 10:00:30 +00:00
|
|
|
#include <QMap>
|
|
|
|
#include <QVector>
|
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
2022-03-23 09:49:35 +00:00
|
|
|
/*
|
|
|
|
* Everytime the scale of the gesture changes by this much, the callback changes by 1.
|
|
|
|
* This is the amount of change for 1 unit of change, like switch by 1 desktop.
|
|
|
|
* */
|
|
|
|
static const qreal DEFAULT_UNIT_SCALE_DELTA = .2; // 20%
|
|
|
|
|
2017-03-18 10:00:30 +00:00
|
|
|
class Gesture : public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
~Gesture() override;
|
|
|
|
protected:
|
|
|
|
explicit Gesture(QObject *parent);
|
|
|
|
|
|
|
|
Q_SIGNALS:
|
|
|
|
/**
|
|
|
|
* Matching of a gesture started and this Gesture might match.
|
2019-01-12 10:31:32 +00:00
|
|
|
* On further evaluation either the signal @ref triggered or
|
|
|
|
* @ref cancelled will get emitted.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-03-18 10:00:30 +00:00
|
|
|
void started();
|
|
|
|
/**
|
|
|
|
* Gesture matching ended and this Gesture matched.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-03-18 10:00:30 +00:00
|
|
|
void triggered();
|
|
|
|
/**
|
|
|
|
* This Gesture no longer matches.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-03-18 10:00:30 +00:00
|
|
|
void cancelled();
|
|
|
|
};
|
|
|
|
|
|
|
|
class SwipeGesture : public Gesture
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
enum class Direction {
|
|
|
|
Down,
|
|
|
|
Left,
|
|
|
|
Up,
|
2022-02-02 21:51:39 +00:00
|
|
|
Right,
|
2017-03-18 10:00:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
explicit SwipeGesture(QObject *parent = nullptr);
|
|
|
|
~SwipeGesture() override;
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
bool minimumFingerCountIsRelevant() const;
|
|
|
|
void setMinimumFingerCount(uint count);
|
|
|
|
uint minimumFingerCount() const;
|
|
|
|
|
|
|
|
bool maximumFingerCountIsRelevant() const;
|
|
|
|
void setMaximumFingerCount(uint count);
|
|
|
|
uint maximumFingerCount() const;
|
|
|
|
|
|
|
|
Direction direction() const;
|
|
|
|
void setDirection(Direction direction);
|
|
|
|
|
|
|
|
void setMinimumX(int x);
|
|
|
|
int minimumX() const;
|
|
|
|
bool minimumXIsRelevant() const;
|
|
|
|
void setMinimumY(int y);
|
|
|
|
int minimumY() const;
|
|
|
|
bool minimumYIsRelevant() const;
|
|
|
|
|
|
|
|
void setMaximumX(int x);
|
|
|
|
int maximumX() const;
|
|
|
|
bool maximumXIsRelevant() const;
|
|
|
|
void setMaximumY(int y);
|
|
|
|
int maximumY() const;
|
|
|
|
bool maximumYIsRelevant() const;
|
2017-03-19 10:40:03 +00:00
|
|
|
void setStartGeometry(const QRect &geometry);
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
QSizeF minimumDelta() const;
|
|
|
|
void setMinimumDelta(const QSizeF &delta);
|
|
|
|
bool isMinimumDeltaRelevant() const;
|
2017-03-19 10:40:03 +00:00
|
|
|
|
|
|
|
qreal minimumDeltaReachedProgress(const QSizeF &delta) const;
|
|
|
|
bool minimumDeltaReached(const QSizeF &delta) const;
|
|
|
|
|
|
|
|
Q_SIGNALS:
|
|
|
|
/**
|
2019-01-12 10:31:32 +00:00
|
|
|
* The progress of the gesture if a minimumDelta is set.
|
2017-03-19 10:40:03 +00:00
|
|
|
* The progress is reported in [0.0,1.0]
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-03-19 10:40:03 +00:00
|
|
|
void progress(qreal);
|
|
|
|
|
2022-03-09 16:17:37 +00:00
|
|
|
/**
|
|
|
|
* The progress in actual pixel distance traveled by the fingers
|
|
|
|
*/
|
|
|
|
void deltaProgress(const QSizeF &delta);
|
|
|
|
|
2017-03-18 10:00:30 +00:00
|
|
|
private:
|
|
|
|
bool m_minimumFingerCountRelevant = false;
|
|
|
|
uint m_minimumFingerCount = 0;
|
|
|
|
bool m_maximumFingerCountRelevant = false;
|
|
|
|
uint m_maximumFingerCount = 0;
|
|
|
|
Direction m_direction = Direction::Down;
|
2017-03-19 10:40:03 +00:00
|
|
|
bool m_minimumXRelevant = false;
|
|
|
|
int m_minimumX = 0;
|
|
|
|
bool m_minimumYRelevant = false;
|
|
|
|
int m_minimumY = 0;
|
|
|
|
bool m_maximumXRelevant = false;
|
|
|
|
int m_maximumX = 0;
|
|
|
|
bool m_maximumYRelevant = false;
|
|
|
|
int m_maximumY = 0;
|
|
|
|
bool m_minimumDeltaRelevant = false;
|
|
|
|
QSizeF m_minimumDelta;
|
2017-03-18 10:00:30 +00:00
|
|
|
};
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
class PinchGesture : public Gesture
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
enum class Direction {
|
|
|
|
Expanding,
|
|
|
|
Contracting,
|
|
|
|
};
|
|
|
|
|
|
|
|
explicit PinchGesture(QObject *parent = nullptr);
|
|
|
|
~PinchGesture() override;
|
|
|
|
|
|
|
|
bool minimumFingerCountIsRelevant() const;
|
|
|
|
void setMinimumFingerCount(uint count);
|
|
|
|
uint minimumFingerCount() const;
|
|
|
|
|
|
|
|
bool maximumFingerCountIsRelevant() const;
|
|
|
|
void setMaximumFingerCount(uint count);
|
|
|
|
uint maximumFingerCount() const;
|
|
|
|
|
|
|
|
Direction direction() const;
|
|
|
|
void setDirection(Direction direction);
|
|
|
|
|
|
|
|
qreal minimumScaleDelta() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* scaleDelta is the % scale difference needed to trigger
|
|
|
|
* 0.25 will trigger when scale reaches 0.75 or 1.25
|
|
|
|
*/
|
|
|
|
void setMinimumScaleDelta(const qreal &scaleDelta);
|
|
|
|
bool isMinimumScaleDeltaRelevant() const;
|
|
|
|
|
|
|
|
qreal minimumScaleDeltaReachedProgress(const qreal &scaleDelta) const;
|
|
|
|
bool minimumScaleDeltaReached(const qreal &scaleDelta) const;
|
|
|
|
|
|
|
|
Q_SIGNALS:
|
|
|
|
/**
|
|
|
|
* The progress of the gesture if a minimumDelta is set.
|
|
|
|
* The progress is reported in [0.0,1.0]
|
|
|
|
*/
|
|
|
|
void progress(qreal);
|
|
|
|
|
|
|
|
private:
|
|
|
|
bool m_minimumFingerCountRelevant = false;
|
|
|
|
uint m_minimumFingerCount = 0;
|
|
|
|
bool m_maximumFingerCountRelevant = false;
|
|
|
|
uint m_maximumFingerCount = 0;
|
|
|
|
Direction m_direction = Direction::Expanding;
|
|
|
|
bool m_minimumScaleDeltaRelevant = false;
|
2022-03-23 09:49:35 +00:00
|
|
|
qreal m_minimumScaleDelta = DEFAULT_UNIT_SCALE_DELTA;
|
2022-02-02 21:51:39 +00:00
|
|
|
};
|
|
|
|
|
2017-03-22 20:03:04 +00:00
|
|
|
class KWIN_EXPORT GestureRecognizer : public QObject
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
GestureRecognizer(QObject *parent = nullptr);
|
|
|
|
~GestureRecognizer() override;
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
void registerSwipeGesture(SwipeGesture *gesture);
|
|
|
|
void unregisterSwipeGesture(SwipeGesture *gesture);
|
|
|
|
void registerPinchGesture(PinchGesture *gesture);
|
|
|
|
void unregisterPinchGesture(PinchGesture *gesture);
|
|
|
|
|
|
|
|
int startSwipeGesture(uint fingerCount);
|
|
|
|
int startSwipeGesture(const QPointF &startPos);
|
2017-03-18 10:00:30 +00:00
|
|
|
|
|
|
|
void updateSwipeGesture(const QSizeF &delta);
|
|
|
|
void cancelSwipeGesture();
|
|
|
|
void endSwipeGesture();
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
int startPinchGesture(uint fingerCount);
|
|
|
|
void updatePinchGesture(qreal scale, qreal angleDelta, const QSizeF& posDelta);
|
|
|
|
void cancelPinchGesture();
|
|
|
|
void endPinchGesture();
|
|
|
|
|
2017-03-18 10:00:30 +00:00
|
|
|
private:
|
2022-02-02 21:51:39 +00:00
|
|
|
void cancelActiveGestures();
|
2017-03-19 10:40:03 +00:00
|
|
|
enum class StartPositionBehavior {
|
|
|
|
Relevant,
|
2022-02-02 21:51:39 +00:00
|
|
|
Irrelevant,
|
|
|
|
};
|
|
|
|
enum class Axis {
|
|
|
|
Horizontal,
|
|
|
|
Vertical,
|
|
|
|
None,
|
2017-03-19 10:40:03 +00:00
|
|
|
};
|
|
|
|
int startSwipeGesture(uint fingerCount, const QPointF &startPos, StartPositionBehavior startPosBehavior);
|
2022-02-02 21:51:39 +00:00
|
|
|
QVector<SwipeGesture *> m_swipeGestures;
|
|
|
|
QVector<PinchGesture *> m_pinchGestures;
|
|
|
|
QVector<SwipeGesture *> m_activeSwipeGestures;
|
|
|
|
QVector<PinchGesture *> m_activePinchGestures;
|
|
|
|
QMap<Gesture *, QMetaObject::Connection> m_destroyConnections;
|
|
|
|
|
2021-06-25 23:00:56 +00:00
|
|
|
QSizeF m_currentDelta = QSizeF(0, 0);
|
2022-02-02 21:51:39 +00:00
|
|
|
qreal m_currentScale = 1; // For Pinch Gesture recognition
|
|
|
|
uint m_currentFingerCount = 0;
|
|
|
|
Axis m_currentSwipeAxis = Axis::None;
|
2017-03-18 10:00:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Q_DECLARE_METATYPE(KWin::SwipeGesture::Direction)
|
2022-02-02 21:51:39 +00:00
|
|
|
Q_DECLARE_METATYPE(KWin::PinchGesture::Direction)
|
2017-03-18 10:00:30 +00:00
|
|
|
|
|
|
|
#endif
|