/** * SPDX-FileCopyrightText: 2019 (c) Matthieu Gallien * SPDX-FileCopyrightText: 2021 Bart De Vries * * SPDX-License-Identifier: LGPL-3.0-or-later */ #include "powermanagementinterface.h" #include "powermanagementinterfacelogging.h" #include #if !defined Q_OS_ANDROID && !defined Q_OS_WIN #include #include #include #endif #if defined Q_OS_WIN #include #endif #include #include #include #if !defined Q_OS_ANDROID && !defined Q_OS_WIN #include "gnomesessioninterface.h" #include "inhibitinterface.h" #endif class PowerManagementInterfacePrivate { public: bool mPreventSleep = false; bool mInhibitedSleep = false; uint mInhibitSleepCookie = 0; uint mGnomeSleepCookie = 0; #if !defined Q_OS_ANDROID && !defined Q_OS_WIN OrgFreedesktopPowerManagementInhibitInterface *mInhibitInterface; OrgGnomeSessionManagerInterface *mGnomeInterface; #endif }; PowerManagementInterface::PowerManagementInterface(QObject *parent) : QObject(parent) , d(std::make_unique()) { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN d->mInhibitInterface = new OrgFreedesktopPowerManagementInhibitInterface(QStringLiteral("org.freedesktop.PowerManagement.Inhibit"), QStringLiteral("/org/freedesktop/PowerManagement/Inhibit"), QDBusConnection::sessionBus(), this); d->mGnomeInterface = new OrgGnomeSessionManagerInterface(QStringLiteral("org.gnome.SessionManager"), QStringLiteral("/org/gnome/SessionManager"), QDBusConnection::sessionBus(), this); #endif } PowerManagementInterface::~PowerManagementInterface() { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN delete d->mInhibitInterface; delete d->mGnomeInterface; #endif } bool PowerManagementInterface::preventSleep() const { return d->mPreventSleep; } bool PowerManagementInterface::sleepInhibited() const { return d->mInhibitedSleep; } void PowerManagementInterface::setPreventSleep(bool value) { if (d->mPreventSleep == value) { return; } if (value) { inhibitSleepPlasmaWorkspace(); inhibitSleepGnomeWorkspace(); d->mPreventSleep = true; } else { uninhibitSleepPlasmaWorkspace(); uninhibitSleepGnomeWorkspace(); d->mPreventSleep = false; } Q_EMIT preventSleepChanged(); } void PowerManagementInterface::retryInhibitingSleep() { if (d->mPreventSleep && !d->mInhibitedSleep) { inhibitSleepPlasmaWorkspace(); inhibitSleepGnomeWorkspace(); } } void PowerManagementInterface::hostSleepInhibitChanged() { } void PowerManagementInterface::inhibitDBusCallFinishedPlasmaWorkspace(QDBusPendingCallWatcher *aWatcher) { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN QDBusPendingReply reply = *aWatcher; if (reply.isError()) { } else { d->mInhibitSleepCookie = reply.argumentAt<0>(); d->mInhibitedSleep = true; Q_EMIT sleepInhibitedChanged(); } aWatcher->deleteLater(); #else Q_UNUSED(aWatcher) #endif } void PowerManagementInterface::uninhibitDBusCallFinishedPlasmaWorkspace(QDBusPendingCallWatcher *aWatcher) { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN QDBusPendingReply<> reply = *aWatcher; if (reply.isError()) { qCDebug(kastsPowerManagementInterface) << "PowerManagementInterface::uninhibitDBusCallFinished" << reply.error(); } else { d->mInhibitedSleep = false; Q_EMIT sleepInhibitedChanged(); } aWatcher->deleteLater(); #else Q_UNUSED(aWatcher) #endif } void PowerManagementInterface::inhibitDBusCallFinishedGnomeWorkspace(QDBusPendingCallWatcher *aWatcher) { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN QDBusPendingReply reply = *aWatcher; if (reply.isError()) { qCDebug(kastsPowerManagementInterface) << "PowerManagementInterface::inhibitDBusCallFinishedGnomeWorkspace" << reply.error(); } else { d->mGnomeSleepCookie = reply.argumentAt<0>(); d->mInhibitedSleep = true; Q_EMIT sleepInhibitedChanged(); } aWatcher->deleteLater(); #else Q_UNUSED(aWatcher) #endif } void PowerManagementInterface::uninhibitDBusCallFinishedGnomeWorkspace(QDBusPendingCallWatcher *aWatcher) { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN QDBusPendingReply<> reply = *aWatcher; if (reply.isError()) { qCDebug(kastsPowerManagementInterface) << "PowerManagementInterface::uninhibitDBusCallFinished" << reply.error(); } else { d->mInhibitedSleep = false; Q_EMIT sleepInhibitedChanged(); } aWatcher->deleteLater(); #else Q_UNUSED(aWatcher) #endif } void PowerManagementInterface::inhibitSleepPlasmaWorkspace() { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN auto asyncReply = d->mInhibitInterface->Inhibit(QCoreApplication::applicationName(), i18nc("Explanation for sleep inhibit during play of music", "Playing Music")); auto replyWatcher = new QDBusPendingCallWatcher(asyncReply, this); QObject::connect(replyWatcher, &QDBusPendingCallWatcher::finished, this, &PowerManagementInterface::inhibitDBusCallFinishedPlasmaWorkspace); #endif } void PowerManagementInterface::uninhibitSleepPlasmaWorkspace() { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN auto asyncReply = d->mInhibitInterface->UnInhibit(d->mInhibitSleepCookie); auto replyWatcher = new QDBusPendingCallWatcher(asyncReply, this); QObject::connect(replyWatcher, &QDBusPendingCallWatcher::finished, this, &PowerManagementInterface::uninhibitDBusCallFinishedPlasmaWorkspace); #endif } void PowerManagementInterface::inhibitSleepGnomeWorkspace() { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN // flags are: 4 = "Inhibit suspending the session or computer" // 8 = "Inhibit the session being marked as idle" auto asyncReply = d->mGnomeInterface->Inhibit(QCoreApplication::applicationName(), uint(0), i18nc("Explanation for sleep inhibit during play of music", "Playing Music"), uint(12)); auto replyWatcher = new QDBusPendingCallWatcher(asyncReply, this); QObject::connect(replyWatcher, &QDBusPendingCallWatcher::finished, this, &PowerManagementInterface::inhibitDBusCallFinishedGnomeWorkspace); #endif } void PowerManagementInterface::uninhibitSleepGnomeWorkspace() { #if !defined Q_OS_ANDROID && !defined Q_OS_WIN auto asyncReply = d->mGnomeInterface->Uninhibit(d->mGnomeSleepCookie); auto replyWatcher = new QDBusPendingCallWatcher(asyncReply, this); QObject::connect(replyWatcher, &QDBusPendingCallWatcher::finished, this, &PowerManagementInterface::uninhibitDBusCallFinishedGnomeWorkspace); #endif } void PowerManagementInterface::inhibitSleepWindowsWorkspace() { #if defined Q_OS_WIN SetThreadExecutionState(ES_CONTINUOUS | ES_SYSTEM_REQUIRED); #endif } void PowerManagementInterface::uninhibitSleepWindowsWorkspace() { #if defined Q_OS_WIN SetThreadExecutionState(ES_CONTINUOUS); #endif }