[libnotificationmanager] introduce the WatchedNotificationsModel

Summary: This allows one to subscribe to notifications from notification server.

Test Plan:
tested using very simple QML

```
import QtQuick 2.0
import org.kde.notificationmanager 1.1 as Notifications

...
Notifications.NotificationWatchedModel {
    id: model
}

```

Reviewers: #plasma, broulik, davidedmundson

Reviewed By: #plasma, broulik

Subscribers: nicolasfella, plasma-devel

Tags: #plasma

Differential Revision: https://phabricator.kde.org/D28509
wilder-portage-prov
Bhushan Shah 6 years ago
parent 63deb6a012
commit 2357e6f2e1
  1. 4
      libnotificationmanager/CMakeLists.txt
  2. 393
      libnotificationmanager/abstractnotificationsmodel.cpp
  3. 90
      libnotificationmanager/abstractnotificationsmodel.h
  4. 58
      libnotificationmanager/abstractnotificationsmodel_p.h
  5. 11
      libnotificationmanager/dbus/org.kde.notificationmanager.xml
  6. 2
      libnotificationmanager/declarative/notificationmanagerplugin.cpp
  7. 16
      libnotificationmanager/notification.cpp
  8. 10
      libnotificationmanager/notification.h
  9. 3
      libnotificationmanager/notification_p.h
  10. 464
      libnotificationmanager/notificationsmodel.cpp
  11. 48
      libnotificationmanager/notificationsmodel.h
  12. 61
      libnotificationmanager/server_p.cpp
  13. 8
      libnotificationmanager/server_p.h
  14. 178
      libnotificationmanager/watchednotificationsmodel.cpp
  15. 60
      libnotificationmanager/watchednotificationsmodel.h

