forked from Qortal/Brooklyn
190 lines
6.3 KiB
C++
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);
|
|
}
|
|
}
|
|
|