Use isNull on QSizeF to check for a zero delta instead of comparing it with a default-constructed QSizeF, which in practice initializes to (-1.0,-1.0). This caused relative motion events to be omitted if the delta happened to be equal to (-1.0,-1.0), causing mouse jumping in some applications. BUG: 444510 Signed-off-by: John Brooks <john@fastquake.com>
264 lines
8.2 KiB
C++
264 lines
8.2 KiB
C++
/*
|
|
SPDX-FileCopyrightText: 2015 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
|
|
*/
|
|
#include "fakeinput_interface.h"
|
|
#include "display.h"
|
|
|
|
#include <QPointF>
|
|
#include <memory>
|
|
#include <vector>
|
|
|
|
#include <qwayland-server-fake-input.h>
|
|
#include <wayland-server.h>
|
|
|
|
namespace KWaylandServer
|
|
{
|
|
static const quint32 s_version = 4;
|
|
|
|
class FakeInputInterfacePrivate : public QtWaylandServer::org_kde_kwin_fake_input
|
|
{
|
|
public:
|
|
FakeInputInterfacePrivate(FakeInputInterface *_q, Display *display);
|
|
std::map<wl_resource *, std::unique_ptr<FakeInputDevice>> devices;
|
|
|
|
private:
|
|
FakeInputDevice *device(wl_resource *r);
|
|
|
|
FakeInputInterface *q;
|
|
static QList<quint32> touchIds;
|
|
|
|
protected:
|
|
void org_kde_kwin_fake_input_bind_resource(Resource *resource) override;
|
|
void org_kde_kwin_fake_input_destroy_resource(Resource *resource) override;
|
|
void org_kde_kwin_fake_input_authenticate(Resource *resource, const QString &application, const QString &reason) override;
|
|
void org_kde_kwin_fake_input_pointer_motion(Resource *resource, wl_fixed_t delta_x, wl_fixed_t delta_y) override;
|
|
void org_kde_kwin_fake_input_button(Resource *resource, uint32_t button, uint32_t state) override;
|
|
void org_kde_kwin_fake_input_axis(Resource *resource, uint32_t axis, wl_fixed_t value) override;
|
|
void org_kde_kwin_fake_input_touch_down(Resource *resource, uint32_t id, wl_fixed_t x, wl_fixed_t y) override;
|
|
void org_kde_kwin_fake_input_touch_motion(Resource *resource, uint32_t id, wl_fixed_t x, wl_fixed_t y) override;
|
|
void org_kde_kwin_fake_input_touch_up(Resource *resource, uint32_t id) override;
|
|
void org_kde_kwin_fake_input_touch_cancel(Resource *resource) override;
|
|
void org_kde_kwin_fake_input_touch_frame(Resource *resource) override;
|
|
void org_kde_kwin_fake_input_pointer_motion_absolute(Resource *resource, wl_fixed_t x, wl_fixed_t y) override;
|
|
void org_kde_kwin_fake_input_keyboard_key(Resource *resource, uint32_t button, uint32_t state) override;
|
|
};
|
|
|
|
QList<quint32> FakeInputInterfacePrivate::touchIds = QList<quint32>();
|
|
|
|
FakeInputInterfacePrivate::FakeInputInterfacePrivate(FakeInputInterface *_q, Display *display)
|
|
: QtWaylandServer::org_kde_kwin_fake_input(*display, s_version)
|
|
, q(_q)
|
|
{
|
|
}
|
|
|
|
FakeInputInterface::FakeInputInterface(Display *display)
|
|
: d(std::make_unique<FakeInputInterfacePrivate>(this, display))
|
|
{
|
|
}
|
|
|
|
FakeInputInterface::~FakeInputInterface() = default;
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_bind_resource(Resource *resource)
|
|
{
|
|
auto device = new FakeInputDevice(q, resource->handle);
|
|
devices[resource->handle] = std::unique_ptr<FakeInputDevice>(device);
|
|
Q_EMIT q->deviceCreated(device);
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_destroy_resource(Resource *resource)
|
|
{
|
|
auto it = devices.find(resource->handle);
|
|
if (it != devices.end()) {
|
|
const auto [resource, device] = std::move(*it);
|
|
devices.erase(it);
|
|
Q_EMIT q->deviceDestroyed(device.get());
|
|
}
|
|
}
|
|
|
|
FakeInputDevice *FakeInputInterfacePrivate::device(wl_resource *r)
|
|
{
|
|
return devices[r].get();
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_authenticate(Resource *resource, const QString &application, const QString &reason)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d) {
|
|
return;
|
|
}
|
|
Q_EMIT d->authenticationRequested(application, reason);
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_pointer_motion(Resource *resource, wl_fixed_t delta_x, wl_fixed_t delta_y)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
Q_EMIT d->pointerMotionRequested(QPointF(wl_fixed_to_double(delta_x), wl_fixed_to_double(delta_y)));
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_button(Resource *resource, uint32_t button, uint32_t state)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
switch (state) {
|
|
case WL_POINTER_BUTTON_STATE_PRESSED:
|
|
Q_EMIT d->pointerButtonPressRequested(button);
|
|
break;
|
|
case WL_POINTER_BUTTON_STATE_RELEASED:
|
|
Q_EMIT d->pointerButtonReleaseRequested(button);
|
|
break;
|
|
default:
|
|
// nothing
|
|
break;
|
|
}
|
|
}
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_axis(Resource *resource, uint32_t axis, wl_fixed_t value)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
Qt::Orientation orientation;
|
|
switch (axis) {
|
|
case WL_POINTER_AXIS_HORIZONTAL_SCROLL:
|
|
orientation = Qt::Horizontal;
|
|
break;
|
|
case WL_POINTER_AXIS_VERTICAL_SCROLL:
|
|
orientation = Qt::Vertical;
|
|
break;
|
|
default:
|
|
// invalid
|
|
return;
|
|
}
|
|
Q_EMIT d->pointerAxisRequested(orientation, wl_fixed_to_double(value));
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_touch_down(Resource *resource, uint32_t id, wl_fixed_t x, wl_fixed_t y)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
if (touchIds.contains(id)) {
|
|
return;
|
|
}
|
|
touchIds << id;
|
|
Q_EMIT d->touchDownRequested(id, QPointF(wl_fixed_to_double(x), wl_fixed_to_double(y)));
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_touch_motion(Resource *resource, uint32_t id, wl_fixed_t x, wl_fixed_t y)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
if (!touchIds.contains(id)) {
|
|
return;
|
|
}
|
|
Q_EMIT d->touchMotionRequested(id, QPointF(wl_fixed_to_double(x), wl_fixed_to_double(y)));
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_touch_up(Resource *resource, uint32_t id)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
if (!touchIds.contains(id)) {
|
|
return;
|
|
}
|
|
touchIds.removeOne(id);
|
|
Q_EMIT d->touchUpRequested(id);
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_touch_cancel(Resource *resource)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
touchIds.clear();
|
|
Q_EMIT d->touchCancelRequested();
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_touch_frame(Resource *resource)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
Q_EMIT d->touchFrameRequested();
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_pointer_motion_absolute(Resource *resource, wl_fixed_t x, wl_fixed_t y)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
Q_EMIT d->pointerMotionAbsoluteRequested(QPointF(wl_fixed_to_double(x), wl_fixed_to_double(y)));
|
|
}
|
|
|
|
void FakeInputInterfacePrivate::org_kde_kwin_fake_input_keyboard_key(Resource *resource, uint32_t button, uint32_t state)
|
|
{
|
|
FakeInputDevice *d = device(resource->handle);
|
|
if (!d || !d->isAuthenticated()) {
|
|
return;
|
|
}
|
|
switch (state) {
|
|
case WL_KEYBOARD_KEY_STATE_PRESSED:
|
|
Q_EMIT d->keyboardKeyPressRequested(button);
|
|
break;
|
|
case WL_KEYBOARD_KEY_STATE_RELEASED:
|
|
Q_EMIT d->keyboardKeyReleaseRequested(button);
|
|
break;
|
|
default:
|
|
// nothing
|
|
break;
|
|
}
|
|
}
|
|
|
|
class FakeInputDevicePrivate
|
|
{
|
|
public:
|
|
FakeInputDevicePrivate(FakeInputInterface *interface, wl_resource *resource);
|
|
wl_resource *resource;
|
|
FakeInputInterface *interface;
|
|
bool authenticated = false;
|
|
};
|
|
|
|
FakeInputDevicePrivate::FakeInputDevicePrivate(FakeInputInterface *interface, wl_resource *resource)
|
|
: resource(resource)
|
|
, interface(interface)
|
|
{
|
|
}
|
|
|
|
FakeInputDevice::FakeInputDevice(FakeInputInterface *parent, wl_resource *resource)
|
|
: d(std::make_unique<FakeInputDevicePrivate>(parent, resource))
|
|
{
|
|
}
|
|
|
|
FakeInputDevice::~FakeInputDevice() = default;
|
|
|
|
void FakeInputDevice::setAuthentication(bool authenticated)
|
|
{
|
|
d->authenticated = authenticated;
|
|
}
|
|
|
|
wl_resource *FakeInputDevice::resource()
|
|
{
|
|
return d->resource;
|
|
}
|
|
|
|
bool FakeInputDevice::isAuthenticated() const
|
|
{
|
|
return d->authenticated;
|
|
}
|
|
|
|
}
|