@ -14,11 +14,13 @@ set(notificationmanager_LIB_SRCS
notifications.cpp
notification.cpp
abstractnotificationsmodel.cpp
notificationsmodel.cpp
notificationfilterproxymodel.cpp
notificationsortproxymodel.cpp
notificationgroupingproxymodel.cpp
notificationgroupcollapsingproxymodel.cpp
watchednotificationsmodel.cpp
jobsmodel.cpp
jobsmodel_p.cpp
@ -47,6 +49,8 @@ kconfig_add_kcfg_files(notificationmanager_LIB_SRCS kcfg/behaviorsettings.kcfgc
# DBus
# Notifications
qt5_add_dbus_adaptor(notificationmanager_LIB_SRCS dbus/org.freedesktop.Notifications.xml server_p.h NotificationManager::ServerPrivate)
qt5_add_dbus_adaptor(notificationmanager_LIB_SRCS dbus/org.kde.notificationmanager.xml server_p.h NotificationManager::ServerPrivate)
qt5_add_dbus_interface(notificationmanager_LIB_SRCS dbus/org.freedesktop.Notifications.xml fdonotifications_interface)
# JobView
qt5_add_dbus_adaptor(notificationmanager_LIB_SRCS dbus/org.kde.kuiserver.xml jobsmodel_p.h NotificationManager::JobsModelPrivate)
qt5_add_dbus_adaptor(notificationmanager_LIB_SRCS dbus/org.kde.JobViewServer.xml jobsmodel_p.h NotificationManager::JobsModelPrivate)

@ -0,0 +1,393 @@
/*
* Copyright 2018-2019 Kai Uwe Broulik <kde@privat.broulik.de>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) version 3, or any
* later version accepted by the membership of KDE e.V. (or its
* successor approved by the membership of KDE e.V.), which shall
* act as a proxy defined in Section 6 of version 3 of the license.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#include "abstractnotificationsmodel.h"
#include "abstractnotificationsmodel_p.h"
#include "debug.h"
#include "server.h"
#include "utils_p.h"
#include "notifications.h"
#include "notification.h"
#include "notification_p.h"
#include <QDebug>
#include <QProcess>
#include <QTimer>
#include <KShell>
#include <algorithm>
#include <functional>
static const int s_notificationsLimit = 1000;
using namespace NotificationManager;
AbstractNotificationsModel::Private::Private(AbstractNotificationsModel *q)
: q(q)
, lastRead(QDateTime::currentDateTimeUtc())
{
}
AbstractNotificationsModel::Private::~Private()
{
qDeleteAll(notificationTimeouts);
notificationTimeouts.clear();
}
void AbstractNotificationsModel::Private::onNotificationAdded(const Notification &notification)
{
// Once we reach a certain insane number of notifications discard some old ones
// as we keep pixmaps around etc
if (notifications.count() >= s_notificationsLimit) {
const int cleanupCount = s_notificationsLimit / 2;
qCDebug(NOTIFICATIONMANAGER) << "Reached the notification limit of" << s_notificationsLimit << ", discarding the oldest" << cleanupCount << "notifications";
q->beginRemoveRows(QModelIndex(), 0, cleanupCount - 1);
for (int i = 0 ; i < cleanupCount; ++i) {
notifications.removeAt(0);
// TODO close gracefully?
}
q->endRemoveRows();
}
setupNotificationTimeout(notification);
q->beginInsertRows(QModelIndex(), notifications.count(), notifications.count());
notifications.append(std::move(notification));
q->endInsertRows();
}
void AbstractNotificationsModel::Private::onNotificationReplaced(uint replacedId, const Notification &notification)
{
const int row = q->rowOfNotification(replacedId);
if (row == -1) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to replace notification with id" << replacedId << "which doesn't exist, creating a new one. This is an application bug!";
onNotificationAdded(notification);
return;
}
setupNotificationTimeout(notification);
notifications[row] = notification;
const QModelIndex idx = q->index(row, 0);
emit q->dataChanged(idx, idx);
}
void AbstractNotificationsModel::Private::onNotificationRemoved(uint removedId, Server::CloseReason reason)
{
const int row = q->rowOfNotification(removedId);
if (row == -1) {
return;
}
q->stopTimeout(removedId);
// When a notification expired, keep it around in the history and mark it as such
if (reason == Server::CloseReason::Expired) {
const QModelIndex idx = q->index(row, 0);
Notification &notification = notifications[row];
notification.setExpired(true);
// Since the notification is "closed" it cannot have any actions
// unless it is "resident" which we don't support
notification.setActions(QStringList());
emit q->dataChanged(idx, idx, {
Notifications::ExpiredRole,
// TODO only emit those if actually changed?
Notifications::ActionNamesRole,
Notifications::ActionLabelsRole,
Notifications::HasDefaultActionRole,
Notifications::DefaultActionLabelRole,
Notifications::ConfigurableRole
});
return;
}
// Otherwise if explicitly closed by either user or app, remove it
q->beginRemoveRows(QModelIndex(), row, row);
notifications.removeAt(row);
q->endRemoveRows();
}
void AbstractNotificationsModel::Private::setupNotificationTimeout(const Notification &notification)
{
if (notification.timeout() == 0) {
// In case it got replaced by a persistent notification
q->stopTimeout(notification.id());
return;
}
QTimer *timer = notificationTimeouts.value(notification.id());
if (!timer) {
timer = new QTimer();
timer->setSingleShot(true);
connect(timer, &QTimer::timeout, q, [this, timer] {
const uint id = timer->property("notificationId").toUInt();
q->expire(id);
});
notificationTimeouts.insert(notification.id(), timer);
}
timer->stop();
timer->setProperty("notificationId", notification.id());
timer->setInterval(60000 /*1min*/ + (notification.timeout() == -1 ? 120000 /*2min, max configurable default timeout*/ : notification.timeout()));
timer->start();
}
int AbstractNotificationsModel::rowOfNotification(uint id) const
{
auto it = std::find_if(d->notifications.constBegin(), d->notifications.constEnd(), [id](const Notification &item) {
return item.id() == id;
});
if (it == d->notifications.constEnd()) {
return -1;
}
return std::distance(d->notifications.constBegin(), it);
}
AbstractNotificationsModel::AbstractNotificationsModel()
: QAbstractListModel(nullptr)
, d(new Private(this))
{
}
AbstractNotificationsModel::~AbstractNotificationsModel() = default;
QDateTime AbstractNotificationsModel::lastRead() const
{
return d->lastRead;
}
void AbstractNotificationsModel::setLastRead(const QDateTime &lastRead)
{
if (d->lastRead != lastRead) {
d->lastRead = lastRead;
emit lastReadChanged();
}
}
QVariant AbstractNotificationsModel::data(const QModelIndex &index, int role) const
{
if (!checkIndex(index)) {
return QVariant();
}
const Notification &notification = d->notifications.at(index.row());
switch (role) {
case Notifications::IdRole: return notification.id();
case Notifications::TypeRole: return Notifications::NotificationType;
case Notifications::CreatedRole:
if (notification.created().isValid()) {
return notification.created();
}
break;
case Notifications::UpdatedRole:
if (notification.updated().isValid()) {
return notification.updated();
}
break;
case Notifications::SummaryRole: return notification.summary();
case Notifications::BodyRole: return notification.body();
case Notifications::IconNameRole:
if (notification.image().isNull()) {
return notification.icon();
}
break;
case Notifications::ImageRole:
if (!notification.image().isNull()) {
return notification.image();
}
break;
case Notifications::DesktopEntryRole: return notification.desktopEntry();
case Notifications::NotifyRcNameRole: return notification.notifyRcName();
case Notifications::ApplicationNameRole: return notification.applicationName();
case Notifications::ApplicationIconNameRole: return notification.applicationIconName();
case Notifications::OriginNameRole: return notification.originName();
case Notifications::ActionNamesRole: return notification.actionNames();
case Notifications::ActionLabelsRole: return notification.actionLabels();
case Notifications::HasDefaultActionRole: return notification.hasDefaultAction();
case Notifications::DefaultActionLabelRole: return notification.defaultActionLabel();
case Notifications::UrlsRole: return QVariant::fromValue(notification.urls());
case Notifications::UrgencyRole: return static_cast<int>(notification.urgency());
case Notifications::UserActionFeedbackRole: return notification.userActionFeedback();
case Notifications::TimeoutRole: return notification.timeout();
case Notifications::ClosableRole: return true;
case Notifications::ConfigurableRole: return notification.configurable();
case Notifications::ConfigureActionLabelRole: return notification.configureActionLabel();
case Notifications::ExpiredRole: return notification.expired();
case Notifications::ReadRole: return notification.read();
case Notifications::HasReplyActionRole: return notification.hasReplyAction();
case Notifications::ReplyActionLabelRole: return notification.replyActionLabel();
case Notifications::ReplyPlaceholderTextRole: return notification.replyPlaceholderText();
case Notifications::ReplySubmitButtonTextRole: return notification.replySubmitButtonText();
case Notifications::ReplySubmitButtonIconNameRole: return notification.replySubmitButtonIconName();
}
return QVariant();
}
bool AbstractNotificationsModel::setData(const QModelIndex &index, const QVariant &value, int role)
{
if (!checkIndex(index)) {
return false;
}
Notification &notification = d->notifications[index.row()];
switch (role) {
case Notifications::ReadRole:
if (value.toBool() != notification.read()) {
notification.setRead(value.toBool());
return true;
}
break;
}
return false;
}
int AbstractNotificationsModel::rowCount(const QModelIndex &parent) const
{
if (parent.isValid()) {
return 0;
}
return d->notifications.count();
}
QHash<int, QByteArray> AbstractNotificationsModel::roleNames() const
{
return Utils::roleNames();
}
void AbstractNotificationsModel::startTimeout(uint notificationId)
{
const int row = rowOfNotification(notificationId);
if (row == -1) {
return;
}
const Notification &notification = d->notifications.at(row);
if (!notification.timeout() || notification.expired()) {
return;
}
d->setupNotificationTimeout(notification);
}
void AbstractNotificationsModel::stopTimeout(uint notificationId)
{
delete d->notificationTimeouts.take(notificationId);
}
void AbstractNotificationsModel::clear(Notifications::ClearFlags flags)
{
if (d->notifications.isEmpty()) {
return;
}
// Tries to remove a contiguous group if possible as the likely case is
// you have n unread notifications at the end of the list, we don't want to
// remove and signal each item individually
QVector<QPair<int, int>> clearQueue;
QPair<int, int> clearRange{-1, -1};
for (int i = d->notifications.count() - 1; i >= 0; --i) {
const Notification &notification = d->notifications.at(i);
bool clear = (flags.testFlag(Notifications::ClearExpired) && notification.expired());
if (clear) {
if (clearRange.second == -1) {
clearRange.second = i;
}
clearRange.first = i;
} else {
if (clearRange.first != -1) {
clearQueue.append(clearRange);
clearRange.first = -1;
clearRange.second = -1;
}
}
}
if (clearRange.first != -1) {
clearQueue.append(clearRange);
clearRange.first = -1;
clearRange.second = -1;
}
for (const auto &range : clearQueue) {
beginRemoveRows(QModelIndex(), range.first, range.second);
for (int i = range.second; i >= range.first; --i) {
d->notifications.removeAt(i);
}
endRemoveRows();
}
}
void AbstractNotificationsModel::onNotificationAdded(const Notification &notification)
{
d->onNotificationAdded(notification);
}
void AbstractNotificationsModel::onNotificationReplaced(uint replacedId, const Notification &notification)
{
d->onNotificationReplaced(replacedId, notification);
}
void AbstractNotificationsModel::onNotificationRemoved(uint notificationId, Server::CloseReason reason)
{
d->onNotificationRemoved(notificationId, reason);
}
void AbstractNotificationsModel::setupNotificationTimeout(const Notification &notification)
{
d->setupNotificationTimeout(notification);
}
const QVector<Notification>& AbstractNotificationsModel::notifications()
{
return d->notifications;
}

