2015-12-25 03:52:19 +00:00
|
|
|
// Copyright (c) 2015 GitHub, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2019-06-19 20:46:59 +00:00
|
|
|
#include "shell/browser/notifications/linux/libnotify_notification.h"
|
2015-12-25 03:52:19 +00:00
|
|
|
|
2018-02-20 13:53:10 +00:00
|
|
|
#include <string>
|
2017-06-23 10:39:42 +00:00
|
|
|
|
2024-01-05 11:18:31 +00:00
|
|
|
#include "base/containers/flat_set.h"
|
2015-12-25 03:52:19 +00:00
|
|
|
#include "base/files/file_enumerator.h"
|
2023-10-05 23:59:39 +00:00
|
|
|
#include "base/functional/bind.h"
|
2018-02-20 13:53:10 +00:00
|
|
|
#include "base/logging.h"
|
2015-12-25 03:52:19 +00:00
|
|
|
#include "base/strings/utf_string_conversions.h"
|
2019-06-19 20:46:59 +00:00
|
|
|
#include "shell/browser/notifications/notification_delegate.h"
|
2019-10-28 22:12:35 +00:00
|
|
|
#include "shell/browser/ui/gtk_util.h"
|
2019-06-19 20:46:59 +00:00
|
|
|
#include "shell/common/application_info.h"
|
|
|
|
#include "shell/common/platform_util.h"
|
2015-12-25 03:52:19 +00:00
|
|
|
#include "third_party/skia/include/core/SkBitmap.h"
|
2022-04-18 04:24:32 +00:00
|
|
|
#include "ui/gtk/gtk_util.h" // nogncheck
|
2015-12-25 03:52:19 +00:00
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
namespace electron {
|
2015-12-25 03:52:19 +00:00
|
|
|
|
|
|
|
namespace {
|
2016-05-17 11:30:53 +00:00
|
|
|
|
2016-04-13 02:43:22 +00:00
|
|
|
LibNotifyLoader libnotify_loader_;
|
2015-12-25 03:52:19 +00:00
|
|
|
|
2024-01-05 11:18:31 +00:00
|
|
|
const base::flat_set<std::string>& GetServerCapabilities() {
|
|
|
|
static base::flat_set<std::string> caps;
|
2018-02-20 13:53:10 +00:00
|
|
|
if (caps.empty()) {
|
2018-04-17 22:41:47 +00:00
|
|
|
auto* capabilities = libnotify_loader_.notify_get_server_caps();
|
|
|
|
for (auto* l = capabilities; l != nullptr; l = l->next)
|
2018-02-20 13:53:10 +00:00
|
|
|
caps.insert(static_cast<const char*>(l->data));
|
|
|
|
g_list_free_full(capabilities, g_free);
|
|
|
|
}
|
|
|
|
return caps;
|
|
|
|
}
|
2016-04-14 16:24:00 +00:00
|
|
|
|
2018-02-20 13:53:10 +00:00
|
|
|
bool HasCapability(const std::string& capability) {
|
2024-01-05 11:18:31 +00:00
|
|
|
return GetServerCapabilities().contains(capability);
|
2016-04-14 16:24:00 +00:00
|
|
|
}
|
|
|
|
|
2016-04-13 02:43:22 +00:00
|
|
|
bool NotifierSupportsActions() {
|
2015-12-25 03:52:19 +00:00
|
|
|
if (getenv("ELECTRON_USE_UBUNTU_NOTIFIER"))
|
2016-04-13 02:43:22 +00:00
|
|
|
return false;
|
|
|
|
|
2018-02-20 13:53:10 +00:00
|
|
|
return HasCapability("actions");
|
2015-12-25 03:52:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void log_and_clear_error(GError* error, const char* context) {
|
2018-04-18 01:56:12 +00:00
|
|
|
LOG(ERROR) << context << ": domain=" << error->domain
|
|
|
|
<< " code=" << error->code << " message=\"" << error->message
|
|
|
|
<< '"';
|
2015-12-25 03:52:19 +00:00
|
|
|
g_error_free(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
// static
|
|
|
|
bool LibnotifyNotification::Initialize() {
|
2016-05-17 11:30:53 +00:00
|
|
|
if (!libnotify_loader_.Load("libnotify.so.4") && // most common one
|
|
|
|
!libnotify_loader_.Load("libnotify.so.5") &&
|
2015-12-25 03:52:19 +00:00
|
|
|
!libnotify_loader_.Load("libnotify.so.1") &&
|
|
|
|
!libnotify_loader_.Load("libnotify.so")) {
|
2018-02-20 13:53:10 +00:00
|
|
|
LOG(WARNING) << "Unable to find libnotify; notifications disabled";
|
2015-12-25 03:52:19 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!libnotify_loader_.notify_is_initted() &&
|
2018-10-24 10:49:10 +00:00
|
|
|
!libnotify_loader_.notify_init(GetApplicationName().c_str())) {
|
2018-02-20 13:53:10 +00:00
|
|
|
LOG(WARNING) << "Unable to initialize libnotify; notifications disabled";
|
2015-12-25 03:52:19 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
LibnotifyNotification::LibnotifyNotification(NotificationDelegate* delegate,
|
|
|
|
NotificationPresenter* presenter)
|
2021-01-26 18:16:21 +00:00
|
|
|
: Notification(delegate, presenter) {}
|
2015-12-25 03:52:19 +00:00
|
|
|
|
|
|
|
LibnotifyNotification::~LibnotifyNotification() {
|
2016-10-25 13:32:06 +00:00
|
|
|
if (notification_) {
|
|
|
|
g_signal_handlers_disconnect_by_data(notification_, this);
|
|
|
|
g_object_unref(notification_);
|
|
|
|
}
|
2015-12-25 03:52:19 +00:00
|
|
|
}
|
|
|
|
|
2017-06-24 11:03:27 +00:00
|
|
|
void LibnotifyNotification::Show(const NotificationOptions& options) {
|
2015-12-25 03:52:19 +00:00
|
|
|
notification_ = libnotify_loader_.notify_notification_new(
|
2017-06-24 11:03:27 +00:00
|
|
|
base::UTF16ToUTF8(options.title).c_str(),
|
2018-04-18 01:56:12 +00:00
|
|
|
base::UTF16ToUTF8(options.msg).c_str(), nullptr);
|
2015-12-25 03:52:19 +00:00
|
|
|
|
2023-10-05 23:59:39 +00:00
|
|
|
signal_ = ScopedGSignal(
|
|
|
|
notification_, "closed",
|
|
|
|
base::BindRepeating(&LibnotifyNotification::OnNotificationClosed,
|
|
|
|
base::Unretained(this)));
|
2015-12-25 03:52:19 +00:00
|
|
|
|
2016-04-13 02:43:22 +00:00
|
|
|
// NB: On Unity and on any other DE using Notify-OSD, adding a notification
|
|
|
|
// action will cause the notification to display as a modal dialog box.
|
|
|
|
if (NotifierSupportsActions()) {
|
2015-12-25 03:52:19 +00:00
|
|
|
libnotify_loader_.notify_notification_add_action(
|
2023-10-05 23:59:39 +00:00
|
|
|
notification_, "default", "View", OnNotificationView, this, nullptr);
|
2015-12-25 03:52:19 +00:00
|
|
|
}
|
|
|
|
|
2019-09-19 05:35:20 +00:00
|
|
|
NotifyUrgency urgency = NOTIFY_URGENCY_NORMAL;
|
2021-04-27 21:27:34 +00:00
|
|
|
if (options.urgency == u"critical") {
|
2019-09-19 05:35:20 +00:00
|
|
|
urgency = NOTIFY_URGENCY_CRITICAL;
|
2021-04-27 21:27:34 +00:00
|
|
|
} else if (options.urgency == u"low") {
|
2019-09-19 05:35:20 +00:00
|
|
|
urgency = NOTIFY_URGENCY_LOW;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the urgency level of the notification.
|
|
|
|
libnotify_loader_.notify_notification_set_urgency(notification_, urgency);
|
|
|
|
|
2017-06-24 11:03:27 +00:00
|
|
|
if (!options.icon.drawsNothing()) {
|
2019-10-28 22:12:35 +00:00
|
|
|
GdkPixbuf* pixbuf = gtk_util::GdkPixbufFromSkBitmap(options.icon);
|
2018-04-18 01:56:12 +00:00
|
|
|
libnotify_loader_.notify_notification_set_image_from_pixbuf(notification_,
|
|
|
|
pixbuf);
|
2015-12-25 03:52:19 +00:00
|
|
|
g_object_unref(pixbuf);
|
|
|
|
}
|
|
|
|
|
2019-10-09 15:22:21 +00:00
|
|
|
// Set the timeout duration for the notification
|
2021-04-27 21:27:34 +00:00
|
|
|
bool neverTimeout = options.timeout_type == u"never";
|
2019-10-09 15:22:21 +00:00
|
|
|
int timeout = (neverTimeout) ? NOTIFY_EXPIRES_NEVER : NOTIFY_EXPIRES_DEFAULT;
|
|
|
|
libnotify_loader_.notify_notification_set_timeout(notification_, timeout);
|
|
|
|
|
2017-06-24 11:03:27 +00:00
|
|
|
if (!options.tag.empty()) {
|
|
|
|
GQuark id = g_quark_from_string(options.tag.c_str());
|
2023-10-03 19:26:35 +00:00
|
|
|
g_object_set(G_OBJECT(notification_), "id", id, nullptr);
|
2016-04-13 04:08:35 +00:00
|
|
|
}
|
|
|
|
|
2016-04-14 16:31:02 +00:00
|
|
|
// Always try to append notifications.
|
|
|
|
// Unique tags can be used to prevent this.
|
|
|
|
if (HasCapability("append")) {
|
2018-04-18 01:56:12 +00:00
|
|
|
libnotify_loader_.notify_notification_set_hint_string(notification_,
|
|
|
|
"append", "true");
|
2016-04-14 16:31:02 +00:00
|
|
|
} else if (HasCapability("x-canonical-append")) {
|
|
|
|
libnotify_loader_.notify_notification_set_hint_string(
|
|
|
|
notification_, "x-canonical-append", "true");
|
|
|
|
}
|
|
|
|
|
2018-03-12 00:33:06 +00:00
|
|
|
// Send the desktop name to identify the application
|
|
|
|
// The desktop-entry is the part before the .desktop
|
2022-07-11 18:26:18 +00:00
|
|
|
std::string desktop_id = platform_util::GetXdgAppId();
|
|
|
|
if (!desktop_id.empty()) {
|
2018-03-12 00:33:06 +00:00
|
|
|
libnotify_loader_.notify_notification_set_hint_string(
|
|
|
|
notification_, "desktop-entry", desktop_id.c_str());
|
|
|
|
}
|
|
|
|
|
2015-12-25 03:52:19 +00:00
|
|
|
GError* error = nullptr;
|
|
|
|
libnotify_loader_.notify_notification_show(notification_, &error);
|
|
|
|
if (error) {
|
|
|
|
log_and_clear_error(error, "notify_notification_show");
|
|
|
|
NotificationFailed();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-05-30 09:06:51 +00:00
|
|
|
if (delegate())
|
|
|
|
delegate()->NotificationDisplayed();
|
2015-12-25 03:52:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LibnotifyNotification::Dismiss() {
|
2016-10-28 18:24:47 +00:00
|
|
|
if (!notification_) {
|
|
|
|
Destroy();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-12-25 03:52:19 +00:00
|
|
|
GError* error = nullptr;
|
|
|
|
libnotify_loader_.notify_notification_close(notification_, &error);
|
|
|
|
if (error) {
|
|
|
|
log_and_clear_error(error, "notify_notification_close");
|
|
|
|
Destroy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LibnotifyNotification::OnNotificationClosed(
|
|
|
|
NotifyNotification* notification) {
|
2016-04-15 07:14:13 +00:00
|
|
|
NotificationDismissed();
|
2015-12-25 03:52:19 +00:00
|
|
|
}
|
|
|
|
|
2018-04-18 01:56:12 +00:00
|
|
|
void LibnotifyNotification::OnNotificationView(NotifyNotification* notification,
|
2023-10-05 23:59:39 +00:00
|
|
|
char* action,
|
|
|
|
gpointer user_data) {
|
|
|
|
LibnotifyNotification* that = static_cast<LibnotifyNotification*>(user_data);
|
|
|
|
DCHECK(that);
|
|
|
|
that->NotificationClicked();
|
2015-12-25 03:52:19 +00:00
|
|
|
}
|
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
} // namespace electron
|