From a8ff9d39a77292239be0f6b3c52d6ea5041187f4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Martin=20Gr=C3=A4=C3=9Flin?= Date: Fri, 6 Nov 2015 12:57:24 +0100 Subject: [PATCH] Add a plugin for KIdleTime Basically a simplified fork from kwayland-integration. We cannot use the idletime plugin from kwayland-integration as it a) doesn't react on our own qpa plugin name b) performs blocking roundtrips in the main thread -> freeze This simplifies by using our internal registry and we don't even check whether Seat and Idle are announced: we know they are. Reviewed-By: Bhushan Shah --- CMakeLists.txt | 1 + plugins/CMakeLists.txt | 1 + plugins/idletime/CMakeLists.txt | 17 +++++ plugins/idletime/kwin.json | 3 + plugins/idletime/poller.cpp | 124 ++++++++++++++++++++++++++++++++ plugins/idletime/poller.h | 67 +++++++++++++++++ 6 files changed, 213 insertions(+) create mode 100644 plugins/idletime/CMakeLists.txt create mode 100644 plugins/idletime/kwin.json create mode 100644 plugins/idletime/poller.cpp create mode 100644 plugins/idletime/poller.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 0ad589fd9c..ed5d0fe031 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -77,6 +77,7 @@ find_package(KF5 ${KF5_MIN_VERSION} REQUIRED COMPONENTS WidgetsAddons WindowSystem IconThemes + IdleTime ) # required frameworks by config modules find_package(KF5 ${KF5_MIN_VERSION} REQUIRED COMPONENTS diff --git a/plugins/CMakeLists.txt b/plugins/CMakeLists.txt index 74fc189f09..bf59cac3e4 100644 --- a/plugins/CMakeLists.txt +++ b/plugins/CMakeLists.txt @@ -1,2 +1,3 @@ add_subdirectory(kglobalaccel) add_subdirectory(qpa) +add_subdirectory(idletime) diff --git a/plugins/idletime/CMakeLists.txt b/plugins/idletime/CMakeLists.txt new file mode 100644 index 0000000000..791e5f7eb2 --- /dev/null +++ b/plugins/idletime/CMakeLists.txt @@ -0,0 +1,17 @@ +set(idletime_plugin_SRCS + poller.cpp +) + +add_library(KF5IdleTimeKWinWaylandPrivatePlugin MODULE ${idletime_plugin_SRCS}) +target_link_libraries(KF5IdleTimeKWinWaylandPrivatePlugin + kwin + KF5::IdleTime + KF5::WaylandClient +) + +install( + TARGETS + KF5IdleTimeKWinWaylandPrivatePlugin + DESTINATION + ${PLUGIN_INSTALL_DIR}/kf5/org.kde.kidletime.platforms/ +) diff --git a/plugins/idletime/kwin.json b/plugins/idletime/kwin.json new file mode 100644 index 0000000000..aaf6fd00dd --- /dev/null +++ b/plugins/idletime/kwin.json @@ -0,0 +1,3 @@ +{ + "platforms": ["wayland-org.kde.kwin.qpa"] +} diff --git a/plugins/idletime/poller.cpp b/plugins/idletime/poller.cpp new file mode 100644 index 0000000000..ce582e7856 --- /dev/null +++ b/plugins/idletime/poller.cpp @@ -0,0 +1,124 @@ +/******************************************************************** + KWin - the KDE window manager + This file is part of the KDE project. + +Copyright (C) 2015 Martin Gräßlin + +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 . +*********************************************************************/ +#include "poller.h" +#include "../../wayland_server.h" + +#include +#include +#include + +Poller::Poller(QObject *parent) + : AbstractSystemPoller(parent) +{ +} + +Poller::~Poller() = default; + +bool Poller::isAvailable() +{ + return true; +} + +bool Poller::setUpPoller() +{ + auto registry = KWin::waylandServer()->internalClientRegistry(); + if (!m_seat) { + const auto iface = registry->interface(KWayland::Client::Registry::Interface::Seat); + m_seat = registry->createSeat(iface.name, iface.version, this); + } + if (!m_idle) { + const auto iface = registry->interface(KWayland::Client::Registry::Interface::Idle); + m_idle = registry->createIdle(iface.name, iface.version, this); + } + return m_seat->isValid() && m_idle->isValid(); +} + +void Poller::unloadPoller() +{ +} + +void Poller::addTimeout(int nextTimeout) +{ + if (m_timeouts.contains(nextTimeout)) { + return; + } + if (!m_idle) { + return; + } + auto timeout = m_idle->getTimeout(nextTimeout, m_seat, this); + m_timeouts.insert(nextTimeout, timeout); + connect(timeout, &KWayland::Client::IdleTimeout::idle, this, + [this, nextTimeout] { + emit timeoutReached(nextTimeout); + } + ); + connect(timeout, &KWayland::Client::IdleTimeout::resumeFromIdle, this, &Poller::resumingFromIdle); +} + +void Poller::removeTimeout(int nextTimeout) +{ + auto it = m_timeouts.find(nextTimeout); + if (it == m_timeouts.end()) { + return; + } + delete it.value(); + m_timeouts.erase(it); +} + +QList< int > Poller::timeouts() const +{ + return QList(); +} + +void Poller::catchIdleEvent() +{ + if (m_catchResumeTimeout) { + // already setup + return; + } + if (!m_idle) { + return; + } + m_catchResumeTimeout = m_idle->getTimeout(0, m_seat, this); + connect(m_catchResumeTimeout, &KWayland::Client::IdleTimeout::resumeFromIdle, this, + [this] { + stopCatchingIdleEvents(); + emit resumingFromIdle(); + } + ); +} + +void Poller::stopCatchingIdleEvents() +{ + delete m_catchResumeTimeout; + m_catchResumeTimeout = nullptr; +} + +int Poller::forcePollRequest() +{ + return 0; +} + +void Poller::simulateUserActivity() +{ + for (auto it = m_timeouts.constBegin(); it != m_timeouts.constEnd(); ++it) { + it.value()->simulateUserActivity(); + } +} diff --git a/plugins/idletime/poller.h b/plugins/idletime/poller.h new file mode 100644 index 0000000000..c14ba29a33 --- /dev/null +++ b/plugins/idletime/poller.h @@ -0,0 +1,67 @@ +/******************************************************************** + KWin - the KDE window manager + This file is part of the KDE project. + +Copyright (C) 2015 Martin Gräßlin + +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 . +*********************************************************************/ +#ifndef POLLER_H +#define POLLER_H + +#include + +#include + +namespace KWayland +{ +namespace Client +{ +class Seat; +class Idle; +class IdleTimeout; +} +} + +class Poller : public AbstractSystemPoller +{ + Q_OBJECT + Q_PLUGIN_METADATA(IID "org.kde.kidletime.AbstractSystemPoller" FILE "kwin.json") + Q_INTERFACES(AbstractSystemPoller) + +public: + Poller(QObject *parent = 0); + virtual ~Poller(); + + bool isAvailable() override; + bool setUpPoller() override; + void unloadPoller() override; + +public Q_SLOTS: + void addTimeout(int nextTimeout) override; + void removeTimeout(int nextTimeout) override; + QList timeouts() const override; + int forcePollRequest() override; + void catchIdleEvent() override; + void stopCatchingIdleEvents() override; + void simulateUserActivity() override; + +private: + KWayland::Client::Seat *m_seat = nullptr; + KWayland::Client::Idle *m_idle = nullptr; + KWayland::Client::IdleTimeout *m_catchResumeTimeout = nullptr; + QHash m_timeouts; +}; + +#endif