/*
* Copyright (C) 2023, KylinSoft Co., Ltd.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*
* Authors: iaom
*/
#include
#include
#include "server.h"
#include "server-private.h"
#include "notificationsadaptor.h"
#include "notificationserveradaptor.h"
#include "notification-server-config.h"
#include "utils.h"
#include "notification-close-reason.h"
#include "notification-settings/settings-manager.h"
using namespace NotificationServer;
ServerPrivate::ServerPrivate(QObject *parent) : QObject(parent), m_notificationWatchers(new QDBusServiceWatcher(this))
{
m_notificationWatchers->setConnection(QDBusConnection::sessionBus());
m_notificationWatchers->setWatchMode(QDBusServiceWatcher::WatchForUnregistration);
connect(m_notificationWatchers, &QDBusServiceWatcher::serviceUnregistered,
m_notificationWatchers, &QDBusServiceWatcher::removeWatchedService);
}
ServerPrivate::~ServerPrivate() = default;
QStringList ServerPrivate::GetCapabilities() const
{
return QStringList();
}
uint ServerPrivate::Notify(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)
{
uint id = 0;
if(replaces_id > 0) {
id = replaces_id;
} else {
id = m_increasedNotificationId++;
}
QVariantMap newHints = hints;
newHints.insert(QStringLiteral("x-ukui-createdTime"), QDateTime::currentDateTime().toString());
uint pid = 0;
QDBusReply pidReply = connection().interface()->servicePid(message().service());
if(pidReply.isValid()) {
pid = pidReply.value();
}
if(newHints.value(QStringLiteral("x-ukui-display")).toString().isEmpty() && pid > 0) {
newHints.insert(QStringLiteral("x-ukui-display"), UkuiNotification::Utils::displayFromPid(pid));
}
//如果desktop-entry只有文件名,需要获取全路径
QString desktopEntry = newHints.value(QStringLiteral("desktop-entry")).toString();
if(!desktopEntry.isEmpty()) {
desktopEntry = UkuiNotification::Utils::desktopEntryFromName(desktopEntry);
}
if(desktopEntry.isEmpty() && pid > 0) {
desktopEntry = UkuiNotification::Utils::desktopEntryFromPid(pid);
}
newHints.insert(QStringLiteral("desktop-entry"), desktopEntry);
qDebug() << "New message received:" << app_name << id << app_icon << summary << body
<< actions << hints << timeout << "new hints" << newHints;
if(m_notificationWatchers->watchedServices().isEmpty()) {
Notification notification;
notification.m_appName = app_name;
notification.m_id = id;
notification.m_appIcon = app_icon;
notification.m_summary = summary;
notification.m_body = body;
notification.m_actions = actions;
notification.m_hints = newHints;
m_notificationsCache.insert(id, notification);
return id;
}
for(const QString &service : m_notificationWatchers->watchedServices()) {
QDBusMessage msg = QDBusMessage::createMethodCall(service,
QStringLiteral("/NotificationClient"),
QStringLiteral("org.ukui.NotificationClient"),
QStringLiteral("Notify"));
msg.setArguments({app_name, id, app_icon, summary, body, actions, newHints, timeout});
QDBusConnection::sessionBus().call(msg, QDBus::NoBlock);
}
return id;
}
QString ServerPrivate::GetServerInformation(QString &vendor, QString &version, QString &spec_version) const
{
vendor = QStringLiteral("Kylin");
version = QLatin1String(NOTIFICATION_SERVER_VERSION);
spec_version = QStringLiteral("1.2");
return QStringLiteral("UKUI");
}
void ServerPrivate::CloseNotification(uint id)
{
if(m_notificationsCache.contains(id)) {
m_notificationsCache.remove(id);
}
qDebug() << "CloseNotification(Revoked)" << id;
Q_EMIT NotificationClosed(id, UkuiNotification::NotificationCloseReason::Revoked);
}
bool ServerPrivate::init()
{
new NotificationsAdaptor(this);
new NotificationServerAdaptor(this);
QDBusConnection conn = QDBusConnection::sessionBus();
auto registration = conn.interface()->registerService(notificationServiceName(),
QDBusConnectionInterface::ReplaceExistingService,
QDBusConnectionInterface::DontAllowReplacement);
if (registration.value() != QDBusConnectionInterface::ServiceRegistered) {
qWarning() << "Failed to register Notification service on DBus!";
return false;
}
if(!conn.registerObject(notificationServicePath(), this)) {
qWarning() << "Failed to register Notification DBus object!" << conn.lastError().message();
return false;
}
UkuiNotification::SettingsManager::self();
return true;
}
QString ServerPrivate::notificationServiceName()
{
return QStringLiteral("org.freedesktop.Notifications");
}
QString ServerPrivate::notificationServicePath()
{
return QStringLiteral("/org/freedesktop/Notifications");
}
QString ServerPrivate::notificationServiceInterface()
{
return QStringLiteral("org.freedesktop.Notifications");;
}
void ServerPrivate::RegisterClient()
{
m_notificationWatchers->addWatchedService(message().service());
qDebug() << "Watched services:" << m_notificationWatchers->watchedServices();
sendCache();
}
void ServerPrivate::UnRegisterClient()
{
m_notificationWatchers->removeWatchedService(message().service());
}
void ServerPrivate::CloseNotification(uint id, uint reason)
{
if(m_notificationsCache.contains(id)) {
m_notificationsCache.remove(id);
}
qDebug() << "CloseNotification" << id << reason;
Q_EMIT NotificationClosed(id, reason);
}
void ServerPrivate::InvokeAction(uint id, const QString &action_key)
{
qDebug() << "InvokeAction" << id << action_key;
Q_EMIT ActionInvoked(id, action_key);
}
void ServerPrivate::sendCache()
{
for(const uint &id : m_notificationsCache.keys()) {
auto notification = m_notificationsCache.take(id);
for(const QString &service : m_notificationWatchers->watchedServices()) {
QDBusMessage msg = QDBusMessage::createMethodCall(service,
QStringLiteral("/NotificationClient"),
QStringLiteral("org.ukui.NotificationClient"),
QStringLiteral("Notify"));
msg.setArguments({notification.m_appName, notification.m_id, notification.m_appIcon, notification.m_summary,
notification.m_body, notification.m_actions, notification.m_hints, notification.m_timeout});
QDBusConnection::sessionBus().call(msg, QDBus::NoBlock);
}
}
}
void ServerPrivate::UpdateUnreadMessagesNumber(const QString &desktopEntry, uint number)
{
Q_EMIT UnreadMessageNumberUpdated(desktopEntry, number);
}
Server::Server(QObject *parent): QObject(parent), d(new ServerPrivate(this))
{
}
Server &Server::self()
{
static Server self;
return self;
}
bool Server::init()
{
return d->init();
}
Server::~Server() = default;