2e67f95017
The rationale is explained in https://planet.kde.org/friedrich-kossebau-2023-06-28-include-also-moc-files-of-headers/ In case of KDEConnect, it impressively speeds up compilation. Before it took 390 seconds on a clean build and with this change it took 330 seconds. This is due to the mocs_compilation having to include the header files and thus all their headers. Due to the lots of small plugins we have, this means that the same headers must be compiled plenty of times. When we include the moc files directly in the C++ file, they are already available.
352 lines
14 KiB
C++
352 lines
14 KiB
C++
/**
|
|
* SPDX-FileCopyrightText: 2015 Holger Kaelberer <holger.k@elberer.de>
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
|
*/
|
|
#include "notificationslistener.h"
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <KConfig>
|
|
#include <KConfigGroup>
|
|
#include <QDBusArgument>
|
|
#include <QDBusInterface>
|
|
#include <QImage>
|
|
#include <QScopeGuard>
|
|
#include <QStandardPaths>
|
|
#include <QtDebug>
|
|
|
|
#include <kiconloader.h>
|
|
#include <kicontheme.h>
|
|
|
|
#include <core/device.h>
|
|
#include <core/kdeconnectplugin.h>
|
|
|
|
#include <dbushelper.h>
|
|
|
|
#include "notifyingapplication.h"
|
|
#include "plugin_sendnotification_debug.h"
|
|
#include "sendnotificationsplugin.h"
|
|
|
|
NotificationsListener::NotificationsListener(KdeConnectPlugin *aPlugin)
|
|
: QObject(aPlugin)
|
|
, m_plugin(aPlugin)
|
|
{
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
qRegisterMetaTypeStreamOperators<NotifyingApplication>("NotifyingApplication");
|
|
#endif
|
|
|
|
GError *error = nullptr;
|
|
m_gdbusConnection = g_bus_get_sync(G_BUS_TYPE_SESSION, nullptr, &error);
|
|
g_assert_no_error(error);
|
|
m_gdbusFilterId = g_dbus_connection_add_filter(m_gdbusConnection, NotificationsListener::onMessageFiltered, this, nullptr);
|
|
|
|
g_autoptr(GDBusMessage) msg =
|
|
g_dbus_message_new_method_call("org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus.Monitoring", "BecomeMonitor");
|
|
|
|
GVariantBuilder *arrayBuilder = g_variant_builder_new(G_VARIANT_TYPE("as"));
|
|
g_variant_builder_add(arrayBuilder, "s", "interface='org.freedesktop.Notifications'");
|
|
g_variant_builder_add(arrayBuilder, "s", "member='Notify'");
|
|
|
|
g_dbus_message_set_body(msg, g_variant_new("(asu)", arrayBuilder, 0u));
|
|
g_dbus_connection_send_message(m_gdbusConnection, msg, GDBusSendMessageFlags::G_DBUS_SEND_MESSAGE_FLAGS_NONE, nullptr, &error);
|
|
g_assert_no_error(error);
|
|
|
|
setTranslatedAppName();
|
|
loadApplications();
|
|
|
|
connect(m_plugin->config(), &KdeConnectPluginConfig::configChanged, this, &NotificationsListener::loadApplications);
|
|
}
|
|
|
|
NotificationsListener::~NotificationsListener()
|
|
{
|
|
qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Destroying NotificationsListener";
|
|
g_dbus_connection_remove_filter(m_gdbusConnection, m_gdbusFilterId);
|
|
g_object_unref(m_gdbusConnection);
|
|
}
|
|
|
|
void NotificationsListener::setTranslatedAppName()
|
|
{
|
|
QString filePath =
|
|
QStandardPaths::locate(QStandardPaths::GenericDataLocation, QStringLiteral("knotifications5/kdeconnect.notifyrc"), QStandardPaths::LocateFile);
|
|
if (filePath.isEmpty()) {
|
|
qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Couldn't find kdeconnect.notifyrc to hide kdeconnect notifications on the devices. Using default name.";
|
|
m_translatedAppName = QStringLiteral("KDE Connect");
|
|
return;
|
|
}
|
|
|
|
KConfig config(filePath, KConfig::OpenFlag::SimpleConfig);
|
|
KConfigGroup globalgroup(&config, QStringLiteral("Global"));
|
|
m_translatedAppName = globalgroup.readEntry(QStringLiteral("Name"), QStringLiteral("KDE Connect"));
|
|
}
|
|
|
|
void NotificationsListener::loadApplications()
|
|
{
|
|
m_applications.clear();
|
|
const QVariantList list = m_plugin->config()->getList(QStringLiteral("applications"));
|
|
for (const auto &a : list) {
|
|
NotifyingApplication app = a.value<NotifyingApplication>();
|
|
if (!m_applications.contains(app.name)) {
|
|
m_applications.insert(app.name, app);
|
|
}
|
|
}
|
|
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Loaded" << applications.size() << " applications";
|
|
}
|
|
|
|
bool NotificationsListener::parseImageDataArgument(GVariant *argument,
|
|
int &width,
|
|
int &height,
|
|
int &rowStride,
|
|
int &bitsPerSample,
|
|
int &channels,
|
|
bool &hasAlpha,
|
|
QByteArray &imageData) const
|
|
{
|
|
if (g_variant_n_children(argument) != 7) {
|
|
return false;
|
|
}
|
|
|
|
g_autoptr(GVariant) variant;
|
|
|
|
variant = g_variant_get_child_value(argument, 0);
|
|
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
|
|
return false;
|
|
}
|
|
width = g_variant_get_int32(variant);
|
|
|
|
variant = g_variant_get_child_value(argument, 1);
|
|
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
|
|
return false;
|
|
}
|
|
height = g_variant_get_int32(variant);
|
|
|
|
variant = g_variant_get_child_value(argument, 2);
|
|
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
|
|
return false;
|
|
}
|
|
rowStride = g_variant_get_int32(variant);
|
|
|
|
variant = g_variant_get_child_value(argument, 3);
|
|
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_BOOLEAN)) {
|
|
return false;
|
|
}
|
|
hasAlpha = g_variant_get_boolean(variant);
|
|
|
|
variant = g_variant_get_child_value(argument, 4);
|
|
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
|
|
return false;
|
|
}
|
|
bitsPerSample = g_variant_get_int32(variant);
|
|
|
|
variant = g_variant_get_child_value(argument, 5);
|
|
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_INT32)) {
|
|
return false;
|
|
}
|
|
channels = g_variant_get_int32(variant);
|
|
|
|
variant = g_variant_get_child_value(argument, 6);
|
|
if (g_variant_is_of_type(variant, G_VARIANT_TYPE_ARRAY)) {
|
|
return false;
|
|
}
|
|
imageData = g_variant_get_bytestring(variant);
|
|
|
|
return true;
|
|
}
|
|
|
|
QSharedPointer<QIODevice> NotificationsListener::iconForImageData(GVariant *argument) const
|
|
{
|
|
int width, height, rowStride, bitsPerSample, channels;
|
|
bool hasAlpha;
|
|
QByteArray imageData;
|
|
|
|
if (!parseImageDataArgument(argument, width, height, rowStride, bitsPerSample, channels, hasAlpha, imageData))
|
|
return QSharedPointer<QIODevice>();
|
|
|
|
if (bitsPerSample != 8) {
|
|
qCWarning(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Unsupported image format:"
|
|
<< "width=" << width << "height=" << height << "rowStride=" << rowStride
|
|
<< "bitsPerSample=" << bitsPerSample << "channels=" << channels << "hasAlpha=" << hasAlpha;
|
|
return QSharedPointer<QIODevice>();
|
|
}
|
|
|
|
QImage image(reinterpret_cast<uchar *>(imageData.data()), width, height, rowStride, hasAlpha ? QImage::Format_ARGB32 : QImage::Format_RGB32);
|
|
if (hasAlpha)
|
|
image = image.rgbSwapped(); // RGBA --> ARGB
|
|
|
|
QSharedPointer<QBuffer> buffer = QSharedPointer<QBuffer>(new QBuffer);
|
|
if (!buffer || !buffer->open(QIODevice::WriteOnly) || !image.save(buffer.data(), "PNG")) {
|
|
qCWarning(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Could not initialize image buffer";
|
|
return QSharedPointer<QIODevice>();
|
|
}
|
|
|
|
return buffer;
|
|
}
|
|
|
|
QSharedPointer<QIODevice> NotificationsListener::iconForIconName(const QString &iconName)
|
|
{
|
|
int size = KIconLoader::SizeEnormous; // use big size to allow for good
|
|
// quality on high-DPI mobile devices
|
|
QString iconPath = KIconLoader::global()->iconPath(iconName, -size, true);
|
|
if (!iconPath.isEmpty()) {
|
|
if (!iconPath.endsWith(QLatin1String(".png")) && KIconLoader::global()->theme()->name() != QLatin1String("hicolor")) {
|
|
// try falling back to hicolor theme:
|
|
KIconTheme hicolor(QStringLiteral("hicolor"));
|
|
if (hicolor.isValid()) {
|
|
iconPath = hicolor.iconPath(iconName + QStringLiteral(".png"), size, KIconLoader::MatchBest);
|
|
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Found non-png icon in default theme trying fallback to hicolor:" << iconPath;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (iconPath.endsWith(QLatin1String(".png")))
|
|
return QSharedPointer<QIODevice>(new QFile(iconPath));
|
|
return QSharedPointer<QIODevice>();
|
|
}
|
|
|
|
GDBusMessage *NotificationsListener::onMessageFiltered(GDBusConnection *, GDBusMessage *msg, int, void *parent)
|
|
{
|
|
static unsigned id = 0;
|
|
if (!msg) {
|
|
return msg;
|
|
}
|
|
|
|
const gchar *interface = g_dbus_message_get_interface(msg);
|
|
if (!interface || strcmp(interface, "org.freedesktop.Notifications")) {
|
|
// The first message will be from org.freedesktop.DBus.Monitoring
|
|
return msg;
|
|
}
|
|
const gchar *member = g_dbus_message_get_member(msg);
|
|
if (!member || strcmp(member, "Notify")) {
|
|
// Even if member is set, the monitor will still notify messages from other members.
|
|
return nullptr;
|
|
}
|
|
|
|
g_autoptr(GVariant) bodyVariant = g_dbus_message_get_body(msg);
|
|
Q_ASSERT(bodyVariant);
|
|
|
|
// Data order and types: https://specifications.freedesktop.org/notification-spec/notification-spec-latest.html
|
|
g_autoptr(GVariant) variant = g_variant_get_child_value(bodyVariant, 0);
|
|
const QString appName = QString::fromUtf8(g_variant_get_string(variant, nullptr));
|
|
// skip our own notifications
|
|
auto listener = static_cast<NotificationsListener *>(parent);
|
|
|
|
if (appName == listener->m_translatedAppName) {
|
|
return nullptr;
|
|
}
|
|
|
|
variant = g_variant_get_child_value(bodyVariant, 2);
|
|
const QString appIcon = QString::fromUtf8(g_variant_get_string(variant, nullptr));
|
|
|
|
NotifyingApplication app;
|
|
if (!listener->m_applications.contains(appName)) {
|
|
// new application -> add to config
|
|
app.name = appName;
|
|
app.icon = appIcon;
|
|
app.active = true;
|
|
app.blacklistExpression = QRegularExpression();
|
|
listener->m_applications.insert(app.name, app);
|
|
// update config:
|
|
QVariantList list;
|
|
for (const auto &a : std::as_const(listener->m_applications))
|
|
list << QVariant::fromValue<NotifyingApplication>(a);
|
|
listener->m_plugin->config()->setList(QStringLiteral("applications"), list);
|
|
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Added new application to config:" << app;
|
|
} else {
|
|
app = listener->m_applications.value(appName);
|
|
}
|
|
|
|
if (!app.active) {
|
|
return nullptr;
|
|
}
|
|
|
|
variant = g_variant_get_child_value(bodyVariant, 7);
|
|
const auto timeout = g_variant_get_int32(variant);
|
|
if (timeout > 0 && listener->m_plugin->config()->getBool(QStringLiteral("generalPersistent"), false)) {
|
|
return nullptr;
|
|
}
|
|
|
|
variant = g_variant_get_child_value(bodyVariant, 6);
|
|
std::unordered_map<QString, GVariant *> hints;
|
|
GVariantIter *iter;
|
|
const gchar *key;
|
|
GVariant *value = nullptr;
|
|
g_variant_get(variant, "a{sv}", &iter);
|
|
while (g_variant_iter_loop(iter, "{sv}", &key, &value)) {
|
|
hints.emplace(QString::fromUtf8(key), value);
|
|
}
|
|
g_variant_iter_free(iter);
|
|
|
|
QScopeGuard cleanupHints([&hints] {
|
|
for (auto &[_, hint] : hints) {
|
|
g_variant_unref(hint);
|
|
}
|
|
});
|
|
|
|
int urgency = -1;
|
|
if (auto it = hints.find(QStringLiteral("urgency")); it != hints.end()) {
|
|
if (g_variant_is_of_type(it->second, G_VARIANT_TYPE_BYTE)) {
|
|
urgency = g_variant_get_byte(it->second);
|
|
}
|
|
}
|
|
if (urgency > -1 && urgency < listener->m_plugin->config()->getInt(QStringLiteral("generalUrgency"), 0))
|
|
return nullptr;
|
|
|
|
variant = g_variant_get_child_value(bodyVariant, 3);
|
|
QString ticker = QString::fromUtf8(g_variant_get_string(variant, nullptr));
|
|
|
|
variant = g_variant_get_child_value(bodyVariant, 4);
|
|
const QString body = QString::fromUtf8(g_variant_get_string(variant, nullptr));
|
|
|
|
if (!body.isEmpty() && listener->m_plugin->config()->getBool(QStringLiteral("generalIncludeBody"), true)) {
|
|
ticker += QStringLiteral(": ") + body;
|
|
}
|
|
|
|
if (app.blacklistExpression.isValid() && !app.blacklistExpression.pattern().isEmpty() && app.blacklistExpression.match(ticker).hasMatch()) {
|
|
return nullptr;
|
|
}
|
|
|
|
variant = g_variant_get_child_value(bodyVariant, 1);
|
|
const unsigned replacesId = g_variant_get_uint32(variant);
|
|
|
|
// qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Sending notification from" << appName << ":" <<ticker << "; appIcon=" << appIcon;
|
|
NetworkPacket np(PACKET_TYPE_NOTIFICATION,
|
|
{{QStringLiteral("id"), QString::number(replacesId > 0 ? replacesId : ++id)},
|
|
{QStringLiteral("appName"), appName},
|
|
{QStringLiteral("ticker"), ticker},
|
|
{QStringLiteral("isClearable"), timeout == 0}}); // KNotifications are persistent if
|
|
// timeout == 0, for other notifications
|
|
// clearability is pointless
|
|
|
|
// sync any icon data?
|
|
if (listener->m_plugin->config()->getBool(QStringLiteral("generalSynchronizeIcons"), true)) {
|
|
QSharedPointer<QIODevice> iconSource;
|
|
// try different image sources according to priorities in notifications-
|
|
// spec version 1.2:
|
|
if (auto it = hints.find(QStringLiteral("image-data")); it != hints.end()) {
|
|
iconSource = listener->iconForImageData(it->second);
|
|
} else if (auto it = hints.find(QStringLiteral("image_data")); it != hints.end()) { // 1.1 backward compatibility
|
|
iconSource = listener->iconForImageData(it->second);
|
|
} else if (auto it = hints.find(QStringLiteral("image-path")); it != hints.end()) {
|
|
iconSource = iconForIconName(QString::fromUtf8(g_variant_get_string(it->second, nullptr)));
|
|
} else if (auto it = hints.find(QStringLiteral("image_path")); it != hints.end()) { // 1.1 backward compatibility
|
|
iconSource = iconForIconName(QString::fromUtf8(g_variant_get_string(it->second, nullptr)));
|
|
} else if (!appIcon.isEmpty()) {
|
|
iconSource = iconForIconName(appIcon);
|
|
} else if (auto it = hints.find(QStringLiteral("icon_data")); it != hints.end()) { // < 1.1 backward compatibility
|
|
iconSource = listener->iconForImageData(it->second);
|
|
}
|
|
|
|
if (iconSource)
|
|
np.setPayload(iconSource, iconSource->size());
|
|
}
|
|
|
|
listener->m_plugin->sendPacket(np);
|
|
|
|
qCDebug(KDECONNECT_PLUGIN_SENDNOTIFICATION) << "Got notification appName=" << appName << "replacesId=" << replacesId << "appIcon=" << appIcon
|
|
<< "summary=" << ticker << "body=" << body << "hints=" << hints.size() << "urgency=" << urgency
|
|
<< "timeout=" << timeout;
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
#include "moc_notificationslistener.cpp"
|