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.
197 lines
6.5 KiB
C++
197 lines
6.5 KiB
C++
/**
|
|
* SPDX-FileCopyrightText: 2013 Albert Vaca <albertvaka@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
|
*/
|
|
|
|
#include "notification.h"
|
|
#include "plugin_notification_debug.h"
|
|
|
|
#include <KLocalizedString>
|
|
#include <KNotification>
|
|
#include <KNotificationReplyAction>
|
|
#include <QFile>
|
|
#include <QIcon>
|
|
#include <QJsonArray>
|
|
#include <QPixmap>
|
|
#include <QString>
|
|
#include <QUrl>
|
|
#include <QtGlobal>
|
|
#include <knotifications_version.h>
|
|
|
|
#include <core/filetransferjob.h>
|
|
#include <core/notificationserverinfo.h>
|
|
|
|
QMap<QString, FileTransferJob *> Notification::s_downloadsInProgress;
|
|
|
|
Notification::Notification(const NetworkPacket &np, const Device *device, QObject *parent)
|
|
: QObject(parent)
|
|
, m_imagesDir()
|
|
, m_device(device)
|
|
{
|
|
// Make a own directory for each user so no one can see each others icons
|
|
QString username;
|
|
#ifdef Q_OS_WIN
|
|
username = QString::fromLatin1(qgetenv("USERNAME"));
|
|
#else
|
|
username = QString::fromLatin1(qgetenv("USER"));
|
|
#endif
|
|
|
|
m_imagesDir.setPath(QDir::temp().absoluteFilePath(QStringLiteral("kdeconnect_") + username));
|
|
m_imagesDir.mkpath(m_imagesDir.absolutePath());
|
|
QFile(m_imagesDir.absolutePath()).setPermissions(QFileDevice::ReadOwner | QFileDevice::WriteOwner | QFileDevice::ExeOwner);
|
|
m_ready = false;
|
|
|
|
parseNetworkPacket(np);
|
|
createKNotification(np);
|
|
|
|
connect(m_notification, &KNotification::activated, this, [this](unsigned int actionIndex) {
|
|
// Notification action indices start at 1
|
|
Q_EMIT actionTriggered(m_internalId, m_actions[actionIndex - 1]);
|
|
});
|
|
}
|
|
|
|
void Notification::dismiss()
|
|
{
|
|
if (m_dismissable) {
|
|
Q_EMIT dismissRequested(m_internalId);
|
|
}
|
|
}
|
|
|
|
void Notification::show()
|
|
{
|
|
m_ready = true;
|
|
Q_EMIT ready();
|
|
if (!m_silent) {
|
|
m_notification->sendEvent();
|
|
}
|
|
}
|
|
|
|
void Notification::update(const NetworkPacket &np)
|
|
{
|
|
parseNetworkPacket(np);
|
|
createKNotification(np);
|
|
}
|
|
|
|
void Notification::createKNotification(const NetworkPacket &np)
|
|
{
|
|
if (!m_notification) {
|
|
m_notification = new KNotification(QStringLiteral("notification"), KNotification::CloseOnTimeout, this);
|
|
m_notification->setComponentName(QStringLiteral("kdeconnect"));
|
|
m_notification->setHint(QStringLiteral("resident"),
|
|
true); // This means the notification won't be deleted automatically, but only with KNotifications 5.81
|
|
}
|
|
|
|
QString escapedTitle = m_title.toHtmlEscaped();
|
|
// notification title text does not have markup, but in some cases below it is used in body text so we escape it
|
|
QString escapedText = m_text.toHtmlEscaped();
|
|
QString escapedTicker = m_ticker.toHtmlEscaped();
|
|
|
|
if (NotificationServerInfo::instance().supportedHints().testFlag(NotificationServerInfo::X_KDE_DISPLAY_APPNAME)) {
|
|
m_notification->setTitle(m_title);
|
|
m_notification->setText(escapedText);
|
|
m_notification->setHint(QStringLiteral("x-kde-display-appname"), m_appName.toHtmlEscaped());
|
|
} else {
|
|
m_notification->setTitle(m_appName);
|
|
|
|
if (m_title.isEmpty() && m_text.isEmpty()) {
|
|
m_notification->setText(escapedTicker);
|
|
} else if (m_appName == m_title) {
|
|
m_notification->setText(escapedText);
|
|
} else if (m_title.isEmpty()) {
|
|
m_notification->setText(escapedText);
|
|
} else if (m_text.isEmpty()) {
|
|
m_notification->setText(escapedTitle);
|
|
} else {
|
|
m_notification->setText(escapedTitle + QStringLiteral(": ") + escapedText);
|
|
}
|
|
}
|
|
|
|
m_notification->setHint(QStringLiteral("x-kde-origin-name"), m_device->name());
|
|
|
|
if (!m_requestReplyId.isEmpty()) {
|
|
auto replyAction = std::make_unique<KNotificationReplyAction>(i18nc("@action:button", "Reply"));
|
|
replyAction->setPlaceholderText(i18nc("@info:placeholder", "Reply to %1...", m_appName));
|
|
replyAction->setFallbackBehavior(KNotificationReplyAction::FallbackBehavior::UseRegularAction);
|
|
QObject::connect(replyAction.get(), &KNotificationReplyAction::replied, this, &Notification::replied);
|
|
QObject::connect(replyAction.get(), &KNotificationReplyAction::activated, this, &Notification::reply);
|
|
m_notification->setReplyAction(std::move(replyAction));
|
|
}
|
|
|
|
m_notification->setActions(m_actions);
|
|
|
|
m_hasIcon = m_hasIcon && !m_payloadHash.isEmpty();
|
|
|
|
if (!m_hasIcon) {
|
|
show();
|
|
} else {
|
|
m_iconPath = m_imagesDir.absoluteFilePath(m_payloadHash);
|
|
loadIcon(np);
|
|
}
|
|
}
|
|
|
|
void Notification::loadIcon(const NetworkPacket &np)
|
|
{
|
|
m_ready = false;
|
|
|
|
if (QFileInfo::exists(m_iconPath)) {
|
|
applyIcon();
|
|
show();
|
|
} else {
|
|
FileTransferJob *fileTransferJob = s_downloadsInProgress.value(m_iconPath);
|
|
if (!fileTransferJob) {
|
|
fileTransferJob = np.createPayloadTransferJob(QUrl::fromLocalFile(m_iconPath));
|
|
fileTransferJob->start();
|
|
s_downloadsInProgress[m_iconPath] = fileTransferJob;
|
|
}
|
|
|
|
connect(fileTransferJob, &FileTransferJob::result, this, [this, fileTransferJob] {
|
|
s_downloadsInProgress.remove(m_iconPath);
|
|
if (fileTransferJob->error()) {
|
|
qCDebug(KDECONNECT_PLUGIN_NOTIFICATION) << "Error in FileTransferJob: " << fileTransferJob->errorString();
|
|
} else {
|
|
applyIcon();
|
|
}
|
|
show();
|
|
});
|
|
}
|
|
}
|
|
|
|
void Notification::applyIcon()
|
|
{
|
|
QPixmap icon(m_iconPath, "PNG");
|
|
m_notification->setPixmap(icon);
|
|
}
|
|
|
|
void Notification::reply()
|
|
{
|
|
Q_EMIT replyRequested();
|
|
}
|
|
|
|
void Notification::sendReply(const QString &message)
|
|
{
|
|
Q_EMIT replied(message);
|
|
}
|
|
|
|
void Notification::parseNetworkPacket(const NetworkPacket &np)
|
|
{
|
|
m_internalId = np.get<QString>(QStringLiteral("id"));
|
|
m_appName = np.get<QString>(QStringLiteral("appName"));
|
|
m_ticker = np.get<QString>(QStringLiteral("ticker"));
|
|
m_title = np.get<QString>(QStringLiteral("title"));
|
|
m_text = np.get<QString>(QStringLiteral("text"));
|
|
m_dismissable = np.get<bool>(QStringLiteral("isClearable"));
|
|
m_hasIcon = np.hasPayload();
|
|
m_silent = np.get<bool>(QStringLiteral("silent"));
|
|
m_payloadHash = np.get<QString>(QStringLiteral("payloadHash"));
|
|
m_requestReplyId = np.get<QString>(QStringLiteral("requestReplyId"), QString());
|
|
|
|
m_actions.clear();
|
|
|
|
const auto actions = np.get<QJsonArray>(QStringLiteral("actions"));
|
|
for (const QJsonValue &value : actions) {
|
|
m_actions.append(value.toString());
|
|
}
|
|
}
|
|
|
|
#include "moc_notification.cpp"
|