123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193 |
- // Copyright (c) 2015 GitHub, Inc.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- #include "shell/browser/notifications/linux/libnotify_notification.h"
- #include <string>
- #include "base/containers/flat_set.h"
- #include "base/files/file_enumerator.h"
- #include "base/functional/bind.h"
- #include "base/logging.h"
- #include "base/process/process_handle.h"
- #include "base/strings/utf_string_conversions.h"
- #include "shell/browser/notifications/notification_delegate.h"
- #include "shell/browser/ui/gtk_util.h"
- #include "shell/common/application_info.h"
- #include "shell/common/platform_util.h"
- #include "third_party/skia/include/core/SkBitmap.h"
- #include "ui/gtk/gtk_util.h" // nogncheck
- namespace electron {
- namespace {
- LibNotifyLoader libnotify_loader_;
- const base::flat_set<std::string>& GetServerCapabilities() {
- static base::flat_set<std::string> caps;
- if (caps.empty()) {
- auto* capabilities = libnotify_loader_.notify_get_server_caps();
- for (auto* l = capabilities; l != nullptr; l = l->next)
- caps.insert(static_cast<const char*>(l->data));
- g_list_free_full(capabilities, g_free);
- }
- return caps;
- }
- bool HasCapability(const std::string& capability) {
- return GetServerCapabilities().contains(capability);
- }
- bool NotifierSupportsActions() {
- if (getenv("ELECTRON_USE_UBUNTU_NOTIFIER"))
- return false;
- return HasCapability("actions");
- }
- void log_and_clear_error(GError* error, const char* context) {
- LOG(ERROR) << context << ": domain=" << error->domain
- << " code=" << error->code << " message=\"" << error->message
- << '"';
- g_error_free(error);
- }
- } // namespace
- // static
- bool LibnotifyNotification::Initialize() {
- if (!libnotify_loader_.Load("libnotify.so.4") && // most common one
- !libnotify_loader_.Load("libnotify.so.5") &&
- !libnotify_loader_.Load("libnotify.so.1") &&
- !libnotify_loader_.Load("libnotify.so")) {
- LOG(WARNING) << "Unable to find libnotify; notifications disabled";
- return false;
- }
- if (!libnotify_loader_.notify_is_initted() &&
- !libnotify_loader_.notify_init(GetApplicationName().c_str())) {
- LOG(WARNING) << "Unable to initialize libnotify; notifications disabled";
- return false;
- }
- return true;
- }
- LibnotifyNotification::LibnotifyNotification(NotificationDelegate* delegate,
- NotificationPresenter* presenter)
- : Notification(delegate, presenter) {}
- LibnotifyNotification::~LibnotifyNotification() {
- if (notification_) {
- g_signal_handlers_disconnect_by_data(notification_, this);
- g_object_unref(notification_);
- }
- }
- void LibnotifyNotification::Show(const NotificationOptions& options) {
- notification_ = libnotify_loader_.notify_notification_new(
- base::UTF16ToUTF8(options.title).c_str(),
- base::UTF16ToUTF8(options.msg).c_str(), nullptr);
- signal_ = ScopedGSignal(
- notification_, "closed",
- base::BindRepeating(&LibnotifyNotification::OnNotificationClosed,
- base::Unretained(this)));
- // 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()) {
- libnotify_loader_.notify_notification_add_action(
- notification_, "default", "View", OnNotificationView, this, nullptr);
- }
- NotifyUrgency urgency = NOTIFY_URGENCY_NORMAL;
- if (options.urgency == u"critical") {
- urgency = NOTIFY_URGENCY_CRITICAL;
- } else if (options.urgency == u"low") {
- urgency = NOTIFY_URGENCY_LOW;
- }
- // Set the urgency level of the notification.
- libnotify_loader_.notify_notification_set_urgency(notification_, urgency);
- if (!options.icon.drawsNothing()) {
- GdkPixbuf* pixbuf = gtk_util::GdkPixbufFromSkBitmap(options.icon);
- libnotify_loader_.notify_notification_set_image_from_pixbuf(notification_,
- pixbuf);
- g_object_unref(pixbuf);
- }
- // Set the timeout duration for the notification
- bool neverTimeout = options.timeout_type == u"never";
- int timeout = (neverTimeout) ? NOTIFY_EXPIRES_NEVER : NOTIFY_EXPIRES_DEFAULT;
- libnotify_loader_.notify_notification_set_timeout(notification_, timeout);
- if (!options.tag.empty()) {
- GQuark id = g_quark_from_string(options.tag.c_str());
- g_object_set(G_OBJECT(notification_), "id", id, nullptr);
- }
- // Always try to append notifications.
- // Unique tags can be used to prevent this.
- if (HasCapability("append")) {
- libnotify_loader_.notify_notification_set_hint(
- notification_, "append", g_variant_new_string("true"));
- } else if (HasCapability("x-canonical-append")) {
- libnotify_loader_.notify_notification_set_hint(
- notification_, "x-canonical-append", g_variant_new_string("true"));
- }
- // Send the desktop name to identify the application
- // The desktop-entry is the part before the .desktop
- std::string desktop_id = platform_util::GetXdgAppId();
- if (!desktop_id.empty()) {
- libnotify_loader_.notify_notification_set_hint(
- notification_, "desktop-entry",
- g_variant_new_string(desktop_id.c_str()));
- }
- libnotify_loader_.notify_notification_set_hint(
- notification_, "sender-pid",
- g_variant_new_int64(base::GetCurrentProcId()));
- GError* error = nullptr;
- libnotify_loader_.notify_notification_show(notification_, &error);
- if (error) {
- log_and_clear_error(error, "notify_notification_show");
- NotificationFailed();
- return;
- }
- if (delegate())
- delegate()->NotificationDisplayed();
- }
- void LibnotifyNotification::Dismiss() {
- if (!notification_) {
- return;
- }
- GError* error = nullptr;
- on_dismissing_ = true;
- libnotify_loader_.notify_notification_close(notification_, &error);
- if (error) {
- log_and_clear_error(error, "notify_notification_close");
- }
- on_dismissing_ = false;
- }
- void LibnotifyNotification::OnNotificationClosed(
- NotifyNotification* notification) {
- NotificationDismissed(!on_dismissing_);
- }
- void LibnotifyNotification::OnNotificationView(NotifyNotification* notification,
- char* action,
- gpointer user_data) {
- LibnotifyNotification* that = static_cast<LibnotifyNotification*>(user_data);
- DCHECK(that);
- that->NotificationClicked();
- }
- } // namespace electron
|