@ -0,0 +1,90 @@
/*
* Copyright 2018-2019 Kai Uwe Broulik <kde@privat.broulik.de>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) version 3, or any
* later version accepted by the membership of KDE e.V. (or its
* successor approved by the membership of KDE e.V.), which shall
* act as a proxy defined in Section 6 of version 3 of the license.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef ABSTRACTNOTIFICATIONSMODEL_H
#define ABSTRACTNOTIFICATIONSMODEL_H
#include <QAbstractListModel>
#include <QScopedPointer>
#include <QSharedPointer>
#include <QDateTime>
#include "notifications.h"
#include "notification.h"
#include "server.h"
namespace NotificationManager
{
class Q_DECL_EXPORT AbstractNotificationsModel : public QAbstractListModel
{
Q_OBJECT
public:
~AbstractNotificationsModel() override;
QDateTime lastRead() const;
void setLastRead(const QDateTime &lastRead);
QVariant data(const QModelIndex &index, int role) const override;
bool setData(const QModelIndex &index, const QVariant &value, int role) override;
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
QHash<int, QByteArray> roleNames() const override;
virtual void expire(uint notificationId) = 0;
virtual void close(uint notificationId) = 0;
// Currently configure actions are not exposed in AbstractNotificationsModel to keep it very minimal
// if usecase for this comes up in future, we can revisit it.
virtual void invokeDefaultAction(uint notificationId) = 0;
virtual void invokeAction(uint notificationId, const QString &actionName) = 0;
virtual void reply(uint notificationId, const QString &text) = 0;
void startTimeout(uint notificationId);
void stopTimeout(uint notificationId);
void clear(Notifications::ClearFlags flags);
signals:
void lastReadChanged();
protected:
AbstractNotificationsModel();
void onNotificationAdded(const Notification &notification);
void onNotificationReplaced(uint replacedId, const Notification &notification);
void onNotificationRemoved(uint notificationId, Server::CloseReason reason);
void setupNotificationTimeout(const Notification &notification);
const QVector<Notification>& notifications();
int rowOfNotification(uint id) const;
private:
class Private;
QScopedPointer<Private> d;
Q_DISABLE_COPY(AbstractNotificationsModel)
};
} // namespace NotificationManager
#endif //ABSTRACTNOTIFICATIONSMODEL_H

@ -0,0 +1,58 @@
/*
* Copyright 2018-2019 Kai Uwe Broulik <kde@privat.broulik.de>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) version 3, or any
* later version accepted by the membership of KDE e.V. (or its
* successor approved by the membership of KDE e.V.), which shall
* act as a proxy defined in Section 6 of version 3 of the license.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef ABSTRACTNOTIFICATIONSMODEL_P_H
#define ABSTRACTNOTIFICATIONSMODEL_P_H
#include "notification.h"
#include "server.h"
#include <QDateTime>
namespace NotificationManager
{
class Q_DECL_HIDDEN AbstractNotificationsModel::Private
{
public:
explicit Private(AbstractNotificationsModel *q);
~Private();
void onNotificationAdded(const Notification &notification);
void onNotificationReplaced(uint replacedId, const Notification &notification);
void onNotificationRemoved(uint notificationId, Server::CloseReason reason);
void setupNotificationTimeout(const Notification &notification);
AbstractNotificationsModel *q;
QVector<Notification> notifications;
// Fallback timeout to ensure all notifications expire eventually
// otherwise when it isn't shown to the user and doesn't expire
// an app might wait indefinitely for the notification to do so
QHash<uint /*notificationId*/, QTimer*> notificationTimeouts;
QDateTime lastRead;
};
}
#endif // ABSTRACTNOTIFICATIONSMODEL_P_H

@ -0,0 +1,11 @@
<!DOCTYPE node PUBLIC "-//freedesktop//DTD D-BUS Object Introspection 1.0//EN" "http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd">
<node>
<interface name="org.kde.NotificationManager">
<method name="RegisterWatcher"/>
<method name="UnRegisterWatcher"/>
<method name="InvokeAction">
<arg name="id" type="u" direction="in"/>
<arg name="action_key" type="s" direction="in"/>
</method>
</interface>
</node>

@ -25,6 +25,7 @@
#include "server.h"
#include "serverinfo.h"
#include "settings.h"
#include "watchednotificationsmodel.h"
#include <QQmlEngine>
@ -42,4 +43,5 @@ void NotificationManagerPlugin::registerTypes(const char *uri)
return &Server::self();
});
qmlRegisterUncreatableType<ServerInfo>(uri, 1, 0, "ServerInfo", QStringLiteral("Can only access ServerInfo via Server"));
qmlRegisterType<WatchedNotificationsModel>(uri, 1, 1, "WatchedNotificationsModel");
}

