kdeconnect-kde/plugins/notifications/notification.cpp
Bharadwaj Raju 1f5fd06924 [plasmoid] Add inline reply for notifications instead of opening a dialog
Instead of opening the reply dialog when clicking Reply in the plasmoid, it opens an inline reply UI.
2022-09-26 19:51:28 +00:00

214 lines
7.1 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 "knotifications_version.h"
#include <KNotification>
#if KNOTIFICATIONS_VERSION >= QT_VERSION_CHECK(5, 81, 0)
#include <KNotificationReplyAction>
#endif
#include <KLocalizedString>
#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, QOverload<unsigned int>::of(&KNotification::activated), this, [this](unsigned int actionIndex) {
// Since 5.81 we use KNotification's inline reply instead of our own action
#if KNOTIFICATIONS_VERSION < QT_VERSION_CHECK(5, 81, 0)
// Do nothing for our own reply action
if (!m_requestReplyId.isEmpty() && actionIndex == 1) {
return;
}
#endif
// Notification action indices start at 1
Q_EMIT actionTriggered(m_internalId, m_actions[actionIndex - 1]);
});
}
Notification::~Notification()
{
}
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()) {
#if KNOTIFICATIONS_VERSION >= QT_VERSION_CHECK(5, 81, 0)
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));
#else
m_actions.prepend(i18n("Reply"));
connect(m_notification, &KNotification::action1Activated, this, &Notification::reply, Qt::UniqueConnection);
#endif
}
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());
}
}