nheko/src/notifications/ManagerLinux.cpp

306 lines
12 KiB
C++
Raw Normal View History

2018-05-05 21:40:24 +02:00
#include "notifications/Manager.h"
2021-01-20 00:30:04 +01:00
#include <QDBusConnection>
#include <QDBusMessage>
#include <QDBusMetaType>
#include <QDBusPendingCallWatcher>
#include <QDBusPendingReply>
2021-01-20 00:47:44 +01:00
#include <QDebug>
#include <QImage>
#include <QRegularExpression>
#include <QTextDocumentFragment>
#include <functional>
2021-02-26 21:33:27 +01:00
#include <variant>
#include <mtx/responses/notifications.hpp>
#include "Cache.h"
#include "EventAccessors.h"
#include "Utils.h"
2018-07-14 15:27:51 +02:00
NotificationsManager::NotificationsManager(QObject *parent)
: QObject(parent)
, dbus("org.freedesktop.Notifications",
"/org/freedesktop/Notifications",
"org.freedesktop.Notifications",
QDBusConnection::sessionBus(),
this)
, hasMarkup_{std::invoke([this]() -> bool {
for (auto x : dbus.call("GetCapabilities").arguments())
if (x.toStringList().contains("body-markup"))
return true;
return false;
})}
, hasImages_{std::invoke([this]() -> bool {
for (auto x : dbus.call("GetCapabilities").arguments())
if (x.toStringList().contains("body-images"))
return true;
return false;
})}
{
qDBusRegisterMetaType<QImage>();
2018-07-14 15:27:51 +02:00
QDBusConnection::sessionBus().connect("org.freedesktop.Notifications",
"/org/freedesktop/Notifications",
"org.freedesktop.Notifications",
"ActionInvoked",
this,
SLOT(actionInvoked(uint, QString)));
QDBusConnection::sessionBus().connect("org.freedesktop.Notifications",
"/org/freedesktop/Notifications",
"org.freedesktop.Notifications",
"NotificationClosed",
this,
SLOT(notificationClosed(uint, uint)));
QDBusConnection::sessionBus().connect("org.freedesktop.Notifications",
"/org/freedesktop/Notifications",
"org.freedesktop.Notifications",
"NotificationReplied",
this,
SLOT(notificationReplied(uint, QString)));
}
void
NotificationsManager::postNotification(const mtx::responses::Notification &notification,
const QImage &icon)
{
const auto room_id = QString::fromStdString(notification.room_id);
const auto event_id = QString::fromStdString(mtx::accessors::event_id(notification.event));
const auto room_name =
QString::fromStdString(cache::singleRoomInfo(notification.room_id).name);
const auto text = formatNotification(notification);
systemPostNotification(room_id, event_id, room_name, text, icon);
}
2021-03-05 00:35:15 +01:00
/**
* This function is based on code from
* https://github.com/rohieb/StratumsphereTrayIcon
* Copyright (C) 2012 Roland Hieber <rohieb@rohieb.name>
* Licensed under the GNU General Public License, version 3
*/
void
NotificationsManager::systemPostNotification(const QString &room_id,
const QString &event_id,
const QString &roomName,
const QString &text,
const QImage &icon)
{
2021-01-20 00:47:44 +01:00
QVariantMap hints;
2021-01-20 22:15:14 +01:00
hints["image-data"] = icon;
2021-01-20 00:47:44 +01:00
hints["sound-name"] = "message-new-instant";
QList<QVariant> argumentList;
argumentList << "nheko"; // app_name
argumentList << (uint)0; // replace_id
argumentList << ""; // app_icon
argumentList << roomName; // summary
argumentList << text; // body
2021-01-20 00:47:44 +01:00
// The list of actions has always the action name and then a localized version of that
// action. Currently we just use an empty string for that.
// TODO(Nico): Look into what to actually put there.
argumentList << (QStringList("default") << ""
<< "inline-reply"
<< ""); // actions
argumentList << hints; // hints
argumentList << (int)-1; // timeout in ms
2021-01-20 00:30:04 +01:00
2021-02-15 22:52:19 +01:00
QDBusPendingCall call = dbus.asyncCallWithArgumentList("Notify", argumentList);
2021-01-20 23:59:27 +01:00
auto watcher = new QDBusPendingCallWatcher{call, this};
2021-01-20 22:09:25 +01:00
connect(
watcher, &QDBusPendingCallWatcher::finished, this, [watcher, this, room_id, event_id]() {
2021-01-20 22:09:25 +01:00
if (watcher->reply().type() == QDBusMessage::ErrorMessage)
qDebug() << "D-Bus Error:" << watcher->reply().errorMessage();
else
notificationIds[watcher->reply().arguments().first().toUInt()] =
roomEventId{room_id, event_id};
2021-01-20 22:13:21 +01:00
watcher->deleteLater();
2021-01-20 22:09:25 +01:00
});
}
2021-01-20 00:30:04 +01:00
2020-04-09 20:52:50 +02:00
void
NotificationsManager::closeNotification(uint id)
{
2021-02-15 22:52:19 +01:00
auto call = dbus.asyncCall("CloseNotification", (uint)id); // replace_id
2021-01-20 23:59:27 +01:00
auto watcher = new QDBusPendingCallWatcher{call, this};
2021-02-23 12:42:57 +01:00
connect(watcher, &QDBusPendingCallWatcher::finished, this, [watcher]() {
2021-01-20 22:09:25 +01:00
if (watcher->reply().type() == QDBusMessage::ErrorMessage) {
qDebug() << "D-Bus Error:" << watcher->reply().errorMessage();
};
2021-01-20 22:13:21 +01:00
watcher->deleteLater();
2021-01-20 22:09:25 +01:00
});
2020-04-09 20:52:50 +02:00
}
void
NotificationsManager::removeNotification(const QString &roomId, const QString &eventId)
2020-04-13 16:22:30 +02:00
{
roomEventId reId = {roomId, eventId};
for (auto elem = notificationIds.begin(); elem != notificationIds.end(); ++elem) {
if (elem.value().roomId != roomId)
continue;
2020-04-09 20:52:50 +02:00
2020-04-13 16:22:30 +02:00
// close all notifications matching the eventId or having a lower
// notificationId
// This relies on the notificationId not wrapping around. This allows for
// approximately 2,147,483,647 notifications, so it is a bit unlikely.
// Otherwise we would need to store a 64bit counter instead.
closeNotification(elem.key());
2020-04-09 20:52:50 +02:00
2020-04-13 16:22:30 +02:00
// FIXME: compare index of event id of the read receipt and the notification instead
// of just the id to prevent read receipts of events without notification clearing
// all notifications in that room!
if (elem.value() == reId)
break;
2020-04-09 20:52:50 +02:00
}
}
2018-05-05 21:40:24 +02:00
void
NotificationsManager::actionInvoked(uint id, QString action)
2018-05-05 21:40:24 +02:00
{
if (notificationIds.contains(id)) {
roomEventId idEntry = notificationIds[id];
if (action == "default") {
emit notificationClicked(idEntry.roomId, idEntry.eventId);
}
}
}
void
NotificationsManager::notificationReplied(uint id, QString reply)
{
if (notificationIds.contains(id)) {
roomEventId idEntry = notificationIds[id];
emit sendNotificationReply(idEntry.roomId, idEntry.eventId, reply);
}
}
void
NotificationsManager::notificationClosed(uint id, uint reason)
{
Q_UNUSED(reason);
notificationIds.remove(id);
}
/**
* @param text This should be an HTML-formatted string.
*
* If D-Bus says that notifications can have body markup, this function will
* automatically format the notification to follow the supported HTML subset
* specified at https://www.freedesktop.org/wiki/Specifications/StatusNotifierItem/Markup/
*/
QString
NotificationsManager::formatNotification(const mtx::responses::Notification &notification)
{
const auto sender =
cache::displayName(QString::fromStdString(notification.room_id),
QString::fromStdString(mtx::accessors::sender(notification.event)));
2021-02-26 21:33:27 +01:00
// TODO: decrypt this message if the decryption setting is on in the UserSettings
if (auto msg = std::get_if<mtx::events::EncryptedEvent<mtx::events::msg::Encrypted>>(
&notification.event);
msg != nullptr)
return tr("%1 sent an encrypted message").arg(sender);
const auto messageLeadIn =
((mtx::accessors::msg_type(notification.event) == mtx::events::MessageType::Emote)
? "* " + sender + " "
: sender +
(utils::isReply(notification.event)
? tr(" replied",
"Used to denote that this message is a reply to another "
"message. Displayed as 'foo replied: message'.")
: "") +
": ");
if (hasMarkup_) {
if (hasImages_ &&
mtx::accessors::msg_type(notification.event) == mtx::events::MessageType::Image)
return QString(
"<img src=\"file:///" + cacheImage(notification.event) +
"\" alt=\"" +
mtx::accessors::formattedBodyWithFallback(notification.event) +
"\">")
.prepend(messageLeadIn);
return mtx::accessors::formattedBodyWithFallback(notification.event)
.prepend(messageLeadIn)
.replace("<em>", "<i>")
.replace("</em>", "</i>")
.replace("<strong>", "<b>")
.replace("</strong>", "</b>")
.replace(QRegularExpression("(<mx-reply>.+\\<\\/mx-reply\\>)"), "");
}
2021-02-20 02:13:27 +01:00
return QTextDocumentFragment::fromHtml(
mtx::accessors::formattedBodyWithFallback(notification.event)
.replace(QRegularExpression("(<mx-reply>.+\\<\\/mx-reply\\>)"), ""))
.toPlainText()
.prepend(messageLeadIn);
}
/**
* Automatic marshaling of a QImage for org.freedesktop.Notifications.Notify
*
* This function is from the Clementine project (see
* http://www.clementine-player.org) and licensed under the GNU General Public
* License, version 3 or later.
*
2021-03-05 00:35:15 +01:00
* SPDX-FileCopyrightText: 2010 David Sansome <me@davidsansome.com>
*/
2018-07-14 15:27:51 +02:00
QDBusArgument &
operator<<(QDBusArgument &arg, const QImage &image)
{
if (image.isNull()) {
arg.beginStructure();
arg << 0 << 0 << 0 << false << 0 << 0 << QByteArray();
arg.endStructure();
return arg;
}
2018-07-14 15:27:51 +02:00
QImage scaled = image.scaledToHeight(100, Qt::SmoothTransformation);
scaled = scaled.convertToFormat(QImage::Format_ARGB32);
#if Q_BYTE_ORDER == Q_LITTLE_ENDIAN
2018-07-14 15:27:51 +02:00
// ABGR -> ARGB
QImage i = scaled.rgbSwapped();
#else
2018-07-14 15:27:51 +02:00
// ABGR -> GBAR
QImage i(scaled.size(), scaled.format());
for (int y = 0; y < i.height(); ++y) {
QRgb *p = (QRgb *)scaled.scanLine(y);
QRgb *q = (QRgb *)i.scanLine(y);
QRgb *end = p + scaled.width();
while (p < end) {
*q = qRgba(qGreen(*p), qBlue(*p), qAlpha(*p), qRed(*p));
p++;
q++;
}
}
#endif
2018-07-14 15:27:51 +02:00
arg.beginStructure();
arg << i.width();
arg << i.height();
arg << i.bytesPerLine();
arg << i.hasAlphaChannel();
int channels = i.isGrayscale() ? 1 : (i.hasAlphaChannel() ? 4 : 3);
arg << i.depth() / channels;
arg << channels;
2019-07-05 22:31:01 +02:00
#if QT_VERSION < QT_VERSION_CHECK(5, 10, 0)
arg << QByteArray(reinterpret_cast<const char *>(i.bits()), i.byteCount());
#else
arg << QByteArray(reinterpret_cast<const char *>(i.bits()), i.sizeInBytes());
2019-07-05 22:31:01 +02:00
#endif
2018-07-14 15:27:51 +02:00
arg.endStructure();
return arg;
}
2018-07-14 15:27:51 +02:00
const QDBusArgument &
operator>>(const QDBusArgument &arg, QImage &)
{
// This is needed to link but shouldn't be called.
Q_ASSERT(0);
return arg;
2018-05-05 21:40:24 +02:00
}