Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(27)

Side by Side Diff: chrome/browser/notifications/notification_platform_bridge_linux.cc

Issue 2803873003: Linux native notifications: Support closing and updating notifications (Closed)
Patch Set: Address peter@'s comments Created 3 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2017 The Chromium Authors. All rights reserved. 1 // Copyright 2017 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "chrome/browser/notifications/notification_platform_bridge_linux.h" 5 #include "chrome/browser/notifications/notification_platform_bridge_linux.h"
6 6
7 #include <algorithm>
8
7 #include "base/memory/ptr_util.h" 9 #include "base/memory/ptr_util.h"
8 #include "base/strings/utf_string_conversions.h" 10 #include "base/strings/utf_string_conversions.h"
11 #include "chrome/browser/browser_process.h"
9 #include "chrome/browser/notifications/notification.h" 12 #include "chrome/browser/notifications/notification.h"
10 13
11 namespace { 14 namespace {
12 15
13 const char kFreedesktopNotificationsName[] = "org.freedesktop.Notifications"; 16 const char kFreedesktopNotificationsName[] = "org.freedesktop.Notifications";
14 const char kFreedesktopNotificationsPath[] = "/org/freedesktop/Notifications"; 17 const char kFreedesktopNotificationsPath[] = "/org/freedesktop/Notifications";
15 18
19 // Callback used by GLib when the "Notify" message completes for the
20 // first time.
21 void NotifyCompleteReceiver(GObject* source_object,
22 GAsyncResult* result,
23 gpointer user_data) {
24 GDBusProxy* proxy = G_DBUS_PROXY(source_object);
25 GVariant* value = g_dbus_proxy_call_finish(proxy, result, nullptr);
26 if (!value) {
27 // The message might have been cancelled, in which case
28 // |user_data| points to a destroyed NotificationData.
29 return;
30 }
31 auto* platform_bridge_linux = static_cast<NotificationPlatformBridgeLinux*>(
32 g_browser_process->notification_platform_bridge());
33 platform_bridge_linux->NotifyCompleteInternal(user_data, value);
34 }
35
16 } // namespace 36 } // namespace
17 37
18 // static 38 // static
19 NotificationPlatformBridge* NotificationPlatformBridge::Create() { 39 NotificationPlatformBridge* NotificationPlatformBridge::Create() {
20 GDBusProxy* notification_proxy = g_dbus_proxy_new_for_bus_sync( 40 GDBusProxy* notification_proxy = g_dbus_proxy_new_for_bus_sync(
21 G_BUS_TYPE_SESSION, 41 G_BUS_TYPE_SESSION,
22 static_cast<GDBusProxyFlags>(G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES | 42 static_cast<GDBusProxyFlags>(G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES |
23 G_DBUS_PROXY_FLAGS_DO_NOT_CONNECT_SIGNALS | 43 G_DBUS_PROXY_FLAGS_DO_NOT_CONNECT_SIGNALS |
24 G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START), 44 G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START),
25 nullptr, kFreedesktopNotificationsName, kFreedesktopNotificationsPath, 45 nullptr, kFreedesktopNotificationsName, kFreedesktopNotificationsPath,
26 kFreedesktopNotificationsName, nullptr, nullptr); 46 kFreedesktopNotificationsName, nullptr, nullptr);
27 if (!notification_proxy) 47 if (!notification_proxy)
28 return nullptr; 48 return nullptr;
29 return new NotificationPlatformBridgeLinux(notification_proxy); 49 return new NotificationPlatformBridgeLinux(notification_proxy);
30 } 50 }
31 51
52 struct NotificationPlatformBridgeLinux::NotificationData {
53 NotificationData(const std::string& notification_id,
54 const std::string& profile_id,
55 bool is_incognito)
56 : notification_id(notification_id),
57 profile_id(profile_id),
58 is_incognito(is_incognito) {}
59
60 ~NotificationData() {
61 if (cancellable)
62 g_cancellable_cancel(cancellable);
63 }
64
65 // The ID used by the notification server. Will be 0 until the
66 // first "Notify" message completes.
67 uint32_t dbus_id = 0;
68
69 // Same parameters used by NotificationPlatformBridge::Display().
70 const std::string notification_id;
71 const std::string profile_id;
72 const bool is_incognito;
73
74 // Used to cancel the initial "Notify" message so we don't call
75 // NotificationPlatformBridgeLinux::NotifyCompleteInternal() with a
76 // destroyed Notification.
77 ScopedGObject<GCancellable> cancellable;
78
79 // If not null, the data to update the notification with once
80 // |dbus_id| becomes available.
81 std::unique_ptr<Notification> update_data;
82 NotificationCommon::Type update_type = NotificationCommon::TYPE_MAX;
83
84 // If true, indicates the notification should be closed once
85 // |dbus_id| becomes available.
86 bool should_close = false;
87 };
88
32 NotificationPlatformBridgeLinux::NotificationPlatformBridgeLinux( 89 NotificationPlatformBridgeLinux::NotificationPlatformBridgeLinux(
33 GDBusProxy* notification_proxy) 90 GDBusProxy* notification_proxy)
34 : notification_proxy_(notification_proxy) {} 91 : notification_proxy_(notification_proxy) {}
35 92
36 NotificationPlatformBridgeLinux::~NotificationPlatformBridgeLinux() { 93 NotificationPlatformBridgeLinux::~NotificationPlatformBridgeLinux() {}
37 g_object_unref(notification_proxy_);
38 }
39 94
40 void NotificationPlatformBridgeLinux::Display( 95 void NotificationPlatformBridgeLinux::Display(
41 NotificationCommon::Type notification_type, 96 NotificationCommon::Type notification_type,
42 const std::string& notification_id, 97 const std::string& notification_id,
43 const std::string& profile_id, 98 const std::string& profile_id,
44 bool is_incognito, 99 bool is_incognito,
45 const Notification& notification) { 100 const Notification& notification) {
46 // TODO(thomasanderson): Add a complete implementation. 101 NotificationData* data = FindNotificationData(notification_id, profile_id);
47 g_dbus_proxy_call( 102 if (data) {
48 notification_proxy_, "Notify", 103 // Update an existing notification.
49 g_variant_new("(susssasa{sv}i)", "", 0, "", 104 if (data->dbus_id) {
50 base::UTF16ToUTF8(notification.title()).c_str(), 105 NotifyNow(data->dbus_id, notification_type, notification, nullptr,
51 base::UTF16ToUTF8(notification.message()).c_str(), nullptr, 106 nullptr, nullptr);
52 nullptr, -1), 107 } else {
53 G_DBUS_CALL_FLAGS_NONE, -1, nullptr, nullptr, nullptr); 108 data->update_type = notification_type;
109 data->update_data = base::MakeUnique<Notification>(notification);
110 }
111 } else {
112 // Send the notification for the first time.
113 data = new NotificationData(notification_id, profile_id, is_incognito);
114 data->cancellable.reset(g_cancellable_new());
115 notifications_.emplace(data, base::WrapUnique(data));
116 NotifyNow(0, notification_type, notification, data->cancellable,
117 NotifyCompleteReceiver, data);
118 }
54 } 119 }
55 120
56 void NotificationPlatformBridgeLinux::Close( 121 void NotificationPlatformBridgeLinux::Close(
57 const std::string& profile_id, 122 const std::string& profile_id,
58 const std::string& notification_id) { 123 const std::string& notification_id) {
59 NOTIMPLEMENTED(); 124 NotificationData* data = FindNotificationData(notification_id, profile_id);
125 if (!data)
126 return;
127 if (data->dbus_id) {
128 CloseNow(data->dbus_id);
129 notifications_.erase(data);
130 } else {
131 data->should_close = true;
132 }
60 } 133 }
61 134
62 void NotificationPlatformBridgeLinux::GetDisplayed( 135 void NotificationPlatformBridgeLinux::GetDisplayed(
63 const std::string& profile_id, 136 const std::string& profile_id,
64 bool incognito, 137 bool incognito,
65 const DisplayedNotificationsCallback& callback) const { 138 const DisplayedNotificationsCallback& callback) const {
66 callback.Run(base::MakeUnique<std::set<std::string>>(), false); 139 callback.Run(base::MakeUnique<std::set<std::string>>(), false);
67 } 140 }
141
142 void NotificationPlatformBridgeLinux::NotifyCompleteInternal(gpointer user_data,
143 GVariant* value) {
144 NotificationData* data = reinterpret_cast<NotificationData*>(user_data);
145 if (notifications_.find(data) == notifications_.end())
Lei Zhang 2017/04/06 19:59:33 !base::ContainsKey(notifications_, data) ?
Tom (Use chromium acct) 2017/04/06 22:28:56 Done.
146 return;
147 data->cancellable.reset();
148 if (value && g_variant_is_of_type(value, G_VARIANT_TYPE("(u)")))
149 g_variant_get(value, "(u)", &data->dbus_id);
150
151 if (!data->dbus_id) {
152 // There was some sort of error with creating the notification.
153 notifications_.erase(data);
154 } else if (data->should_close) {
155 CloseNow(data->dbus_id);
156 notifications_.erase(data);
157 } else if (data->update_data) {
158 NotifyNow(data->dbus_id, data->update_type, *data->update_data, nullptr,
159 nullptr, nullptr);
160 data->update_data.reset();
161 }
162 }
163
164 void NotificationPlatformBridgeLinux::NotifyNow(
165 uint32_t dbus_id,
166 NotificationCommon::Type notification_type,
167 const Notification& notification,
168 GCancellable* cancellable,
169 GAsyncReadyCallback callback,
170 gpointer user_data) {
171 // TODO(thomasanderson): Add a complete implementation.
172 const std::string title = base::UTF16ToUTF8(notification.title());
173 const std::string message = base::UTF16ToUTF8(notification.message());
174 GVariant* parameters =
175 g_variant_new("(susssasa{sv}i)", "", dbus_id, "", title.c_str(),
176 message.c_str(), nullptr, nullptr, -1);
177 g_dbus_proxy_call(notification_proxy_, "Notify", parameters,
178 G_DBUS_CALL_FLAGS_NONE, -1, cancellable, callback,
179 user_data);
180 }
181
182 void NotificationPlatformBridgeLinux::CloseNow(uint32_t dbus_id) {
183 g_dbus_proxy_call(notification_proxy_, "CloseNotification",
184 g_variant_new("(u)", dbus_id), G_DBUS_CALL_FLAGS_NONE, -1,
185 nullptr, nullptr, nullptr);
186 }
187
188 NotificationPlatformBridgeLinux::NotificationData*
189 NotificationPlatformBridgeLinux::FindNotificationData(
190 const std::string& notification_id,
191 const std::string& profile_id) {
192 for (const auto& pair : notifications_) {
193 NotificationData* data = pair.first;
194 if (data->notification_id == notification_id &&
195 data->profile_id == profile_id) {
196 return data;
197 }
198 }
199
200 return nullptr;
201 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698