/* KWin - the KDE window manager This file is part of the KDE project. SPDX-FileCopyrightText: 2017 Martin Gräßlin SPDX-License-Identifier: GPL-2.0-or-later */ #ifndef KWIN_GESTURES_H #define KWIN_GESTURES_H #include "kwinglobals.h" #include #include #include #include #include #include #include namespace KWin { static const QSet DEFAULT_VALID_FINGER_COUNTS = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; /* * 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% class Gesture : public QObject { Q_OBJECT public: ~Gesture() override; /** * This gesture framework allows for one gesture to * capture a range of fingers. * * This function adds an acceptable number of fingers * to the set of finger counts this gesture can * be identified by. * * By default, any number of fingers are accepted. * (see DEFAULT_VALID_FINGER_COUNTS) */ void addFingerCount(uint numFingers); bool isFingerCountAcceptable(uint fingers) const; QSet acceptableFingerCounts() const; GestureDirections direction() const; void setDirection(GestureDirections direction); protected: explicit Gesture(QObject *parent); GestureDirections m_direction; Q_SIGNALS: /** * Matching of a gesture started and this Gesture might match. * On further evaluation either the signal @ref triggered or * @ref cancelled will get emitted. */ void started(); /** * Gesture matching ended and this Gesture matched. */ void triggered(); /** * This Gesture no longer matches. */ void cancelled(); private: QSet m_validFingerCounts = DEFAULT_VALID_FINGER_COUNTS; }; class SwipeGesture : public Gesture { Q_OBJECT public: explicit SwipeGesture(QObject *parent = nullptr); ~SwipeGesture() override; 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; void setStartGeometry(const QRect &geometry); QSizeF minimumDelta() const; void setMinimumDelta(const QSizeF &delta); bool isMinimumDeltaRelevant() const; qreal deltaToProgress(const QSizeF &delta) const; bool minimumDeltaReached(const QSizeF &delta) 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); /** * The progress in actual pixel distance traveled by the fingers */ void deltaProgress(const QSizeF &delta); private: 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; }; class PinchGesture : public Gesture { Q_OBJECT public: explicit PinchGesture(QObject *parent = nullptr); ~PinchGesture() override; 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 scaleDeltaToProgress(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_minimumScaleDeltaRelevant = false; qreal m_minimumScaleDelta = DEFAULT_UNIT_SCALE_DELTA; }; class KWIN_EXPORT GestureRecognizer : public QObject { Q_OBJECT public: GestureRecognizer(QObject *parent = nullptr); ~GestureRecognizer() override; 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); void updateSwipeGesture(const QSizeF &delta); void cancelSwipeGesture(); void endSwipeGesture(); int startPinchGesture(uint fingerCount); void updatePinchGesture(qreal scale, qreal angleDelta, const QSizeF &posDelta); void cancelPinchGesture(); void endPinchGesture(); private: void cancelActiveGestures(); enum class StartPositionBehavior { Relevant, Irrelevant, }; enum class Axis { Horizontal, Vertical, None, }; int startSwipeGesture(uint fingerCount, const QPointF &startPos, StartPositionBehavior startPosBehavior); QVector m_swipeGestures; QVector m_pinchGestures; QVector m_activeSwipeGestures; QVector m_activePinchGestures; QMap m_destroyConnections; QSizeF m_currentDelta = QSizeF(0, 0); qreal m_currentScale = 1; // For Pinch Gesture recognition uint m_currentFingerCount = 0; Axis m_currentSwipeAxis = Axis::None; }; } #endif