summary refs log tree commit diff
path: root/src/notifications
diff options
context:
space:
mode:
authorNicolas Werner <nicolas.werner@hotmail.de>2020-04-09 20:52:50 +0200
committerNicolas Werner <nicolas.werner@hotmail.de>2020-04-11 01:09:47 +0200
commit2022775dd052b6b1a1a82c1f418e892e40dd4fc8 (patch)
treeea134a0cda5e0a8b7fa24476b311118b4ed682b4 /src/notifications
parentUse Item for transparent rectangles (diff)
downloadnheko-2022775dd052b6b1a1a82c1f418e892e40dd4fc8.tar.xz
Clear notifications when event is read
Diffstat (limited to 'src/notifications')
-rw-r--r--src/notifications/Manager.h30
-rw-r--r--src/notifications/ManagerLinux.cpp42
-rw-r--r--src/notifications/ManagerMac.mm5
-rw-r--r--src/notifications/ManagerWin.cpp5
4 files changed, 81 insertions, 1 deletions
diff --git a/src/notifications/Manager.h b/src/notifications/Manager.h

index 56541ece..6cbecbc6 100644 --- a/src/notifications/Manager.h +++ b/src/notifications/Manager.h
@@ -1,5 +1,6 @@ #pragma once +#include <QHash> #include <QImage> #include <QObject> #include <QString> @@ -15,6 +16,27 @@ struct roomEventId QString eventId; }; +inline bool +operator<(const roomEventId &a, const roomEventId &b) +{ + if (a.roomId == b.roomId) + return a.eventId < b.eventId; + else + return a.roomId < b.roomId; +} + +inline bool +operator==(const roomEventId &a, const roomEventId &b) +{ + return a.roomId == b.roomId && a.eventId == b.eventId; +} + +inline uint +qHash(const roomEventId &v, uint seed) +{ + return qHash(v.roomId, seed) ^ qHash(v.eventId, seed); +} + class NotificationsManager : public QObject { Q_OBJECT @@ -31,13 +53,21 @@ public: signals: void notificationClicked(const QString roomId, const QString eventId); +public slots: + void removeNotification(const QString &roomId, const QString &eventId); + #if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD) +public: + void closeNotifications(QString roomId); + private: QDBusInterface dbus; uint showNotification(const QString summary, const QString text, const QImage image); + void closeNotification(uint id); // notification ID to (room ID, event ID) QMap<uint, roomEventId> notificationIds; + QHash<roomEventId, uint> eventToNotificationId; #endif // these slots are platform specific (D-Bus only) diff --git a/src/notifications/ManagerLinux.cpp b/src/notifications/ManagerLinux.cpp
index 1914f61c..33fcb41d 100644 --- a/src/notifications/ManagerLinux.cpp +++ b/src/notifications/ManagerLinux.cpp
@@ -40,6 +40,7 @@ NotificationsManager::postNotification(const QString &roomid, { uint id = showNotification(roomname, sender + ": " + text, icon); notificationIds[id] = roomEventId{roomid, eventid}; + eventToNotificationId[roomEventId{roomid, eventid}] = id; } /** * This function is based on code from @@ -54,6 +55,7 @@ NotificationsManager::showNotification(const QString summary, { QVariantMap hints; hints["image-data"] = image; + hints["sound-name"] = "message-new-instant"; QList<QVariant> argumentList; argumentList << "nheko"; // app_name argumentList << (uint)0; // replace_id @@ -79,6 +81,44 @@ NotificationsManager::showNotification(const QString summary, } void +NotificationsManager::closeNotification(uint id) +{ + QList<QVariant> argumentList; + argumentList << (uint)id; // replace_id + + static QDBusInterface closeCall("org.freedesktop.Notifications", + "/org/freedesktop/Notifications", + "org.freedesktop.Notifications"); + QDBusMessage reply = + closeCall.callWithArgumentList(QDBus::AutoDetect, "CloseNotification", argumentList); + if (reply.type() == QDBusMessage::ErrorMessage) { + qDebug() << "D-Bus Error:" << reply.errorMessage(); + } +} + +void +NotificationsManager::removeNotification(const QString &roomId, const QString &eventId) +{ + roomEventId reId = {roomId, eventId}; + if (eventToNotificationId.contains(reId)) { + for (auto elem = notificationIds.begin(); elem != notificationIds.end(); ++elem) { + if (elem.value().roomId != roomId) + continue; + + // 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()); + + if (elem.value() == reId) + break; + } + } +} + +void NotificationsManager::actionInvoked(uint id, QString action) { if (action == "default" && notificationIds.contains(id)) { @@ -91,7 +131,7 @@ void NotificationsManager::notificationClosed(uint id, uint reason) { Q_UNUSED(reason); - notificationIds.remove(id); + eventToNotificationId.remove(notificationIds.take(id)); } /** diff --git a/src/notifications/ManagerMac.mm b/src/notifications/ManagerMac.mm
index 66ef713f..e46b0b82 100644 --- a/src/notifications/ManagerMac.mm +++ b/src/notifications/ManagerMac.mm
@@ -46,3 +46,8 @@ void NotificationsManager::notificationClosed(uint, uint) { } + +void +NotificationsManager::removeNotification(const QString &roomId, const QString &eventId) +{} + diff --git a/src/notifications/ManagerWin.cpp b/src/notifications/ManagerWin.cpp
index 9cc4da9b..b00bac2e 100644 --- a/src/notifications/ManagerWin.cpp +++ b/src/notifications/ManagerWin.cpp
@@ -63,3 +63,8 @@ NotificationsManager::postNotification(const QString &room_id, void NotificationsManager::actionInvoked(uint, QString) {} void NotificationsManager::notificationClosed(uint, uint) {} + +void +NotificationsManager::removeNotification(const QString &roomId, const QString &eventId) +{} +