Advance Wayland and KDE package bring-up

Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
This commit is contained in:
2026-04-14 10:51:06 +01:00
parent 51f3c21121
commit cf12defd28
15214 changed files with 20594243 additions and 269 deletions
@@ -0,0 +1,327 @@
/*
This file is part of the KDE libraries
SPDX-FileCopyrightText: 2005 Olivier Goffart <ogoffart at kde.org>
SPDX-FileCopyrightText: 2013-2015 Martin Klapetek <mklapetek@kde.org>
SPDX-FileCopyrightText: 2017 Eike Hein <hein@kde.org>
SPDX-FileCopyrightText: 2022 Harald Sitter <sitter@kde.org>
SPDX-License-Identifier: LGPL-2.0-only
*/
#include "knotification.h"
#include "knotification_p.h"
#include "knotificationmanager_p.h"
#include <config-knotifications.h>
#include <QFileInfo>
#include <QHash>
#ifdef HAVE_DBUS
#include <QDBusConnection>
#include <QDBusConnectionInterface>
#endif
#include "knotificationplugin.h"
#include "knotificationreplyaction.h"
#include "knotifyconfig.h"
#if defined(Q_OS_ANDROID)
#include "notifybyandroid.h"
#elif defined(Q_OS_MACOS)
#include "notifybymacosnotificationcenter.h"
#elif defined(WITH_SNORETOAST)
#include "notifybysnore.h"
#elif defined(HAVE_DBUS)
#include "notifybypopup.h"
#include "notifybyportal.h"
#endif
#include "debug_p.h"
#if defined(HAVE_CANBERRA)
#include "notifybyaudio.h"
#endif
typedef QHash<QString, QString> Dict;
struct Q_DECL_HIDDEN KNotificationManager::Private {
QHash<int, KNotification *> notifications;
QHash<QString, KNotificationPlugin *> notifyPlugins;
QStringList dirtyConfigCache;
bool portalDBusServiceExists = false;
};
class KNotificationManagerSingleton
{
public:
KNotificationManager instance;
};
Q_GLOBAL_STATIC(KNotificationManagerSingleton, s_self)
KNotificationManager *KNotificationManager::self()
{
return &s_self()->instance;
}
KNotificationManager::KNotificationManager()
: d(new Private)
{
qDeleteAll(d->notifyPlugins);
d->notifyPlugins.clear();
#ifdef HAVE_DBUS
if (isInsideSandbox()) {
QDBusConnectionInterface *interface = QDBusConnection::sessionBus().interface();
d->portalDBusServiceExists = interface->isServiceRegistered(QStringLiteral("org.freedesktop.portal.Desktop"));
}
QDBusConnection::sessionBus().connect(QString(),
QStringLiteral("/Config"),
QStringLiteral("org.kde.knotification"),
QStringLiteral("reparseConfiguration"),
this,
SLOT(reparseConfiguration(QString)));
#endif
}
KNotificationManager::~KNotificationManager() = default;
KNotificationPlugin *KNotificationManager::pluginForAction(const QString &action)
{
KNotificationPlugin *plugin = d->notifyPlugins.value(action);
// We already loaded a plugin for this action.
if (plugin) {
return plugin;
}
auto addPlugin = [this](KNotificationPlugin *plugin) {
d->notifyPlugins[plugin->optionName()] = plugin;
connect(plugin, &KNotificationPlugin::finished, this, &KNotificationManager::notifyPluginFinished);
connect(plugin, &KNotificationPlugin::xdgActivationTokenReceived, this, &KNotificationManager::xdgActivationTokenReceived);
connect(plugin, &KNotificationPlugin::actionInvoked, this, &KNotificationManager::notificationActivated);
connect(plugin, &KNotificationPlugin::replied, this, &KNotificationManager::notificationReplied);
};
// Load plugin.
// We have a series of built-ins up first, and fall back to trying
// to instantiate an externally supplied plugin.
if (action == QLatin1String("Popup")) {
#if defined(Q_OS_ANDROID)
plugin = new NotifyByAndroid(this);
#elif defined(WITH_SNORETOAST)
plugin = new NotifyBySnore(this);
#elif defined(Q_OS_MACOS)
plugin = new NotifyByMacOSNotificationCenter(this);
#elif defined(HAVE_DBUS)
if (d->portalDBusServiceExists) {
plugin = new NotifyByPortal(this);
} else {
plugin = new NotifyByPopup(this);
}
#endif
addPlugin(plugin);
} else if (action == QLatin1String("Sound")) {
#if defined(HAVE_CANBERRA)
plugin = new NotifyByAudio(this);
addPlugin(plugin);
#endif
}
return plugin;
}
void KNotificationManager::notifyPluginFinished(KNotification *notification)
{
if (!notification || !d->notifications.contains(notification->id())) {
return;
}
notification->deref();
}
void KNotificationManager::notificationActivated(int id, const QString &actionId)
{
if (d->notifications.contains(id)) {
qCDebug(LOG_KNOTIFICATIONS) << id << " " << actionId;
KNotification *n = d->notifications[id];
n->activate(actionId);
// Resident actions delegate control over notification lifetime to the client
if (!n->hints().value(QStringLiteral("resident")).toBool()) {
close(id);
}
}
}
void KNotificationManager::xdgActivationTokenReceived(int id, const QString &token)
{
KNotification *n = d->notifications.value(id);
if (n) {
qCDebug(LOG_KNOTIFICATIONS) << "Token received for" << id << token;
n->d->xdgActivationToken = token;
Q_EMIT n->xdgActivationTokenChanged();
}
}
void KNotificationManager::notificationReplied(int id, const QString &text)
{
if (KNotification *n = d->notifications.value(id)) {
if (auto *replyAction = n->replyAction()) {
// cannot really send out a "activate inline-reply" signal from plugin to manager
// so we instead assume empty reply is not supported and means normal invocation
if (text.isEmpty() && replyAction->fallbackBehavior() == KNotificationReplyAction::FallbackBehavior::UseRegularAction) {
Q_EMIT replyAction->activated();
} else {
Q_EMIT replyAction->replied(text);
}
close(id);
}
}
}
void KNotificationManager::notificationClosed()
{
KNotification *notification = qobject_cast<KNotification *>(sender());
if (!notification) {
return;
}
// We cannot do d->notifications.find(notification->id()); here because the
// notification->id() is -1 or -2 at this point, so we need to look for value
for (auto iter = d->notifications.begin(); iter != d->notifications.end(); ++iter) {
if (iter.value() == notification) {
d->notifications.erase(iter);
break;
}
}
}
void KNotificationManager::close(int id)
{
if (d->notifications.contains(id)) {
KNotification *n = d->notifications.value(id);
qCDebug(LOG_KNOTIFICATIONS) << "Closing notification" << id;
// Find plugins that are actually acting on this notification
// call close() only on those, otherwise each KNotificationPlugin::close()
// will call finish() which may close-and-delete the KNotification object
// before it finishes calling close on all the other plugins.
// For example: Action=Popup is a single actions but there is 5 loaded
// plugins, calling close() on the second would already close-and-delete
// the notification
KNotifyConfig notifyConfig(n->appName(), n->eventId());
QString notifyActions = notifyConfig.readEntry(QStringLiteral("Action"));
const auto listActions = notifyActions.split(QLatin1Char('|'));
for (const QString &action : listActions) {
if (!d->notifyPlugins.contains(action)) {
qCDebug(LOG_KNOTIFICATIONS) << "No plugin for action" << action;
continue;
}
d->notifyPlugins[action]->close(n);
}
}
}
void KNotificationManager::notify(KNotification *n)
{
KNotifyConfig notifyConfig(n->appName(), n->eventId());
if (d->dirtyConfigCache.contains(n->appName())) {
notifyConfig.reparseSingleConfiguration(n->appName());
d->dirtyConfigCache.removeOne(n->appName());
}
if (!notifyConfig.isValid()) {
qCWarning(LOG_KNOTIFICATIONS) << "No event config could be found for event id" << n->eventId() << "under notifyrc file for app" << n->appName();
}
const QString notifyActions = notifyConfig.readEntry(QStringLiteral("Action"));
if (notifyActions.isEmpty() || notifyActions == QLatin1String("None")) {
// this will cause KNotification closing itself fast
n->ref();
n->deref();
return;
}
d->notifications.insert(n->id(), n);
// TODO KF6 d-pointer KNotifyConfig and add this there
if (n->urgency() == KNotification::DefaultUrgency) {
const QString urgency = notifyConfig.readEntry(QStringLiteral("Urgency"));
if (urgency == QLatin1String("Low")) {
n->setUrgency(KNotification::LowUrgency);
} else if (urgency == QLatin1String("Normal")) {
n->setUrgency(KNotification::NormalUrgency);
} else if (urgency == QLatin1String("High")) {
n->setUrgency(KNotification::HighUrgency);
} else if (urgency == QLatin1String("Critical")) {
n->setUrgency(KNotification::CriticalUrgency);
}
}
const auto actionsList = notifyActions.split(QLatin1Char('|'));
// Make sure all plugins can ref the notification
// otherwise a plugin may finish and deref before everyone got a chance to ref
for (const QString &action : actionsList) {
KNotificationPlugin *notifyPlugin = pluginForAction(action);
if (!notifyPlugin) {
qCDebug(LOG_KNOTIFICATIONS) << "No plugin for action" << action;
continue;
}
n->ref();
}
for (const QString &action : actionsList) {
KNotificationPlugin *notifyPlugin = pluginForAction(action);
if (!notifyPlugin) {
qCDebug(LOG_KNOTIFICATIONS) << "No plugin for action" << action;
continue;
}
qCDebug(LOG_KNOTIFICATIONS) << "Calling notify on" << notifyPlugin->optionName();
notifyPlugin->notify(n, notifyConfig);
}
connect(n, &KNotification::closed, this, &KNotificationManager::notificationClosed);
}
void KNotificationManager::update(KNotification *n)
{
KNotifyConfig notifyConfig(n->appName(), n->eventId());
for (KNotificationPlugin *p : std::as_const(d->notifyPlugins)) {
p->update(n, notifyConfig);
}
}
void KNotificationManager::reemit(KNotification *n)
{
notify(n);
}
void KNotificationManager::reparseConfiguration(const QString &app)
{
if (!d->dirtyConfigCache.contains(app)) {
d->dirtyConfigCache << app;
}
}
bool KNotificationManager::isInsideSandbox()
{
// logic is taken from KSandbox::isInside()
static const bool isFlatpak = QFileInfo::exists(QStringLiteral("/.flatpak-info"));
static const bool isSnap = qEnvironmentVariableIsSet("SNAP");
return isFlatpak || isSnap;
}
#include "moc_knotificationmanager_p.cpp"