Brooklyn/plasma/workspace/applets/systemtray/statusnotifieritemhost.cpp
2022-03-05 22:41:29 +05:00

190 lines
6.3 KiB
C++

/*
SPDX-FileCopyrightText: 2009 Marco Martin <notmart@gmail.com>
SPDX-FileCopyrightText: 2009 Matthieu Gallien <matthieu_gallien@yahoo.fr>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "statusnotifieritemhost.h"
#include "statusnotifieritemsource.h"
#include <QStringList>
#include "dbusproperties.h"
#include "debug.h"
#include <iostream>
class StatusNotifierItemHostSingleton
{
public:
StatusNotifierItemHost self;
};
Q_GLOBAL_STATIC(StatusNotifierItemHostSingleton, privateStatusNotifierItemHostSelf)
static const QString s_watcherServiceName(QStringLiteral("org.kde.StatusNotifierWatcher"));
StatusNotifierItemHost::StatusNotifierItemHost()
: QObject()
, m_statusNotifierWatcher(nullptr)
{
init();
}
StatusNotifierItemHost::~StatusNotifierItemHost()
{
QDBusConnection::sessionBus().unregisterService(m_serviceName);
}
StatusNotifierItemHost *StatusNotifierItemHost::self()
{
return &privateStatusNotifierItemHostSelf()->self;
}
const QList<QString> StatusNotifierItemHost::services() const
{
return m_sniServices.keys();
}
StatusNotifierItemSource *StatusNotifierItemHost::itemForService(const QString service)
{
return m_sniServices.value(service);
}
void StatusNotifierItemHost::init()
{
if (QDBusConnection::sessionBus().isConnected()) {
m_serviceName = "org.kde.StatusNotifierHost-" + QString::number(QCoreApplication::applicationPid());
QDBusConnection::sessionBus().registerService(m_serviceName);
QDBusServiceWatcher *watcher =
new QDBusServiceWatcher(s_watcherServiceName, QDBusConnection::sessionBus(), QDBusServiceWatcher::WatchForOwnerChange, this);
connect(watcher, &QDBusServiceWatcher::serviceOwnerChanged, this, &StatusNotifierItemHost::serviceChange);
registerWatcher(s_watcherServiceName);
}
}
void StatusNotifierItemHost::serviceChange(const QString &name, const QString &oldOwner, const QString &newOwner)
{
qCDebug(SYSTEM_TRAY) << "Service" << name << "status change, old owner:" << oldOwner << "new:" << newOwner;
if (newOwner.isEmpty()) {
// unregistered
unregisterWatcher(name);
} else if (oldOwner.isEmpty()) {
// registered
registerWatcher(name);
}
}
void StatusNotifierItemHost::registerWatcher(const QString &service)
{
if (service == s_watcherServiceName) {
delete m_statusNotifierWatcher;
m_statusNotifierWatcher =
new org::kde::StatusNotifierWatcher(s_watcherServiceName, QStringLiteral("/StatusNotifierWatcher"), QDBusConnection::sessionBus());
if (m_statusNotifierWatcher->isValid()) {
m_statusNotifierWatcher->call(QDBus::NoBlock, QStringLiteral("RegisterStatusNotifierHost"), m_serviceName);
OrgFreedesktopDBusPropertiesInterface propetriesIface(m_statusNotifierWatcher->service(),
m_statusNotifierWatcher->path(),
m_statusNotifierWatcher->connection());
connect(m_statusNotifierWatcher,
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemRegistered,
this,
&StatusNotifierItemHost::serviceRegistered);
connect(m_statusNotifierWatcher,
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemUnregistered,
this,
&StatusNotifierItemHost::serviceUnregistered);
QDBusPendingReply<QDBusVariant> pendingItems = propetriesIface.Get(m_statusNotifierWatcher->interface(), "RegisteredStatusNotifierItems");
QDBusPendingCallWatcher *watcher = new QDBusPendingCallWatcher(pendingItems, this);
connect(watcher, &QDBusPendingCallWatcher::finished, this, [=]() {
watcher->deleteLater();
QDBusReply<QDBusVariant> reply = *watcher;
QStringList registeredItems = reply.value().variant().toStringList();
foreach (const QString &service, registeredItems) {
if (!m_sniServices.contains(service)) { // due to async nature of this call, service may be already there
addSNIService(service);
}
}
});
} else {
delete m_statusNotifierWatcher;
m_statusNotifierWatcher = nullptr;
qCDebug(SYSTEM_TRAY) << "System tray daemon not reachable";
}
}
}
void StatusNotifierItemHost::unregisterWatcher(const QString &service)
{
if (service == s_watcherServiceName) {
qCDebug(SYSTEM_TRAY) << s_watcherServiceName << "disappeared";
disconnect(m_statusNotifierWatcher,
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemRegistered,
this,
&StatusNotifierItemHost::serviceRegistered);
disconnect(m_statusNotifierWatcher,
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemUnregistered,
this,
&StatusNotifierItemHost::serviceUnregistered);
removeAllSNIServices();
delete m_statusNotifierWatcher;
m_statusNotifierWatcher = nullptr;
}
}
void StatusNotifierItemHost::serviceRegistered(const QString &service)
{
qCDebug(SYSTEM_TRAY) << "Registering" << service;
addSNIService(service);
}
void StatusNotifierItemHost::serviceUnregistered(const QString &service)
{
removeSNIService(service);
}
void StatusNotifierItemHost::removeAllSNIServices()
{
QHashIterator<QString, StatusNotifierItemSource *> it(m_sniServices);
while (it.hasNext()) {
it.next();
StatusNotifierItemSource *item = it.value();
item->disconnect();
item->deleteLater();
Q_EMIT itemRemoved(it.key());
}
m_sniServices.clear();
}
void StatusNotifierItemHost::addSNIService(const QString &service)
{
StatusNotifierItemSource *item = new StatusNotifierItemSource(service, this);
m_sniServices.insert(service, item);
Q_EMIT itemAdded(service);
}
void StatusNotifierItemHost::removeSNIService(const QString &service)
{
if (m_sniServices.contains(service)) {
auto item = m_sniServices.value(service);
item->disconnect();
item->deleteLater();
m_sniServices.remove(service);
Q_EMIT itemRemoved(service);
}
}