KNotifications

knotification.cpp
1/*
2 This file is part of the KDE libraries
3 SPDX-FileCopyrightText: 2005-2006 Olivier Goffart <ogoffart at kde.org>
4 SPDX-FileCopyrightText: 2013-2014 Martin Klapetek <mklapetek@kde.org>
5
6 code from KNotify/KNotifyClient
7 SPDX-FileCopyrightText: 1997 Christian Esken <esken@kde.org>
8 SPDX-FileCopyrightText: 2000 Charles Samuels <charles@kde.org>
9 SPDX-FileCopyrightText: 2000 Stefan Schimanski <1Stein@gmx.de>
10 SPDX-FileCopyrightText: 2000 Matthias Ettrich <ettrich@kde.org>
11 SPDX-FileCopyrightText: 2000 Waldo Bastian <bastian@kde.org>
12 SPDX-FileCopyrightText: 2000-2003 Carsten Pfeiffer <pfeiffer@kde.org>
13 SPDX-FileCopyrightText: 2005 Allan Sandfeld Jensen <kde@carewolf.com>
14
15 SPDX-License-Identifier: LGPL-2.0-only
16*/
17
18#include "knotification.h"
19#include "debug_p.h"
20#include "knotification_p.h"
21#include "knotificationmanager_p.h"
22#include "knotificationreplyaction.h"
23
24#include <config-knotifications.h>
25
26#include <QGuiApplication>
27
28#include <QStringList>
29#include <QUrl>
30
31// incremental notification ID
32static int notificationIdCounter = 0;
33
34class KNotificationActionPrivate
35{
36public:
37 QString label;
38 QString id;
39};
40
41KNotificationAction::KNotificationAction(QObject *parent)
42 : QObject(parent)
43 , d(new KNotificationActionPrivate)
44{
45}
46
47KNotificationAction::KNotificationAction(const QString &label)
48 : QObject()
49 , d(new KNotificationActionPrivate)
50{
51 d->label = label;
52}
53
54KNotificationAction::~KNotificationAction()
55{
56}
57
59{
60 return d->label;
61}
62
64{
65 if (d->label != label) {
66 d->label = label;
68 }
69}
70
71QString KNotificationAction::id() const
72{
73 return d->id;
74}
75
76void KNotificationAction::setId(const QString &id)
77{
78 d->id = id;
79}
80
83 , d(new Private)
84{
85 d->eventId = eventId;
86 d->flags = flags;
87 connect(&d->updateTimer, &QTimer::timeout, this, &KNotification::update);
88 d->updateTimer.setSingleShot(true);
89 d->updateTimer.setInterval(100);
90 d->id = ++notificationIdCounter;
91}
92
93KNotification::~KNotification()
94{
95 if (d->ownsActions) {
96 qDeleteAll(d->actions);
97 delete d->defaultAction;
98 }
99
100 if (d->id >= 0) {
101 KNotificationManager::self()->close(d->id);
102 }
103}
104
106{
107 return d->eventId;
108}
109
111{
112 if (d->eventId != eventId) {
113 d->eventId = eventId;
115 }
116}
117
119{
120 return d->title;
121}
122
124{
125 return d->text;
126}
127
129{
130 if (title == d->title) {
131 return;
132 }
133
134 d->needUpdate = true;
135 d->title = title;
137 if (d->id >= 0) {
138 d->updateTimer.start();
139 }
140}
141
143{
144 if (text == d->text) {
145 return;
146 }
147
148 d->needUpdate = true;
149 d->text = text;
151 if (d->id >= 0) {
152 d->updateTimer.start();
153 }
154}
155
157{
158 if (icon == d->iconName) {
159 return;
160 }
161
162 d->needUpdate = true;
163 d->iconName = icon;
165 if (d->id >= 0) {
166 d->updateTimer.start();
167 }
168}
169
171{
172 return d->iconName;
173}
174
176{
177 return d->pixmap;
178}
179
181{
182 d->needUpdate = true;
183 d->pixmap = pix;
184 if (d->id >= 0) {
185 d->updateTimer.start();
186 }
187}
188
189QList<KNotificationAction *> KNotification::actions() const
190{
191 return d->actions;
192}
193
195{
196 if (d->ownsActions) {
197 qDeleteAll(d->actions);
198 }
199 d->actions.clear();
200 d->actionIdCounter = 1;
201
202 d->needUpdate = true;
203 if (d->id >= 0) {
204 d->updateTimer.start();
205 }
206}
207
209{
210 d->needUpdate = true;
211
212 KNotificationAction *action = new KNotificationAction(label);
213 connect(action, &KNotificationAction::labelChanged, this, [this] {
214 d->needUpdate = true;
215 if (d->id >= 0) {
216 d->updateTimer.start();
217 }
218 });
219 action->setId(QString::number(d->actionIdCounter));
220 d->actionIdCounter++;
221
222 d->actions << action;
223 d->ownsActions = true;
225
226 if (d->id >= 0) {
227 d->updateTimer.start();
228 }
229
230 return action;
231}
232
233void KNotification::setActionsQml(QList<KNotificationAction *> actions)
234{
235 if (actions == d->actions) {
236 return;
237 }
238
239 d->actions.clear();
240
241 d->needUpdate = true;
242 d->actions = actions;
243 d->ownsActions = false;
245
246 int idCounter = 1;
247
248 for (KNotificationAction *action : d->actions) {
249 action->setId(QString::number(idCounter));
250 ++idCounter;
251 }
252
253 if (d->id >= 0) {
254 d->updateTimer.start();
255 }
256}
257
259{
260 return d->replyAction.get();
261}
262
263void KNotification::setReplyAction(std::unique_ptr<KNotificationReplyAction> replyAction)
264{
265 if (replyAction == d->replyAction) {
266 return;
267 }
268
269 d->needUpdate = true;
270 d->replyAction = std::move(replyAction);
271 if (d->id >= 0) {
272 d->updateTimer.start();
273 }
274}
275
277{
278 if (d->ownsActions) {
279 delete d->defaultAction;
280 }
281
282 d->needUpdate = true;
283 d->ownsActions = true;
284 d->defaultAction = new KNotificationAction(label);
285 connect(d->defaultAction, &KNotificationAction::labelChanged, this, [this] {
286 d->needUpdate = true;
287 if (d->id >= 0) {
288 d->updateTimer.start();
289 }
290 });
291
292 d->defaultAction->setId(QStringLiteral("default"));
293
294 Q_EMIT defaultActionChanged();
295 if (d->id >= 0) {
296 d->updateTimer.start();
297 }
298
299 return d->defaultAction;
300}
301
302void KNotification::setDefaultActionQml(KNotificationAction *defaultAction)
303{
304 if (defaultAction == d->defaultAction) {
305 return;
306 }
307
308 d->needUpdate = true;
309 d->defaultAction = defaultAction;
310 d->ownsActions = false;
311
312 d->defaultAction->setId(QStringLiteral("default"));
313
315 if (d->id >= 0) {
316 d->updateTimer.start();
317 }
318}
319
321{
322 return d->defaultAction;
323}
324
326{
327 return d->flags;
328}
329
331{
332 if (d->flags == flags) {
333 return;
334 }
335
336 d->needUpdate = true;
337 d->flags = flags;
339 if (d->id >= 0) {
340 d->updateTimer.start();
341 }
342}
343
345{
346 return d->componentName;
347}
348
350{
351 if (d->componentName != c) {
352 d->componentName = c;
354 }
355}
356
358{
359 return QUrl::fromStringList(d->hints[QStringLiteral("x-kde-urls")].toStringList());
360}
361
363{
364 setHint(QStringLiteral("x-kde-urls"), QUrl::toStringList(urls));
366}
367
369{
370 return d->urgency;
371}
372
374{
375 if (d->urgency == urgency) {
376 return;
377 }
378
379 d->needUpdate = true;
380 d->urgency = urgency;
382 if (d->id >= 0) {
383 d->updateTimer.start();
384 }
385}
386
387void KNotification::activate(const QString &actionId)
388{
389 if (d->defaultAction && actionId == QLatin1String("default")) {
390 Q_EMIT d->defaultAction->activated();
391 }
392
393 for (KNotificationAction *action : d->actions) {
394 if (action->id() == actionId) {
395 Q_EMIT action->activated();
396 }
397 }
398}
399
401{
402 if (d->id >= 0) {
403 KNotificationManager::self()->close(d->id);
404 }
405
406 if (d->id == -1) {
407 d->id = -2;
408 Q_EMIT closed();
409 if (d->autoDelete) {
410 deleteLater();
411 } else {
412 // reset for being reused
413 d->isNew = true;
414 d->id = ++notificationIdCounter;
415 }
416 }
417}
418
419static QString defaultComponentName()
420{
421#if defined(Q_OS_ANDROID)
422 return QStringLiteral("android_defaults");
423#else
424 return QStringLiteral("plasma_workspace");
425#endif
426}
427
429 const QString &title,
430 const QString &text,
431 const QPixmap &pixmap,
433 const QString &componentName)
434{
435 KNotification *notify = new KNotification(eventid, flags);
436 notify->setTitle(title);
437 notify->setText(text);
438 notify->setPixmap(pixmap);
439 notify->setComponentName((flags & DefaultEvent) ? defaultComponentName() : componentName);
440
442
443 return notify;
444}
445
448{
449 return event(eventid, QString(), text, pixmap, flags, componentName);
450}
451
453{
454 return event(standardEventToEventId(eventid), title, text, pixmap, flags | DefaultEvent);
455}
456
458{
459 return event(eventid, QString(), text, pixmap, flags);
460}
461
463 const QString &title,
464 const QString &text,
465 const QString &iconName,
467 const QString &componentName)
468{
469 KNotification *notify = new KNotification(eventid, flags);
470 notify->setTitle(title);
471 notify->setText(text);
472 notify->setIconName(iconName);
473 notify->setComponentName((flags & DefaultEvent) ? defaultComponentName() : componentName);
474
476
477 return notify;
478}
479
481{
482 return event(standardEventToEventId(eventid), title, text, iconName, flags | DefaultEvent);
483}
484
486{
487 return event(standardEventToEventId(eventid), title, text, standardEventToIconName(eventid), flags | DefaultEvent);
488}
489
490void KNotification::ref()
491{
492 d->ref++;
493}
494void KNotification::deref()
495{
496 Q_ASSERT(d->ref > 0);
497 d->ref--;
498 if (d->ref == 0) {
499 d->id = -1;
500 close();
501 }
502}
503
504void KNotification::beep(const QString &reason)
505{
506 event(QStringLiteral("beep"), reason, QPixmap(), CloseOnTimeout | DefaultEvent);
507}
508
510{
511 d->needUpdate = false;
512 if (d->isNew) {
513 d->isNew = false;
514 KNotificationManager::self()->notify(this);
515 } else {
516 KNotificationManager::self()->reemit(this);
517 }
518}
519
520int KNotification::id()
521{
522 if (!d) {
523 return -1;
524 }
525 return d->id;
526}
527
529{
530 QString appname;
531
532 if (d->flags & DefaultEvent) {
533 appname = defaultComponentName();
534 } else if (!d->componentName.isEmpty()) {
535 appname = d->componentName;
536 } else {
538 }
539
540 return appname;
541}
542
544{
545 return d->autoDelete;
546}
547
549{
550 if (d->autoDelete != autoDelete) {
551 d->autoDelete = autoDelete;
553 }
554}
555
556void KNotification::update()
557{
558 if (d->needUpdate) {
559 KNotificationManager::self()->update(this);
560 }
561}
562
563QString KNotification::standardEventToEventId(KNotification::StandardEvent event)
564{
565 QString eventId;
566 switch (event) {
567 case Warning:
568 eventId = QStringLiteral("warning");
569 break;
570 case Error:
571 eventId = QStringLiteral("fatalerror");
572 break;
573 case Catastrophe:
574 eventId = QStringLiteral("catastrophe");
575 break;
576 case Notification: // fall through
577 default:
578 eventId = QStringLiteral("notification");
579 break;
580 }
581 return eventId;
582}
583
584QString KNotification::standardEventToIconName(KNotification::StandardEvent event)
585{
586 QString iconName;
587 switch (event) {
588 case Warning:
589 iconName = QStringLiteral("dialog-warning");
590 break;
591 case Error:
592 iconName = QStringLiteral("dialog-error");
593 break;
594 case Catastrophe:
595 iconName = QStringLiteral("dialog-error");
596 break;
597 case Notification: // fall through
598 default:
599 iconName = QStringLiteral("dialog-information");
600 break;
601 }
602 return iconName;
603}
604
605void KNotification::setHint(const QString &hint, const QVariant &value)
606{
607 if (value == d->hints.value(hint)) {
608 return;
609 }
610
611 d->needUpdate = true;
612 d->hints[hint] = value;
613 if (d->id >= 0) {
614 d->updateTimer.start();
615 }
617}
618
619QVariantMap KNotification::hints() const
620{
621 return d->hints;
622}
623
624void KNotification::setHints(const QVariantMap &hints)
625{
626 if (hints == d->hints) {
627 return;
628 }
629
630 d->needUpdate = true;
631 d->hints = hints;
632 if (d->id >= 0) {
633 d->updateTimer.start();
634 }
636}
637
639{
640 return d->xdgActivationToken;
641}
642
644{
645 if (window == d->window) {
646 return;
647 }
648
649 disconnect(d->window, &QWindow::activeChanged, this, &KNotification::slotWindowActiveChanged);
650 d->window = window;
651 connect(d->window, &QWindow::activeChanged, this, &KNotification::slotWindowActiveChanged);
652}
653
654void KNotification::slotWindowActiveChanged()
655{
656 if (d->window->isActive() && (d->flags & CloseWhenWindowActivated)) {
657 close();
658 }
659}
660
662{
663 return d->window;
664}
665
666#include "moc_knotification.cpp"
This class represents a notification.
void labelChanged(const QString &label)
Emitted when label changed.
void setLabel(const QString &label)
Set the user-facing label for the action.
void activated()
Emitted when the user activates the action.
KNotification is the main class for creating notifications.
void defaultActionChanged()
Emitted when defaultAction changed.
void clearActions()
Removes all actions previously added by addAction() from the notification.
QFlags< NotificationFlag > NotificationFlags
Stores a combination of NotificationFlag values.
void sendEvent()
Send the notification to the server.
QWindow * window() const
The window associated with this notification.
KNotificationReplyAction * replyAction() const
void urgencyChanged()
Emitted when urgency changed.
void eventIdChanged()
Emitted when eventId changed.
QString title
Set the title of the notification popup.
void setTitle(const QString &title)
Set the title of the notification popup.
void setEventId(const QString &eventId)
Set the event id, if not already passed to the constructor.
KNotification(const QString &eventId, NotificationFlags flags=CloseOnTimeout, QObject *parent=nullptr)
Create a new notification.
KNotificationAction * defaultAction() const
void iconNameChanged()
Emitted when iconName changed.
void closed()
Emitted when the notification is closed.
QPixmap pixmap() const
void setAutoDelete(bool autoDelete)
Sets whether this notification object will be automatically deleted after closing.
void actionsChanged()
Emitted when actions changed.
void close()
Close the notification without activating it.
void hintsChanged()
Emitted when hints changes.
QVariantMap hints
static KNotification * event(const QString &eventId, const QString &title, const QString &text, const QPixmap &pixmap=QPixmap(), const NotificationFlags &flags=CloseOnTimeout, const QString &componentName=QString())
emit an event
NotificationFlags flags
Set the notification flags.
Urgency
The urgency of a notification.
static void beep(const QString &reason=QString())
This is a simple substitution for QApplication::beep()
void setFlags(const NotificationFlags &flags)
Set the notification flags.
void titleChanged()
Emitted when title changed.
void setPixmap(const QPixmap &pix)
Set the pixmap that will be shown in the popup.
QString eventId
Set the event id, if not already passed to the constructor.
QString text
Set the notification text that will appear in the popup.
Urgency urgency
Sets the urgency of the notification.
void autoDeleteChanged()
Emitted when autoDelete changed.
Q_INVOKABLE void setHint(const QString &hint, const QVariant &value)
void setIconName(const QString &icon)
Set the icon that will be shown in the popup.
QString appName() const
void setComponentName(const QString &componentName)
The componentData is used to determine the location of the config file.
StandardEvent
default events you can use in the event function
bool isAutoDelete() const
Returns whether this notification object will be automatically deleted after closing.
KNotificationAction * addAction(const QString &label)
Add an action to the notification.
QString iconName
Set the icon that will be shown in the popup.
void textChanged()
Emitted when text changed.
void urlsChanged()
Emitted when urls changed.
bool autoDelete
Sets whether this notification object will be automatically deleted after closing.
void flagsChanged()
Emitted when flags changed.
QList< QUrl > urls
Sets URLs associated with this notification.
void setUrls(const QList< QUrl > &urls)
Sets URLs associated with this notification.
QString xdgActivationToken
void setWindow(QWindow *window)
Sets the window associated with this notification.
@ CloseOnTimeout
The notification will be automatically closed after a timeout.
@ CloseWhenWindowActivated
The notification will be automatically closed if the window() becomes activated.
void componentNameChanged()
Emitted when componentName changed.
void setReplyAction(std::unique_ptr< KNotificationReplyAction > replyAction)
Add an inline reply action to the notification.
QString componentName
The componentData is used to determine the location of the config file.
void setUrgency(Urgency urgency)
Sets the urgency of the notification.
KNotificationAction * addDefaultAction(const QString &label)
Add a default action that will be triggered when the notification is activated (typically,...
void setHints(const QVariantMap &hints)
void setText(const QString &text)
Set the notification text that will appear in the popup.
QObject(QObject *parent)
Q_EMITQ_EMIT
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
void deleteLater()
bool disconnect(const QMetaObject::Connection &connection)
QObject * parent() const const
QString number(double n, char format, int precision)
void timeout()
QList< QUrl > fromStringList(const QStringList &urls, ParsingMode mode)
QStringList toStringList(const QList< QUrl > &urls, FormattingOptions options)
T value() const const
void activeChanged()
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 12:04:04 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.