Summary: Each Edge creates a SwipeGesture for touch activation. The swipe needs to be a single finger starting from the edge into the screen for at least 20 %. The SwipeGesture and GestureRecognizer is extended to support the use cases of the touch screen edge swipe. New features supported by the gesture system are: * minimum and maximum position * a minimum delta for the swipe * progress signal based on the minimum delta * starting a swipe with a start point The Edge has the progress signal connected to its approach signal, thus visual feedback is provided through the screen edge effect. The screen edge system supports touch only for the edges (corners are too difficult to activate on touch screens). At the moment the following features are supported: * screen edge show/raise of windows (e.g. auto hidden panels) * trigger the configured action * trigger the configured callback function (e.g. script) In future it might make sense to add a touch specific configuration action to support different actions for screen edges activated by mouse and touch. BUG: 370323 Test Plan: configured a screen edge and triggered through touch, added an auto-hiding panel and triggered through touch Reviewers: #kwin, #plasma_on_wayland Subscribers: plasma-devel Tags: #plasma_on_wayland Differential Revision: https://phabricator.kde.org/D5106
209 lines
6.4 KiB
C++
209 lines
6.4 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
Copyright (C) 2017 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
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, see <http://www.gnu.org/licenses/>.
|
|
*********************************************************************/
|
|
#include "gestures.h"
|
|
|
|
#include <QRect>
|
|
#include <functional>
|
|
|
|
namespace KWin
|
|
{
|
|
|
|
Gesture::Gesture(QObject *parent)
|
|
: QObject(parent)
|
|
{
|
|
}
|
|
|
|
Gesture::~Gesture() = default;
|
|
|
|
SwipeGesture::SwipeGesture(QObject *parent)
|
|
: Gesture(parent)
|
|
{
|
|
}
|
|
|
|
SwipeGesture::~SwipeGesture() = default;
|
|
|
|
void SwipeGesture::setStartGeometry(const QRect &geometry)
|
|
{
|
|
setMinimumX(geometry.x());
|
|
setMinimumY(geometry.y());
|
|
setMaximumX(geometry.x() + geometry.width());
|
|
setMaximumY(geometry.y() + geometry.height());
|
|
}
|
|
|
|
qreal SwipeGesture::minimumDeltaReachedProgress(const QSizeF &delta) const
|
|
{
|
|
if (!m_minimumDeltaRelevant || m_minimumDelta.isNull()) {
|
|
return 1.0;
|
|
}
|
|
switch (m_direction) {
|
|
case Direction::Up:
|
|
case Direction::Down:
|
|
return std::min(std::abs(delta.height()) / std::abs(m_minimumDelta.height()), 1.0);
|
|
case Direction::Left:
|
|
case Direction::Right:
|
|
return std::min(std::abs(delta.width()) / std::abs(m_minimumDelta.width()), 1.0);
|
|
default:
|
|
Q_UNREACHABLE();
|
|
}
|
|
}
|
|
|
|
bool SwipeGesture::minimumDeltaReached(const QSizeF &delta) const
|
|
{
|
|
return minimumDeltaReachedProgress(delta) >= 1.0;
|
|
}
|
|
|
|
GestureRecognizer::GestureRecognizer(QObject *parent)
|
|
: QObject(parent)
|
|
{
|
|
}
|
|
|
|
GestureRecognizer::~GestureRecognizer() = default;
|
|
|
|
void GestureRecognizer::registerGesture(KWin::Gesture* gesture)
|
|
{
|
|
Q_ASSERT(!m_gestures.contains(gesture));
|
|
auto connection = connect(gesture, &QObject::destroyed, this, std::bind(&GestureRecognizer::unregisterGesture, this, gesture));
|
|
m_destroyConnections.insert(gesture, connection);
|
|
m_gestures << gesture;
|
|
}
|
|
|
|
void GestureRecognizer::unregisterGesture(KWin::Gesture* gesture)
|
|
{
|
|
auto it = m_destroyConnections.find(gesture);
|
|
if (it != m_destroyConnections.end()) {
|
|
disconnect(it.value());
|
|
m_destroyConnections.erase(it);
|
|
}
|
|
m_gestures.removeAll(gesture);
|
|
if (m_activeSwipeGestures.removeOne(gesture)) {
|
|
emit gesture->cancelled();
|
|
}
|
|
}
|
|
|
|
int GestureRecognizer::startSwipeGesture(uint fingerCount, const QPointF &startPos, StartPositionBehavior startPosBehavior)
|
|
{
|
|
int count = 0;
|
|
// TODO: verify that no gesture is running
|
|
for (Gesture *gesture : qAsConst(m_gestures)) {
|
|
SwipeGesture *swipeGesture = qobject_cast<SwipeGesture*>(gesture);
|
|
if (!gesture) {
|
|
continue;
|
|
}
|
|
if (swipeGesture->minimumFingerCountIsRelevant()) {
|
|
if (swipeGesture->minimumFingerCount() > fingerCount) {
|
|
continue;
|
|
}
|
|
}
|
|
if (swipeGesture->maximumFingerCountIsRelevant()) {
|
|
if (swipeGesture->maximumFingerCount() < fingerCount) {
|
|
continue;
|
|
}
|
|
}
|
|
if (startPosBehavior == StartPositionBehavior::Relevant) {
|
|
if (swipeGesture->minimumXIsRelevant()) {
|
|
if (swipeGesture->minimumX() > startPos.x()) {
|
|
continue;
|
|
}
|
|
}
|
|
if (swipeGesture->maximumXIsRelevant()) {
|
|
if (swipeGesture->maximumX() < startPos.x()) {
|
|
continue;
|
|
}
|
|
}
|
|
if (swipeGesture->minimumYIsRelevant()) {
|
|
if (swipeGesture->minimumY() > startPos.y()) {
|
|
continue;
|
|
}
|
|
}
|
|
if (swipeGesture->maximumYIsRelevant()) {
|
|
if (swipeGesture->maximumY() < startPos.y()) {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
// direction doesn't matter yet
|
|
m_activeSwipeGestures << swipeGesture;
|
|
count++;
|
|
emit swipeGesture->started();
|
|
}
|
|
return count;
|
|
}
|
|
|
|
void GestureRecognizer::updateSwipeGesture(const QSizeF &delta)
|
|
{
|
|
m_swipeUpdates << delta;
|
|
// determine the direction of the swipe
|
|
if (delta.width() == delta.height()) {
|
|
// special case of diagonal, this is not yet supported, thus cancel all gestures
|
|
cancelActiveSwipeGestures();
|
|
return;
|
|
}
|
|
SwipeGesture::Direction direction;
|
|
if (std::abs(delta.width()) > std::abs(delta.height())) {
|
|
// horizontal
|
|
direction = delta.width() < 0 ? SwipeGesture::Direction::Left : SwipeGesture::Direction::Right;
|
|
} else {
|
|
// vertical
|
|
direction = delta.height() < 0 ? SwipeGesture::Direction::Up : SwipeGesture::Direction::Down;
|
|
}
|
|
const QSizeF combinedDelta = std::accumulate(m_swipeUpdates.constBegin(), m_swipeUpdates.constEnd(), QSizeF(0, 0));
|
|
for (auto it = m_activeSwipeGestures.begin(); it != m_activeSwipeGestures.end();) {
|
|
auto g = qobject_cast<SwipeGesture*>(*it);
|
|
if (g->direction() == direction) {
|
|
if (g->isMinimumDeltaRelevant()) {
|
|
emit g->progress(g->minimumDeltaReachedProgress(combinedDelta));
|
|
}
|
|
it++;
|
|
} else {
|
|
emit g->cancelled();
|
|
it = m_activeSwipeGestures.erase(it);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GestureRecognizer::cancelActiveSwipeGestures()
|
|
{
|
|
for (auto g : qAsConst(m_activeSwipeGestures)) {
|
|
emit g->cancelled();
|
|
}
|
|
m_activeSwipeGestures.clear();
|
|
}
|
|
|
|
void GestureRecognizer::cancelSwipeGesture()
|
|
{
|
|
cancelActiveSwipeGestures();
|
|
m_swipeUpdates.clear();
|
|
}
|
|
|
|
void GestureRecognizer::endSwipeGesture()
|
|
{
|
|
const QSizeF delta = std::accumulate(m_swipeUpdates.constBegin(), m_swipeUpdates.constEnd(), QSizeF(0, 0));
|
|
for (auto g : qAsConst(m_activeSwipeGestures)) {
|
|
if (static_cast<SwipeGesture*>(g)->minimumDeltaReached(delta)) {
|
|
emit g->triggered();
|
|
} else {
|
|
emit g->cancelled();
|
|
}
|
|
}
|
|
m_activeSwipeGestures.clear();
|
|
m_swipeUpdates.clear();
|
|
}
|
|
|
|
}
|