@ -532,9 +532,15 @@ QString Notification::body() const
void Notification::setBody(const QString &body)
{
d->rawBody = body;
d->body = Private::sanitize(body.trimmed());
}
QString Notification::rawBody() const
{
return d->rawBody;
}
QString Notification::icon() const
{
return d->icon;
@ -750,6 +756,16 @@ void Notification::setDismissed(bool dismissed)
d->dismissed = dismissed;
}
QVariantMap Notification::hints() const
{
return d->hints;
}
void Notification::setHints(const QVariantMap &hints)
{
d->hints = hints;
}
void Notification::processHints(const QVariantMap &hints)
{
d->processHints(hints);

@ -70,6 +70,11 @@ public:
QString body() const;
void setBody(const QString &body);
// This returns the raw body data as provided by the notification
// this is useful when you want to html sanitization at different
// stage then the notification server.
QString rawBody() const;
QString icon() const;
void setIcon(const QString &icon);
@ -124,10 +129,15 @@ public:
bool dismissed() const;
void setDismissed(bool dismissed);
// Little bit of mess here, we want to sometime keep track of processed hints, and not process it.
QVariantMap hints() const;
void setHints(const QVariantMap &hints);
void processHints(const QVariantMap &hints);
private:
friend class NotificationsModel;
friend class AbstractNotificationsModel;
friend class ServerPrivate;
class Private;

@ -66,6 +66,8 @@ public:
QString summary;
QString body;
// raw body text without sanitize called.
QString rawBody;
// Can be theme icon name or path
QString icon;
QImage image;
@ -97,6 +99,7 @@ public:
QString replySubmitButtonIconName;
QList<QUrl> urls;
QVariantMap hints = QVariantMap();
bool userActionFeedback = false;
Notifications::Urgency urgency = Notifications::NormalUrgency;

@ -1,4 +1,5 @@
/*
* Copyright 2020 Shah Bhushan <bshah@kde.org>
* Copyright 2018-2019 Kai Uwe Broulik <kde@privat.broulik.de>
*
* This library is free software; you can redistribute it and/or
@ -19,365 +20,124 @@
*/
#include "notificationsmodel.h"
#include "debug.h"
#include "server.h"
#include "utils_p.h"
#include "notifications.h"
#include "notification.h"
#include "abstractnotificationsmodel_p.h"
#include "notification_p.h"
#include <QDebug>
#include "debug.h"
#include <QProcess>
#include <QTimer>
#include <KShell>
#include <algorithm>
#include <functional>
static const int s_notificationsLimit = 1000;
using namespace NotificationManager;
class Q_DECL_HIDDEN NotificationsModel::Private
{
public:
explicit Private(NotificationsModel *q);
~Private();
void onNotificationAdded(const Notification &notification);
void onNotificationReplaced(uint replacedId, const Notification &notification);
void onNotificationRemoved(uint notificationId, Server::CloseReason reason);
void setupNotificationTimeout(const Notification &notification);
int rowOfNotification(uint id) const;
NotificationsModel *q;
QVector<Notification> notifications;
// Fallback timeout to ensure all notifications expire eventually
// otherwise when it isn't shown to the user and doesn't expire
// an app might wait indefinitely for the notification to do so
QHash<uint /*notificationId*/, QTimer*> notificationTimeouts;
QDateTime lastRead;
};
NotificationsModel::Private::Private(NotificationsModel *q)
: q(q)
, lastRead(QDateTime::currentDateTimeUtc())
{
}
NotificationsModel::Private::~Private()
{
qDeleteAll(notificationTimeouts);
notificationTimeouts.clear();
}
void NotificationsModel::Private::onNotificationAdded(const Notification &notification)
{
// Once we reach a certain insane number of notifications discard some old ones
// as we keep pixmaps around etc
if (notifications.count() >= s_notificationsLimit) {
const int cleanupCount = s_notificationsLimit / 2;
qCDebug(NOTIFICATIONMANAGER) << "Reached the notification limit of" << s_notificationsLimit << ", discarding the oldest" << cleanupCount << "notifications";
q->beginRemoveRows(QModelIndex(), 0, cleanupCount - 1);
for (int i = 0 ; i < cleanupCount; ++i) {
notifications.removeAt(0);
// TODO close gracefully?
}
q->endRemoveRows();
}
setupNotificationTimeout(notification);
q->beginInsertRows(QModelIndex(), notifications.count(), notifications.count());
notifications.append(std::move(notification));
q->endInsertRows();
}
void NotificationsModel::Private::onNotificationReplaced(uint replacedId, const Notification &notification)
{
const int row = rowOfNotification(replacedId);
if (row == -1) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to replace notification with id" << replacedId << "which doesn't exist, creating a new one. This is an application bug!";
onNotificationAdded(notification);
return;
}
setupNotificationTimeout(notification);
notifications[row] = notification;
const QModelIndex idx = q->index(row, 0);
emit q->dataChanged(idx, idx);
}
void NotificationsModel::Private::onNotificationRemoved(uint removedId, Server::CloseReason reason)
{
const int row = rowOfNotification(removedId);
if (row == -1) {
return;
}
q->stopTimeout(removedId);
// When a notification expired, keep it around in the history and mark it as such
if (reason == Server::CloseReason::Expired) {
const QModelIndex idx = q->index(row, 0);
Notification &notification = notifications[row];
notification.setExpired(true);
// Since the notification is "closed" it cannot have any actions
// unless it is "resident" which we don't support
notification.setActions(QStringList());
emit q->dataChanged(idx, idx, {
Notifications::ExpiredRole,
// TODO only emit those if actually changed?
Notifications::ActionNamesRole,
Notifications::ActionLabelsRole,
Notifications::HasDefaultActionRole,
Notifications::DefaultActionLabelRole,
Notifications::ConfigurableRole
});
return;
}
// Otherwise if explicitly closed by either user or app, remove it
q->beginRemoveRows(QModelIndex(), row, row);
notifications.removeAt(row);
q->endRemoveRows();
}
void NotificationsModel::Private::setupNotificationTimeout(const Notification &notification)
{
if (notification.timeout() == 0) {
// In case it got replaced by a persistent notification
q->stopTimeout(notification.id());
return;
}
QTimer *timer = notificationTimeouts.value(notification.id());
if (!timer) {
timer = new QTimer();
timer->setSingleShot(true);
connect(timer, &QTimer::timeout, q, [this, timer] {
const uint id = timer->property("notificationId").toUInt();
q->expire(id);
});
notificationTimeouts.insert(notification.id(), timer);
}
timer->stop();
timer->setProperty("notificationId", notification.id());
timer->setInterval(60000 /*1min*/ + (notification.timeout() == -1 ? 120000 /*2min, max configurable default timeout*/ : notification.timeout()));
timer->start();
}
int NotificationsModel::Private::rowOfNotification(uint id) const
NotificationsModel::Ptr NotificationsModel::createNotificationsModel()
{
auto it = std::find_if(notifications.constBegin(), notifications.constEnd(), [id](const Notification &item) {
return item.id() == id;
});
if (it == notifications.constEnd()) {
return -1;
static QWeakPointer<NotificationsModel> s_instance;
if (!s_instance) {
QSharedPointer<NotificationsModel> ptr(new NotificationsModel());
s_instance = ptr.toWeakRef();
return ptr;
}
return std::distance(notifications.constBegin(), it);
return s_instance.toStrongRef();
}
NotificationsModel::NotificationsModel()
: QAbstractListModel(nullptr)
, d(new Private(this))
{
connect(&Server::self(), &Server::notificationAdded, this, [this](const Notification &notification) {
d->onNotificationAdded(notification);
onNotificationAdded(notification);
});
connect(&Server::self(), &Server::notificationReplaced, this, [this](uint replacedId, const Notification &notification) {
d->onNotificationReplaced(replacedId, notification);
onNotificationReplaced(replacedId, notification);
});
connect(&Server::self(), &Server::notificationRemoved, this, [this](uint removedId, Server::CloseReason reason) {
d->onNotificationRemoved(removedId, reason);
onNotificationRemoved(removedId, reason);
});
connect(&Server::self(), &Server::serviceOwnershipLost, this, [this] {
// Expire all notifications as we're defunct now
const auto notifications = d->notifications;
for (const Notification &notification : notifications) {
const auto notificationList = notifications();
for (const Notification &notification : notificationList) {
if (!notification.expired()) {
d->onNotificationRemoved(notification.id(), Server::CloseReason::Expired);
onNotificationRemoved(notification.id(), Server::CloseReason::Expired);
}
}
});
Server::self().init();
}
NotificationsModel::~NotificationsModel() = default;
NotificationsModel::Ptr NotificationsModel::createNotificationsModel()
void NotificationsModel::expire(uint notificationId)
{
static QWeakPointer<NotificationsModel> s_instance;
if (!s_instance) {
QSharedPointer<NotificationsModel> ptr(new NotificationsModel());
s_instance = ptr.toWeakRef();
return ptr;
if (rowOfNotification(notificationId) > -1) {
Server::self().closeNotification(notificationId, Server::CloseReason::Expired);
}
return s_instance.toStrongRef();
}
QDateTime NotificationsModel::lastRead() const
{
return d->lastRead;
}
void NotificationsModel::setLastRead(const QDateTime &lastRead)
void NotificationsModel::close(uint notificationId)
{
if (d->lastRead != lastRead) {
d->lastRead = lastRead;
emit lastReadChanged();
if (rowOfNotification(notificationId) > -1) {
Server::self().closeNotification(notificationId, Server::CloseReason::DismissedByUser);
}
}
QVariant NotificationsModel::data(const QModelIndex &index, int role) const
void NotificationsModel::invokeDefaultAction(uint notificationId)
{
if (!checkIndex(index, QAbstractItemModel::CheckIndexOption::IndexIsValid)) {
return QVariant();
const int row = rowOfNotification(notificationId);
if (row == -1) {
return;
}
const Notification &notification = d->notifications.at(index.row());
switch (role) {
case Notifications::IdRole: return notification.id();
case Notifications::TypeRole: return Notifications::NotificationType;
case Notifications::CreatedRole:
if (notification.created().isValid()) {
return notification.created();
}
break;
case Notifications::UpdatedRole:
if (notification.updated().isValid()) {
return notification.updated();
}
break;
case Notifications::SummaryRole: return notification.summary();
case Notifications::BodyRole: return notification.body();
case Notifications::IconNameRole:
if (notification.image().isNull()) {
return notification.icon();
}
break;
case Notifications::ImageRole:
if (!notification.image().isNull()) {
return notification.image();
}
break;
case Notifications::DesktopEntryRole: return notification.desktopEntry();
case Notifications::NotifyRcNameRole: return notification.notifyRcName();
case Notifications::ApplicationNameRole: return notification.applicationName();
case Notifications::ApplicationIconNameRole: return notification.applicationIconName();
case Notifications::OriginNameRole: return notification.originName();
case Notifications::ActionNamesRole: return notification.actionNames();
case Notifications::ActionLabelsRole: return notification.actionLabels();
case Notifications::HasDefaultActionRole: return notification.hasDefaultAction();
case Notifications::DefaultActionLabelRole: return notification.defaultActionLabel();
case Notifications::UrlsRole: return QVariant::fromValue(notification.urls());
case Notifications::UrgencyRole: return static_cast<int>(notification.urgency());
case Notifications::UserActionFeedbackRole: return notification.userActionFeedback();
case Notifications::TimeoutRole: return notification.timeout();
case Notifications::ClosableRole: return true;
case Notifications::ConfigurableRole: return notification.configurable();
case Notifications::ConfigureActionLabelRole: return notification.configureActionLabel();
case Notifications::ExpiredRole: return notification.expired();
case Notifications::ReadRole: return notification.read();
case Notifications::HasReplyActionRole: return notification.hasReplyAction();
case Notifications::ReplyActionLabelRole: return notification.replyActionLabel();
case Notifications::ReplyPlaceholderTextRole: return notification.replyPlaceholderText();
case Notifications::ReplySubmitButtonTextRole: return notification.replySubmitButtonText();
case Notifications::ReplySubmitButtonIconNameRole: return notification.replySubmitButtonIconName();
const Notification &notification = notifications().at(row);
if (!notification.hasDefaultAction()) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to invoke default action on notification" << notificationId << "which doesn't have one";
return;
}
return QVariant();
Server::self().invokeAction(notificationId, QStringLiteral("default")); // FIXME make a static Notification::defaultActionName() or something
}
bool NotificationsModel::setData(const QModelIndex &index, const QVariant &value, int role)
void NotificationsModel::invokeAction(uint notificationId, const QString &actionName)
{
if (!checkIndex(index, QAbstractItemModel::CheckIndexOption::IndexIsValid)) {
return false;
const int row = rowOfNotification(notificationId);
if (row == -1) {
return;
}
Notification &notification = d->notifications[index.row()];
switch (role) {
case Notifications::ReadRole:
if (value.toBool() != notification.read()) {
notification.setRead(value.toBool());
return true;
}
break;
const Notification &notification = notifications().at(row);
if (!notification.actionNames().contains(actionName)) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to invoke action" << actionName << "on notification" << notificationId << "which it doesn't have";
return;
}
return false;
Server::self().invokeAction(notificationId, actionName);
}
int NotificationsModel::rowCount(const QModelIndex &parent) const
void NotificationsModel::reply(uint notificationId, const QString &text)
{
if (parent.isValid()) {
return 0;
const int row = rowOfNotification(notificationId);
if (row == -1) {
return;
}
return d->notifications.count();
}
QHash<int, QByteArray> NotificationsModel::roleNames() const
{
return Utils::roleNames();
}
void NotificationsModel::expire(uint notificationId)
{
if (d->rowOfNotification(notificationId) > -1) {
Server::self().closeNotification(notificationId, Server::CloseReason::Expired);
const Notification &notification = notifications().at(row);
if (!notification.hasReplyAction()) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to reply to a notification which doesn't have a reply action";
return;
}
}
void NotificationsModel::close(uint notificationId)
{
if (d->rowOfNotification(notificationId) > -1) {
Server::self().closeNotification(notificationId, Server::CloseReason::DismissedByUser);
}
Server::self().reply(notification.dBusService(), notificationId, text);
}
void NotificationsModel::configure(uint notificationId)
{
const int row = d->rowOfNotification(notificationId);
const int row = rowOfNotification(notificationId);
if (row == -1) {
return;
}
const Notification &notification = d->notifications.at(row);
const Notification &notification = notifications().at(row);
if (notification.d->hasConfigureAction) {
Server::self().invokeAction(notificationId, QStringLiteral("settings")); // FIXME make a static Notification::configureActionName() or something
@ -418,119 +178,3 @@ void NotificationsModel::configure(const QString &desktopEntry, const QString &n
KShell::joinArgs(args)
});
}
void NotificationsModel::invokeDefaultAction(uint notificationId)
{
const int row = d->rowOfNotification(notificationId);
if (row == -1) {
return;
}
const Notification &notification = d->notifications.at(row);
if (!notification.hasDefaultAction()) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to invoke default action on notification" << notificationId << "which doesn't have one";
return;
}
Server::self().invokeAction(notificationId, QStringLiteral("default")); // FIXME make a static Notification::defaultActionName() or something
}
void NotificationsModel::invokeAction(uint notificationId, const QString &actionName)
{
const int row = d->rowOfNotification(notificationId);
if (row == -1) {
return;
}
const Notification &notification = d->notifications.at(row);
if (!notification.actionNames().contains(actionName)) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to invoke action" << actionName << "on notification" << notificationId << "which it doesn't have";
return;
}
Server::self().invokeAction(notificationId, actionName);
}
void NotificationsModel::reply(uint notificationId, const QString &text)
{
const int row = d->rowOfNotification(notificationId);
if (row == -1) {
return;
}
const Notification &notification = d->notifications.at(row);
if (!notification.hasReplyAction()) {
qCWarning(NOTIFICATIONMANAGER) << "Trying to reply to a notification which doesn't have a reply action";
return;
}
Server::self().reply(notification.dBusService(), notificationId, text);
}
void NotificationsModel::startTimeout(uint notificationId)
{
const int row = d->rowOfNotification(notificationId);
if (row == -1) {
return;
}
const Notification &notification = d->notifications.at(row);
if (!notification.timeout() || notification.expired()) {
return;
}
d->setupNotificationTimeout(notification);
}
void NotificationsModel::stopTimeout(uint notificationId)
{
delete d->notificationTimeouts.take(notificationId);
}
void NotificationsModel::clear(Notifications::ClearFlags flags)
{
if (d->notifications.isEmpty()) {
return;
}
// Tries to remove a contiguous group if possible as the likely case is
// you have n unread notifications at the end of the list, we don't want to
// remove and signal each item individually
QVector<QPair<int, int>> clearQueue;
QPair<int, int> clearRange{-1, -1};
for (int i = d->notifications.count() - 1; i >= 0; --i) {
const Notification &notification = d->notifications.at(i);
bool clear = (flags.testFlag(Notifications::ClearExpired) && notification.expired());
if (clear) {
if (clearRange.second == -1) {
clearRange.second = i;
}
clearRange.first = i;
} else {
if (clearRange.first != -1) {
clearQueue.append(clearRange);
clearRange.first = -1;
clearRange.second = -1;
}
}
}
if (clearRange.first != -1) {
clearQueue.append(clearRange);
clearRange.first = -1;
clearRange.second = -1;
}
for (const auto &range : clearQueue) {
beginRemoveRows(QModelIndex(), range.first, range.second);
for (int i = range.second; i >= range.first; --i) {
d->notifications.removeAt(i);
}
endRemoveRows();
}
}

@ -1,4 +1,5 @@
/*
* Copyright 2020 Shah Bhushan <bshah@kde.org>
* Copyright 2018-2019 Kai Uwe Broulik <kde@privat.broulik.de>
*
* This library is free software; you can redistribute it and/or
@ -20,56 +21,27 @@
#pragma once
#include <QAbstractListModel>
#include <QScopedPointer>
#include <QSharedPointer>
#include "abstractnotificationsmodel.h"
#include "notifications.h"
namespace NotificationManager {
namespace NotificationManager
class NotificationsModel : public AbstractNotificationsModel
{
class NotificationsModel : public QAbstractListModel
{
Q_OBJECT
public:
~NotificationsModel() override;
using Ptr = QSharedPointer<NotificationsModel>;
static Ptr createNotificationsModel();
void expire(uint notificationId) override;
void close(uint notificationId) override;
QDateTime lastRead() const;
void setLastRead(const QDateTime &lastRead);
QVariant data(const QModelIndex &index, int role) const override;
bool setData(const QModelIndex &index, const QVariant &value, int role) override;
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
QHash<int, QByteArray> roleNames() const override;
void invokeDefaultAction(uint notificationId) override;
void invokeAction(uint notificationId, const QString &actionName) override;
void reply(uint notificationId, const QString &text) override;
void expire(uint notificationId);
void close(uint notificationId);
void configure(uint notificationId);
void configure(const QString &desktopEntry, const QString &notifyRcName, const QString &eventId);
void invokeDefaultAction(uint notificationId);
void invokeAction(uint notificationId, const QString &actionName);
void reply(uint notificationId, const QString &text);
void startTimeout(uint notificationId);
void stopTimeout(uint notificationId);
void clear(Notifications::ClearFlags flags);
signals:
void lastReadChanged();
private:
class Private;
QScopedPointer<Private> d;
NotificationsModel();
Q_DISABLE_COPY(NotificationsModel)
};
} // namespace NotificationManager
}

@ -23,6 +23,7 @@
#include "debug.h"
#include "notificationsadaptor.h"
#include "notificationmanageradaptor.h"
#include "notification.h"
#include "notification_p.h"
@ -45,10 +46,17 @@ using namespace NotificationManager;
ServerPrivate::ServerPrivate(QObject *parent)
: QObject(parent)
, m_inhibitionWatcher(new QDBusServiceWatcher(this))
, m_notificationWatchers (new QDBusServiceWatcher(this))
{
m_inhibitionWatcher->setConnection(QDBusConnection::sessionBus());
m_inhibitionWatcher->setWatchMode(QDBusServiceWatcher::WatchForUnregistration);
connect(m_inhibitionWatcher, &QDBusServiceWatcher::serviceUnregistered, this, &ServerPrivate::onInhibitionServiceUnregistered);
m_notificationWatchers->setConnection(QDBusConnection::sessionBus());
m_notificationWatchers->setWatchMode(QDBusServiceWatcher::WatchForUnregistration);
connect(m_notificationWatchers, &QDBusServiceWatcher::serviceUnregistered, [=](const QString &service) {
m_notificationWatchers->removeWatchedService(service);
});
}
ServerPrivate::~ServerPrivate() = default;
@ -84,6 +92,7 @@ bool ServerPrivate::init()
}
new NotificationsAdaptor(this);
new NotificationManagerAdaptor(this);
if (!m_dbusObjectValid) { // if already registered, don't fail here
m_dbusObjectValid = QDBusConnection::sessionBus().registerObject(notificationServicePath(), this);
@ -223,11 +232,48 @@ uint ServerPrivate::Notify(const QString &app_name, uint replaces_id, const QStr
emit static_cast<Server*>(parent())->notificationAdded(notification);
}
// currently we dispatch all notification, this is ugly
// TODO: come up with proper authentication/user selection
for (const QString &service : m_notificationWatchers->watchedServices()) {
QDBusMessage msg = QDBusMessage::createMethodCall(
service,
QStringLiteral("/NotificationWatcher"),
QStringLiteral("org.kde.NotificationWatcher"),
QStringLiteral("Notify")
);
msg.setArguments({
notificationId,
notification.applicationName(),
replaces_id,
notification.applicationIconName(),
notification.summary(),
// we pass raw body data since this data goes through another sanitization
// in WatchedNotificationsModel when notification object is created.
notification.rawBody(),
notification.actionNames(),
hints,
notification.timeout()
});
QDBusConnection::sessionBus().call(msg, QDBus::NoBlock);
}
return notificationId;
}
void ServerPrivate::CloseNotification(uint id)
{
for (const QString &service : m_notificationWatchers->watchedServices()) {
QDBusMessage msg = QDBusMessage::createMethodCall(
service,
QStringLiteral("/NotificationWatcher"),
QStringLiteral("org.kde.NotificationWatcher"),
QStringLiteral("CloseNotification")
);
msg.setArguments({
id
});
QDBusConnection::sessionBus().call(msg, QDBus::NoBlock);
}
// spec says "If the notification no longer exists, an empty D-BUS Error message is sent back."
static_cast<Server*>(parent())->closeNotification(id, Server::CloseReason::Revoked);
}
@ -482,3 +528,18 @@ void ServerPrivate::clearExternalInhibitions()
emit externalInhibitedChanged();
emit externalInhibitionsChanged();
}
void ServerPrivate::RegisterWatcher()
{
m_notificationWatchers->addWatchedService(message().service());
}
void ServerPrivate::UnRegisterWatcher()
{
m_notificationWatchers->removeWatchedService(message().service());
}
void ServerPrivate::InvokeAction(uint id, const QString& actionKey)
{
ActionInvoked(id, actionKey);
}

@ -22,6 +22,7 @@
#include <QObject>
#include <QDBusContext>
#include <QStringList>
#include "notification.h"
@ -68,6 +69,12 @@ public:
void UnInhibit(uint cookie);
bool inhibited() const; // property getter
// Notifition watcher
void RegisterWatcher();
void UnRegisterWatcher();
void InvokeAction(uint id, const QString &actionKey);
Q_SIGNALS:
// DBus
void NotificationClosed(uint id, uint reason);
@ -121,6 +128,7 @@ private:
mutable QScopedPointer<ServerInfo> m_currentOwner;
QDBusServiceWatcher *m_inhibitionWatcher = nullptr;
QDBusServiceWatcher *m_notificationWatchers = nullptr;
uint m_highestInhibitionCookie = 0;
QHash<uint /*cookie*/, Inhibition> m_externalInhibitions;
QHash<uint /*cookie*/, QString> m_inhibitionServices;

@ -0,0 +1,178 @@
/*
* Copyright 2020 Shah Bhushan <bshah@kde.org>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) version 3, or any
* later version accepted by the membership of KDE e.V. (or its
* successor approved by the membership of KDE e.V.), which shall
* act as a proxy defined in Section 6 of version 3 of the license.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#include "watchednotificationsmodel.h"
#include <QDBusMetaType>
#include <QDBusConnection>
#include <QDBusServiceWatcher>
#include <QDBusConnectionInterface>
#include <QDebug>
#include "fdonotifications_interface.h"
using namespace NotificationManager;
class WatchedNotificationsModel::Private : public QObject
{
Q_OBJECT
public:
explicit Private(WatchedNotificationsModel* q, QObject* parent = nullptr);
~Private();
bool valid = false;
public Q_SLOTS:
Q_SCRIPTABLE void Notify(uint id, const QString &app_name, uint replaces_id, const QString &app_icon,
const QString &summary, const QString &body, const QStringList &actions,
const QVariantMap &hints, int timeout);
Q_SCRIPTABLE void CloseNotification(uint id);
void NotificationClosed(uint id, uint reason);
private:
WatchedNotificationsModel* q;
OrgFreedesktopNotificationsInterface *fdoNotificationsInterface;
};
WatchedNotificationsModel::Private::Private(WatchedNotificationsModel* q, QObject *parent)
: q(q)
, QObject(parent)
{
QDBusConnection dbus = QDBusConnection::sessionBus();
fdoNotificationsInterface = new OrgFreedesktopNotificationsInterface(QStringLiteral("org.freedesktop.Notifications"),
QStringLiteral("/org/freedesktop/Notifications"),
dbus,
this);
connect(fdoNotificationsInterface, &OrgFreedesktopNotificationsInterface::NotificationClosed,
this, &WatchedNotificationsModel::Private::NotificationClosed);
dbus.registerObject("/NotificationWatcher", QStringLiteral("org.kde.NotificationWatcher"), this, QDBusConnection::ExportScriptableSlots);
QDBusMessage msg = QDBusMessage::createMethodCall(
QStringLiteral("org.freedesktop.Notifications"),
QStringLiteral("/org/freedesktop/Notifications"),
QStringLiteral("org.kde.NotificationManager"),
QStringLiteral("RegisterWatcher")
);
QDBusMessage reply = QDBusConnection::sessionBus().call(msg, QDBus::NoBlock);
if(reply.type() != QDBusMessage::ErrorMessage) {
valid = true;
Q_EMIT q->validChanged(valid);
}
}
WatchedNotificationsModel::Private::~Private()
{
QDBusMessage msg = QDBusMessage::createMethodCall(
QStringLiteral("org.freedesktop.Notifications"),
QStringLiteral("/org/freedesktop/Notifications"),
QStringLiteral("org.kde.NotificationManager"),
QStringLiteral("UnRegisterWatcher")
);
QDBusConnection::sessionBus().call(msg, QDBus::NoBlock);
}
void WatchedNotificationsModel::Private::Notify(uint id, const QString &app_name, uint replaces_id, const QString &app_icon,
const QString &summary, const QString &body, const QStringList &actions,
const QVariantMap &hints, int timeout)
{
const bool wasReplaced = replaces_id > 0;
qDebug() << summary;
qDebug() << body;
Notification notification(id);
notification.setSummary(summary);
notification.setBody(body);
notification.setApplicationName(app_name);
notification.setActions(actions);
notification.setTimeout(timeout);
notification.setHints(hints);
notification.setIcon(app_icon);
if(wasReplaced) {
q->onNotificationReplaced(replaces_id, notification);
} else {
q->onNotificationAdded(notification);
}
}
void WatchedNotificationsModel::Private::CloseNotification(uint id)
{
q->onNotificationRemoved(id, Server::CloseReason::Expired);
}
void WatchedNotificationsModel::Private::NotificationClosed(uint id, uint reason)
{
q->onNotificationRemoved(id, static_cast<Server::CloseReason>(reason));
}
WatchedNotificationsModel::WatchedNotificationsModel()
: AbstractNotificationsModel(),
d(new Private(this, nullptr))
{
}
WatchedNotificationsModel::~WatchedNotificationsModel()
{
}
void WatchedNotificationsModel::close(uint notificationId)
{
onNotificationRemoved(notificationId, Server::CloseReason::DismissedByUser);
}
void WatchedNotificationsModel::expire(uint notificationId)
{
onNotificationRemoved(notificationId, Server::CloseReason::Expired);
}
void WatchedNotificationsModel::invokeDefaultAction(uint notificationId)
{
this->invokeAction(notificationId, QStringLiteral("default"));
}
void WatchedNotificationsModel::invokeAction(uint notificationId, const QString &actionName)
{
QDBusConnection dbus = QDBusConnection::sessionBus();
dbus.registerObject("/NotificationWatcher", this, QDBusConnection::ExportScriptableSlots);
QDBusMessage msg = QDBusMessage::createMethodCall(
QStringLiteral("org.freedesktop.Notifications"),
QStringLiteral("/org/freedesktop/Notifications"),
QStringLiteral("org.kde.NotificationManager"),
QStringLiteral("InvokeAction")
);
msg.setArguments({
notificationId,
actionName
});
QDBusConnection::sessionBus().call(msg, QDBus::NoBlock);
}
void WatchedNotificationsModel::reply(uint notificationId, const QString &text)
{
// todo
Q_UNUSED(notificationId)
Q_UNUSED(text)
}
bool WatchedNotificationsModel::valid()
{
return d->valid;
}
#include "watchednotificationsmodel.moc"

@ -0,0 +1,60 @@
/*
* Copyright 2020 Shah Bhushan <bshah@kde.org>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) version 3, or any
* later version accepted by the membership of KDE e.V. (or its
* successor approved by the membership of KDE e.V.), which shall
* act as a proxy defined in Section 6 of version 3 of the license.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef WATCHEDNOTIFICATIONSMODEL_H
#define WATCHEDNOTIFICATIONSMODEL_H
#include "abstractnotificationsmodel.h"
#include "notificationmanager_export.h"
namespace NotificationManager
{
class NOTIFICATIONMANAGER_EXPORT WatchedNotificationsModel : public AbstractNotificationsModel
{
Q_OBJECT
Q_PROPERTY(bool valid READ valid NOTIFY validChanged)
public:
explicit WatchedNotificationsModel();
~WatchedNotificationsModel();
void expire(uint notificationId) override;
void close(uint notificationId) override;
void invokeDefaultAction(uint notificationId) override;
void invokeAction(uint notificationId, const QString &actionName) override;
void reply(uint notificationId, const QString &text) override;
bool valid();
signals:
void validChanged(bool valid);
private:
class Private;
Private * const d;
Q_DISABLE_COPY(WatchedNotificationsModel)
};
}
#endif // WATCHEDNOTIFICATIONSMODEL_H
Loading…
Cancel
Save