You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
156 lines
5.7 KiB
156 lines
5.7 KiB
/* |
|
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 "statusnotifieritem_engine.h" |
|
#include "statusnotifieritemsource.h" |
|
#include <QStringList> |
|
|
|
#include "dbusproperties.h" |
|
|
|
#include "debug.h" |
|
#include <iostream> |
|
|
|
static const QString s_watcherServiceName(QStringLiteral("org.kde.StatusNotifierWatcher")); |
|
|
|
StatusNotifierItemEngine::StatusNotifierItemEngine(QObject *parent, const QVariantList &args) |
|
: Plasma::DataEngine(parent, args) |
|
, m_statusNotifierWatcher(nullptr) |
|
{ |
|
Q_UNUSED(args); |
|
init(); |
|
} |
|
|
|
StatusNotifierItemEngine::~StatusNotifierItemEngine() |
|
{ |
|
QDBusConnection::sessionBus().unregisterService(m_serviceName); |
|
} |
|
|
|
Plasma::Service *StatusNotifierItemEngine::serviceForSource(const QString &name) |
|
{ |
|
StatusNotifierItemSource *source = dynamic_cast<StatusNotifierItemSource *>(containerForSource(name)); |
|
// if source does not exist, return null service |
|
if (!source) { |
|
return Plasma::DataEngine::serviceForSource(name); |
|
} |
|
|
|
Plasma::Service *service = source->createService(); |
|
service->setParent(this); |
|
return service; |
|
} |
|
|
|
void StatusNotifierItemEngine::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, &StatusNotifierItemEngine::serviceChange); |
|
|
|
registerWatcher(s_watcherServiceName); |
|
} |
|
} |
|
|
|
void StatusNotifierItemEngine::serviceChange(const QString &name, const QString &oldOwner, const QString &newOwner) |
|
{ |
|
qCDebug(DATAENGINE_SNI) << "Service" << name << "status change, old owner:" << oldOwner << "new:" << newOwner; |
|
|
|
if (newOwner.isEmpty()) { |
|
// unregistered |
|
unregisterWatcher(name); |
|
} else if (oldOwner.isEmpty()) { |
|
// registered |
|
registerWatcher(name); |
|
} |
|
} |
|
|
|
void StatusNotifierItemEngine::registerWatcher(const QString &service) |
|
{ |
|
// qCDebug(DATAENGINE_SNI)<<"service appeared"<<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()); |
|
|
|
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) { |
|
newItem(service); |
|
} |
|
}); |
|
|
|
connect(m_statusNotifierWatcher, |
|
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemRegistered, |
|
this, |
|
&StatusNotifierItemEngine::serviceRegistered); |
|
connect(m_statusNotifierWatcher, |
|
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemUnregistered, |
|
this, |
|
&StatusNotifierItemEngine::serviceUnregistered); |
|
|
|
} else { |
|
delete m_statusNotifierWatcher; |
|
m_statusNotifierWatcher = nullptr; |
|
qCDebug(DATAENGINE_SNI) << "System tray daemon not reachable"; |
|
} |
|
} |
|
} |
|
|
|
void StatusNotifierItemEngine::unregisterWatcher(const QString &service) |
|
{ |
|
if (service == s_watcherServiceName) { |
|
qCDebug(DATAENGINE_SNI) << s_watcherServiceName << "disappeared"; |
|
|
|
disconnect(m_statusNotifierWatcher, |
|
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemRegistered, |
|
this, |
|
&StatusNotifierItemEngine::serviceRegistered); |
|
disconnect(m_statusNotifierWatcher, |
|
&OrgKdeStatusNotifierWatcherInterface::StatusNotifierItemUnregistered, |
|
this, |
|
&StatusNotifierItemEngine::serviceUnregistered); |
|
|
|
removeAllSources(); |
|
|
|
delete m_statusNotifierWatcher; |
|
m_statusNotifierWatcher = nullptr; |
|
} |
|
} |
|
|
|
void StatusNotifierItemEngine::serviceRegistered(const QString &service) |
|
{ |
|
qCDebug(DATAENGINE_SNI) << "Registering" << service; |
|
newItem(service); |
|
} |
|
|
|
void StatusNotifierItemEngine::serviceUnregistered(const QString &service) |
|
{ |
|
removeSource(service); |
|
} |
|
|
|
void StatusNotifierItemEngine::newItem(const QString &service) |
|
{ |
|
StatusNotifierItemSource *itemSource = new StatusNotifierItemSource(service, this); |
|
addSource(itemSource); |
|
} |
|
|
|
K_PLUGIN_CLASS_WITH_JSON(StatusNotifierItemEngine, "plasma-dataengine-statusnotifieritem.json") |
|
|
|
#include "statusnotifieritem_engine.moc"
|
|
|