electron/brightray/browser/linux/notification_presenter_linux.cc

175 lines
5.4 KiB
C++
Raw Normal View History

// Copyright (c) 2012 The Chromium Authors. All rights reserved.
2013-12-19 05:20:00 +00:00
// Copyright (c) 2013 Patrick Reynolds <piki@github.com>. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE-CHROMIUM file.
#include "browser/linux/notification_presenter_linux.h"
#include "base/bind.h"
#include "base/logging.h"
#include "base/files/file_enumerator.h"
#include "base/strings/utf_string_conversions.h"
#include "chrome/browser/ui/libgtk2ui/skia_utils_gtk2.h"
#include "content/public/browser/desktop_notification_delegate.h"
2015-03-14 18:34:53 +00:00
#include "content/public/common/platform_notification_data.h"
#include "common/application_info.h"
#include <stdlib.h>
2015-08-04 08:43:26 +00:00
#include "third_party/skia/include/core/SkBitmap.h"
using namespace base;
using namespace std;
namespace brightray {
namespace {
static bool unity_has_result = false;
static bool unity_result = false;
static bool UnityIsRunning() {
FileEnumerator* enumerator = NULL;
if (unity_has_result) {
return unity_result;
}
if (getenv("ELECTRON_USE_UBUNTU_NOTIFIER")) {
unity_result = true;
goto out;
}
// Look for the presence of libunity as our hint that we're under Ubuntu
FilePath haystack;
string needle("/usr/lib/libunity-");
enumerator = new FileEnumerator(FilePath("/usr/lib"), false, FileEnumerator::FILES);
while (!((haystack = enumerator->Next()).empty())) {
if (haystack.value().compare(0, needle.length(), needle) == 0) {
unity_result = true;
goto out;
}
}
unity_result = false;
out:
if (enumerator) delete enumerator;
unity_has_result = true;
return unity_result;
}
void log_and_clear_error(GError* error, const char* context) {
2013-12-19 22:01:01 +00:00
LOG(ERROR) << context
<< ": domain=" << error->domain
<< " code=" << error->code
<< " message=\"" << error->message << '"';
g_error_free(error);
}
content::DesktopNotificationDelegate* GetDelegateFromNotification(
NotifyNotification* notification) {
return static_cast<content::DesktopNotificationDelegate*>(
g_object_get_data(G_OBJECT(notification), "delegate"));
}
} // namespace
// static
NotificationPresenter* NotificationPresenter::Create() {
if (!notify_is_initted()) {
notify_init(GetApplicationName().c_str());
}
return new NotificationPresenterLinux;
}
NotificationPresenterLinux::NotificationPresenterLinux()
: notifications_(nullptr) {
}
2013-12-19 05:05:19 +00:00
NotificationPresenterLinux::~NotificationPresenterLinux() {
2013-12-19 05:14:08 +00:00
// unref any outstanding notifications, and then free the list.
if (notifications_)
g_list_free_full(notifications_, g_object_unref);
2013-12-19 05:05:19 +00:00
}
void NotificationPresenterLinux::ShowNotification(
2015-03-14 18:34:53 +00:00
const content::PlatformNotificationData& data,
const SkBitmap& icon,
2014-10-13 03:02:56 +00:00
scoped_ptr<content::DesktopNotificationDelegate> delegate_ptr,
base::Closure* cancel_callback) {
2015-03-14 18:34:53 +00:00
std::string title = base::UTF16ToUTF8(data.title);
std::string body = base::UTF16ToUTF8(data.body);
NotifyNotification* notification = notify_notification_new(title.c_str(), body.c_str(), nullptr);
2014-10-13 03:02:56 +00:00
content::DesktopNotificationDelegate* delegate = delegate_ptr.release();
g_object_set_data_full(G_OBJECT(notification), "delegate", delegate, operator delete);
g_signal_connect(notification, "closed", G_CALLBACK(OnNotificationClosedThunk), this);
// NB: On Unity, adding a notification action will cause the notification
// to display as a modal dialog box. Testing for distros that have "Unity
// Zen Nature" is difficult, we will test for the presence of the indicate
// dbus service
if (!UnityIsRunning()) {
2015-06-05 04:07:27 +00:00
notify_notification_add_action(
notification, "default", "View", OnNotificationViewThunk, this, nullptr);
}
2015-08-04 08:43:26 +00:00
if (!icon.drawsNothing()) {
GdkPixbuf* pixbuf = libgtk2ui::GdkPixbufFromSkBitmap(icon);
notify_notification_set_image_from_pixbuf(notification, pixbuf);
notify_notification_set_timeout(notification, NOTIFY_EXPIRES_DEFAULT);
g_object_unref(pixbuf);
}
GError* error = nullptr;
notify_notification_show(notification, &error);
if (error) {
log_and_clear_error(error, "notify_notification_show");
2015-01-28 01:18:46 +00:00
g_object_unref(notification);
return;
}
notifications_ = g_list_append(notifications_, notification);
delegate->NotificationDisplayed();
if (cancel_callback)
*cancel_callback = base::Bind(
&NotificationPresenterLinux::CancelNotification,
base::Unretained(this),
notification);
}
void NotificationPresenterLinux::CancelNotification(NotifyNotification* notification) {
GError* error = nullptr;
notify_notification_close(notification, &error);
if (error)
log_and_clear_error(error, "notify_notification_close");
GetDelegateFromNotification(notification)->NotificationClosed();
DeleteNotification(notification);
}
void NotificationPresenterLinux::DeleteNotification(NotifyNotification* notification) {
notifications_ = g_list_remove(notifications_, notification);
g_object_unref(notification);
}
void NotificationPresenterLinux::OnNotificationClosed(NotifyNotification* notification) {
2015-08-04 08:44:32 +00:00
if (!notification)
return;
GetDelegateFromNotification(notification)->NotificationClosed();
DeleteNotification(notification);
2013-12-19 05:05:19 +00:00
}
void NotificationPresenterLinux::OnNotificationView(
NotifyNotification* notification, char* action) {
2015-08-04 08:44:32 +00:00
if (!notification)
return;
GetDelegateFromNotification(notification)->NotificationClick();
DeleteNotification(notification);
}
} // namespace brightray