electron/shell/browser/notifications/linux/libnotify_notification.cc

190 lines
6.1 KiB
C++
Raw Normal View History

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.
#include "shell/browser/notifications/linux/libnotify_notification.h"
2015-12-25 03:52:19 +00:00
#include <set>
#include <string>
#include <vector>
2015-12-25 03:52:19 +00:00
#include "base/files/file_enumerator.h"
#include "base/logging.h"
2015-12-25 03:52:19 +00:00
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "chrome/browser/ui/gtk/gtk_util.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"
2015-12-25 03:52:19 +00:00
#include "third_party/skia/include/core/SkBitmap.h"
namespace electron {
2015-12-25 03:52:19 +00:00
namespace {
2016-05-17 11:30:53 +00:00
LibNotifyLoader libnotify_loader_;
2015-12-25 03:52:19 +00:00
const std::set<std::string>& GetServerCapabilities() {
static std::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().count(capability) != 0;
}
bool NotifierSupportsActions() {
2015-12-25 03:52:19 +00:00
if (getenv("ELECTRON_USE_UBUNTU_NOTIFIER"))
return false;
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")) {
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())) {
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)
2018-04-18 01:56:12 +00:00
: Notification(delegate, presenter), notification_(nullptr) {}
2015-12-25 03:52:19 +00:00
LibnotifyNotification::~LibnotifyNotification() {
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
2018-04-18 01:56:12 +00:00
g_signal_connect(notification_, "closed",
G_CALLBACK(OnNotificationClosedThunk), this);
2015-12-25 03:52:19 +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(
notification_, "default", "View", OnNotificationViewThunk, this,
nullptr);
}
NotifyUrgency urgency = NOTIFY_URGENCY_NORMAL;
if (options.urgency == base::ASCIIToUTF16("critical")) {
urgency = NOTIFY_URGENCY_CRITICAL;
} else if (options.urgency == base::ASCIIToUTF16("low")) {
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()) {
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);
}
// Set the timeout duration for the notification
bool neverTimeout = options.timeout_type == base::ASCIIToUTF16("never");
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());
g_object_set(G_OBJECT(notification_), "id", id, NULL);
}
// 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");
} else if (HasCapability("x-canonical-append")) {
libnotify_loader_.notify_notification_set_hint_string(
notification_, "x-canonical-append", "true");
}
// Send the desktop name to identify the application
// The desktop-entry is the part before the .desktop
std::string desktop_id;
if (platform_util::GetDesktopName(&desktop_id)) {
const std::string suffix{".desktop"};
if (base::EndsWith(desktop_id, suffix,
base::CompareCase::INSENSITIVE_ASCII)) {
desktop_id.resize(desktop_id.size() - suffix.size());
}
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() {
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) {
NotificationDismissed();
2015-12-25 03:52:19 +00:00
}
2018-04-18 01:56:12 +00:00
void LibnotifyNotification::OnNotificationView(NotifyNotification* notification,
char* action) {
NotificationClicked();
2015-12-25 03:52:19 +00:00
}
} // namespace electron