02d97aabf4
The rationale is explained in https://planet.kde.org/friedrich-kossebau-2023-06-28-include-also-moc-files-of-headers/ In case of KDEConnect, it impressively speeds up compilation. Before it took 390 seconds on a clean build and with this change it took 330 seconds. This is due to the mocs_compilation having to include the header files and thus all their headers. Due to the lots of small plugins we have, this means that the same headers must be compiled plenty of times. When we include the moc files directly in the C++ file, they are already available.
215 lines
5.4 KiB
C++
215 lines
5.4 KiB
C++
/*
|
|
SPDX-FileCopyrightText: 2018 Roman Gilg <subdiff@gmail.com>
|
|
SPDX-FileCopyrightText: 2021 Aleix Pol Gonzalez <aleixpol@kde.org>
|
|
|
|
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
|
|
*/
|
|
|
|
#include "pointerlockerwayland.h"
|
|
|
|
#include <QDebug>
|
|
|
|
#include "qwayland-pointer-constraints-unstable-v1.h"
|
|
#include "qwayland-relative-pointer-unstable-v1.h"
|
|
#include <QtWaylandClient/qwaylandclientextension.h>
|
|
#include <qpa/qplatformnativeinterface.h>
|
|
|
|
#include <QGuiApplication>
|
|
|
|
class PointerConstraints : public QWaylandClientExtensionTemplate<PointerConstraints>, public QtWayland::zwp_pointer_constraints_v1
|
|
{
|
|
public:
|
|
PointerConstraints()
|
|
: QWaylandClientExtensionTemplate<PointerConstraints>(1)
|
|
{
|
|
}
|
|
};
|
|
|
|
class LockedPointer : public QObject, public QtWayland::zwp_locked_pointer_v1
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
LockedPointer(struct ::zwp_locked_pointer_v1 *object, QObject *parent)
|
|
: QObject(parent)
|
|
, zwp_locked_pointer_v1(object)
|
|
{
|
|
}
|
|
|
|
Q_SIGNAL void locked();
|
|
|
|
Q_SIGNAL void unlocked();
|
|
|
|
private:
|
|
void zwp_locked_pointer_v1_locked() override
|
|
{
|
|
Q_EMIT locked();
|
|
}
|
|
|
|
void zwp_locked_pointer_v1_unlocked() override
|
|
{
|
|
Q_EMIT unlocked();
|
|
}
|
|
};
|
|
|
|
class RelativePointerManagerV1 : public QWaylandClientExtensionTemplate<RelativePointerManagerV1>, public QtWayland::zwp_relative_pointer_manager_v1
|
|
{
|
|
public:
|
|
explicit RelativePointerManagerV1()
|
|
: QWaylandClientExtensionTemplate<RelativePointerManagerV1>(1)
|
|
{
|
|
}
|
|
|
|
~RelativePointerManagerV1()
|
|
{
|
|
destroy();
|
|
}
|
|
};
|
|
|
|
class RelativePointerV1 : public QtWayland::zwp_relative_pointer_v1
|
|
{
|
|
public:
|
|
explicit RelativePointerV1(PointerLockerWayland *locker, struct ::zwp_relative_pointer_v1 *p)
|
|
: QtWayland::zwp_relative_pointer_v1(p)
|
|
, locker(locker)
|
|
{
|
|
}
|
|
|
|
~RelativePointerV1()
|
|
{
|
|
destroy();
|
|
}
|
|
|
|
void zwp_relative_pointer_v1_relative_motion(uint32_t /*utime_hi*/,
|
|
uint32_t /*utime_lo*/,
|
|
wl_fixed_t dx,
|
|
wl_fixed_t dy,
|
|
wl_fixed_t /*dx_unaccel*/,
|
|
wl_fixed_t /*dy_unaccel*/) override
|
|
{
|
|
locker->pointerMoved({wl_fixed_to_double(dx), wl_fixed_to_double(dy)});
|
|
}
|
|
|
|
private:
|
|
PointerLockerWayland *const locker;
|
|
};
|
|
|
|
PointerLockerWayland::PointerLockerWayland(QObject *parent)
|
|
: AbstractPointerLocker(parent)
|
|
{
|
|
m_relativePointerMgr = std::make_unique<RelativePointerManagerV1>();
|
|
m_pointerConstraints = new PointerConstraints;
|
|
}
|
|
|
|
PointerLockerWayland::~PointerLockerWayland()
|
|
{
|
|
delete m_pointerConstraints;
|
|
}
|
|
|
|
bool PointerLockerWayland::isLockEffective() const
|
|
{
|
|
return m_lockedPointer;
|
|
}
|
|
|
|
wl_pointer *PointerLockerWayland::getPointer()
|
|
{
|
|
QPlatformNativeInterface *native = qGuiApp->platformNativeInterface();
|
|
if (!native) {
|
|
return nullptr;
|
|
}
|
|
|
|
window()->create();
|
|
|
|
return reinterpret_cast<wl_pointer *>(native->nativeResourceForIntegration(QByteArrayLiteral("wl_pointer")));
|
|
}
|
|
|
|
void PointerLockerWayland::enforceLock()
|
|
{
|
|
if (!m_isLocked) {
|
|
return;
|
|
}
|
|
|
|
auto pointer = getPointer();
|
|
if (!m_relativePointer) {
|
|
m_relativePointer.reset(new RelativePointerV1(this, m_relativePointerMgr->get_relative_pointer(pointer)));
|
|
}
|
|
|
|
wl_surface *wlSurface = [](QWindow *window) -> wl_surface * {
|
|
if (!window) {
|
|
return nullptr;
|
|
}
|
|
|
|
QPlatformNativeInterface *native = qGuiApp->platformNativeInterface();
|
|
if (!native) {
|
|
return nullptr;
|
|
}
|
|
window->create();
|
|
return reinterpret_cast<wl_surface *>(native->nativeResourceForWindow(QByteArrayLiteral("surface"), window));
|
|
}(m_window);
|
|
|
|
m_lockedPointer =
|
|
new LockedPointer(m_pointerConstraints->lock_pointer(wlSurface, pointer, nullptr, PointerConstraints::lifetime::lifetime_persistent), this);
|
|
|
|
if (!m_lockedPointer) {
|
|
qDebug() << "ERROR when receiving locked pointer!";
|
|
return;
|
|
}
|
|
|
|
connect(m_lockedPointer, &LockedPointer::locked, this, [this] {
|
|
Q_EMIT lockEffectiveChanged(true);
|
|
});
|
|
connect(m_lockedPointer, &LockedPointer::unlocked, this, [this] {
|
|
Q_EMIT lockEffectiveChanged(false);
|
|
});
|
|
}
|
|
|
|
void PointerLockerWayland::setLocked(bool lock)
|
|
{
|
|
if (m_isLocked == lock) {
|
|
return;
|
|
}
|
|
|
|
if (!isSupported()) {
|
|
qWarning() << "Locking before having our interfaces announced";
|
|
return;
|
|
}
|
|
|
|
m_isLocked = lock;
|
|
if (lock) {
|
|
enforceLock();
|
|
} else {
|
|
cleanupLock();
|
|
}
|
|
Q_EMIT lockedChanged(lock);
|
|
}
|
|
|
|
void PointerLockerWayland::cleanupLock()
|
|
{
|
|
if (!m_lockedPointer) {
|
|
return;
|
|
}
|
|
m_lockedPointer->destroy();
|
|
m_lockedPointer->deleteLater();
|
|
m_lockedPointer = nullptr;
|
|
Q_EMIT lockEffectiveChanged(false);
|
|
}
|
|
|
|
void PointerLockerWayland::setWindow(QWindow *window)
|
|
{
|
|
if (m_window == window) {
|
|
return;
|
|
}
|
|
cleanupLock();
|
|
|
|
if (m_window) {
|
|
disconnect(m_window, &QWindow::visibleChanged, this, &PointerLockerWayland::enforceLock);
|
|
}
|
|
AbstractPointerLocker::setWindow(window);
|
|
connect(m_window, &QWindow::visibleChanged, this, &PointerLockerWayland::enforceLock);
|
|
|
|
if (m_isLocked) {
|
|
enforceLock();
|
|
}
|
|
}
|
|
|
|
#include "moc_pointerlockerwayland.cpp"
|
|
#include "pointerlockerwayland.moc"
|