| // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/browser/notifications/desktop_notification_service.h" |
| |
| #include "app/l10n_util.h" |
| #include "app/resource_bundle.h" |
| #include "base/thread.h" |
| #include "base/utf_string_conversions.h" |
| #include "chrome/browser/child_process_host.h" |
| #include "chrome/browser/chrome_thread.h" |
| #include "chrome/browser/extensions/extensions_service.h" |
| #include "chrome/browser/notifications/notification.h" |
| #include "chrome/browser/notifications/notification_object_proxy.h" |
| #include "chrome/browser/notifications/notification_ui_manager.h" |
| #include "chrome/browser/notifications/notifications_prefs_cache.h" |
| #include "chrome/browser/pref_service.h" |
| #include "chrome/browser/profile.h" |
| #include "chrome/browser/renderer_host/render_process_host.h" |
| #include "chrome/browser/renderer_host/render_view_host.h" |
| #include "chrome/browser/renderer_host/site_instance.h" |
| #include "chrome/browser/scoped_pref_update.h" |
| #include "chrome/browser/tab_contents/infobar_delegate.h" |
| #include "chrome/browser/tab_contents/tab_contents.h" |
| #include "chrome/browser/worker_host/worker_process_host.h" |
| #include "chrome/common/notification_service.h" |
| #include "chrome/common/pref_names.h" |
| #include "chrome/common/render_messages.h" |
| #include "chrome/common/url_constants.h" |
| #include "grit/browser_resources.h" |
| #include "grit/chromium_strings.h" |
| #include "grit/generated_resources.h" |
| #include "grit/theme_resources.h" |
| #include "net/base/escape.h" |
| #include "third_party/WebKit/WebKit/chromium/public/WebNotificationPresenter.h" |
| |
| using WebKit::WebNotificationPresenter; |
| |
| // static |
| string16 DesktopNotificationService::CreateDataUrl( |
| const GURL& icon_url, const string16& title, const string16& body) { |
| int resource; |
| string16 line_name; |
| string16 line; |
| std::vector<std::string> subst; |
| if (icon_url.is_valid()) { |
| resource = IDR_NOTIFICATION_ICON_HTML; |
| subst.push_back(icon_url.spec()); |
| subst.push_back(EscapeForHTML(UTF16ToUTF8(title))); |
| subst.push_back(EscapeForHTML(UTF16ToUTF8(body))); |
| } else if (title.empty() || body.empty()) { |
| resource = IDR_NOTIFICATION_1LINE_HTML; |
| line = title.empty() ? body : title; |
| // Strings are div names in the template file. |
| line_name = title.empty() ? ASCIIToUTF16("description") |
| : ASCIIToUTF16("title"); |
| subst.push_back(EscapeForHTML(UTF16ToUTF8(line_name))); |
| subst.push_back(EscapeForHTML(UTF16ToUTF8(line))); |
| } else { |
| resource = IDR_NOTIFICATION_2LINE_HTML; |
| subst.push_back(EscapeForHTML(UTF16ToUTF8(title))); |
| subst.push_back(EscapeForHTML(UTF16ToUTF8(body))); |
| } |
| |
| const base::StringPiece template_html( |
| ResourceBundle::GetSharedInstance().GetRawDataResource( |
| resource)); |
| |
| if (template_html.empty()) { |
| NOTREACHED() << "unable to load template. ID: " << resource; |
| return string16(); |
| } |
| |
| std::string data = ReplaceStringPlaceholders(template_html, subst, NULL); |
| return UTF8ToUTF16("data:text/html;charset=utf-8," + |
| EscapeQueryParamValue(data, false)); |
| } |
| |
| // A task object which calls the renderer to inform the web page that the |
| // permission request has completed. |
| class NotificationPermissionCallbackTask : public Task { |
| public: |
| NotificationPermissionCallbackTask(int process_id, int route_id, |
| int request_id) |
| : process_id_(process_id), |
| route_id_(route_id), |
| request_id_(request_id) { |
| } |
| |
| virtual void Run() { |
| DCHECK(ChromeThread::CurrentlyOn(ChromeThread::IO)); |
| RenderViewHost* host = RenderViewHost::FromID(process_id_, route_id_); |
| if (host) |
| host->Send(new ViewMsg_PermissionRequestDone(route_id_, request_id_)); |
| } |
| |
| private: |
| int process_id_; |
| int route_id_; |
| int request_id_; |
| }; |
| |
| // The delegate for the infobar shown when an origin requests notification |
| // permissions. |
| class NotificationPermissionInfoBarDelegate : public ConfirmInfoBarDelegate { |
| public: |
| NotificationPermissionInfoBarDelegate(TabContents* contents, |
| const GURL& origin, |
| const std::wstring& display_name, |
| int process_id, |
| int route_id, |
| int callback_context) |
| : ConfirmInfoBarDelegate(contents), |
| origin_(origin), |
| display_name_(display_name), |
| profile_(contents->profile()), |
| process_id_(process_id), |
| route_id_(route_id), |
| callback_context_(callback_context), |
| action_taken_(false) { |
| } |
| |
| // Overridden from ConfirmInfoBarDelegate: |
| virtual void InfoBarClosed() { |
| if (!action_taken_) |
| UMA_HISTOGRAM_COUNTS("NotificationPermissionRequest.Ignored", 1); |
| |
| ChromeThread::PostTask( |
| ChromeThread::IO, FROM_HERE, |
| new NotificationPermissionCallbackTask( |
| process_id_, route_id_, callback_context_)); |
| |
| delete this; |
| } |
| |
| virtual std::wstring GetMessageText() const { |
| return l10n_util::GetStringF(IDS_NOTIFICATION_PERMISSIONS, display_name_); |
| } |
| |
| virtual SkBitmap* GetIcon() const { |
| return ResourceBundle::GetSharedInstance().GetBitmapNamed( |
| IDR_PRODUCT_ICON_32); |
| } |
| |
| virtual int GetButtons() const { |
| return BUTTON_OK | BUTTON_CANCEL | BUTTON_OK_DEFAULT; |
| } |
| |
| virtual std::wstring GetButtonLabel(InfoBarButton button) const { |
| return button == BUTTON_OK ? |
| l10n_util::GetString(IDS_NOTIFICATION_PERMISSION_YES) : |
| l10n_util::GetString(IDS_NOTIFICATION_PERMISSION_NO); |
| } |
| |
| virtual bool Accept() { |
| UMA_HISTOGRAM_COUNTS("NotificationPermissionRequest.Allowed", 1); |
| profile_->GetDesktopNotificationService()->GrantPermission(origin_); |
| action_taken_ = true; |
| return true; |
| } |
| |
| virtual bool Cancel() { |
| UMA_HISTOGRAM_COUNTS("NotificationPermissionRequest.Denied", 1); |
| profile_->GetDesktopNotificationService()->DenyPermission(origin_); |
| action_taken_ = true; |
| return true; |
| } |
| |
| private: |
| // The origin we are asking for permissions on. |
| GURL origin_; |
| |
| // The display name for the origin to be displayed. Will be different from |
| // origin_ for extensions. |
| std::wstring display_name_; |
| |
| // The Profile that we restore sessions from. |
| Profile* profile_; |
| |
| // The callback information that tells us how to respond to javascript via |
| // the correct RenderView. |
| int process_id_; |
| int route_id_; |
| int callback_context_; |
| |
| // Whether the user clicked one of the buttons. |
| bool action_taken_; |
| |
| DISALLOW_COPY_AND_ASSIGN(NotificationPermissionInfoBarDelegate); |
| }; |
| |
| DesktopNotificationService::DesktopNotificationService(Profile* profile, |
| NotificationUIManager* ui_manager) |
| : profile_(profile), |
| ui_manager_(ui_manager) { |
| InitPrefs(); |
| StartObserving(); |
| } |
| |
| DesktopNotificationService::~DesktopNotificationService() { |
| StopObserving(); |
| } |
| |
| void DesktopNotificationService::RegisterUserPrefs(PrefService* user_prefs) { |
| if (!user_prefs->FindPreference(prefs::kDesktopNotificationAllowedOrigins)) |
| user_prefs->RegisterListPref(prefs::kDesktopNotificationAllowedOrigins); |
| if (!user_prefs->FindPreference(prefs::kDesktopNotificationDeniedOrigins)) |
| user_prefs->RegisterListPref(prefs::kDesktopNotificationDeniedOrigins); |
| } |
| |
| // Initialize the cache with the allowed and denied origins, or |
| // create the preferences if they don't exist yet. |
| void DesktopNotificationService::InitPrefs() { |
| PrefService* prefs = profile_->GetPrefs(); |
| std::vector<GURL> allowed_origins; |
| std::vector<GURL> denied_origins; |
| |
| if (!profile_->IsOffTheRecord()) { |
| const ListValue* allowed_sites = |
| prefs->GetList(prefs::kDesktopNotificationAllowedOrigins); |
| if (allowed_sites) |
| NotificationsPrefsCache::ListValueToGurlVector(*allowed_sites, |
| &allowed_origins); |
| |
| const ListValue* denied_sites = |
| prefs->GetList(prefs::kDesktopNotificationDeniedOrigins); |
| if (denied_sites) |
| NotificationsPrefsCache::ListValueToGurlVector(*denied_sites, |
| &denied_origins); |
| } |
| |
| prefs_cache_ = new NotificationsPrefsCache(); |
| prefs_cache_->SetCacheAllowedOrigins(allowed_origins); |
| prefs_cache_->SetCacheDeniedOrigins(denied_origins); |
| prefs_cache_->set_is_initialized(true); |
| } |
| |
| void DesktopNotificationService::StartObserving() { |
| if (!profile_->IsOffTheRecord()) { |
| PrefService* prefs = profile_->GetPrefs(); |
| prefs->AddPrefObserver(prefs::kDesktopNotificationAllowedOrigins, this); |
| prefs->AddPrefObserver(prefs::kDesktopNotificationDeniedOrigins, this); |
| } |
| } |
| |
| void DesktopNotificationService::StopObserving() { |
| if (!profile_->IsOffTheRecord()) { |
| PrefService* prefs = profile_->GetPrefs(); |
| prefs->RemovePrefObserver(prefs::kDesktopNotificationAllowedOrigins, this); |
| prefs->RemovePrefObserver(prefs::kDesktopNotificationDeniedOrigins, this); |
| } |
| } |
| |
| void DesktopNotificationService::GrantPermission(const GURL& origin) { |
| DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); |
| PersistPermissionChange(origin, true); |
| |
| // Schedule a cache update on the IO thread. |
| ChromeThread::PostTask( |
| ChromeThread::IO, FROM_HERE, |
| NewRunnableMethod( |
| prefs_cache_.get(), &NotificationsPrefsCache::CacheAllowedOrigin, |
| origin)); |
| } |
| |
| void DesktopNotificationService::DenyPermission(const GURL& origin) { |
| DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); |
| PersistPermissionChange(origin, false); |
| |
| // Schedule a cache update on the IO thread. |
| ChromeThread::PostTask( |
| ChromeThread::IO, FROM_HERE, |
| NewRunnableMethod( |
| prefs_cache_.get(), &NotificationsPrefsCache::CacheDeniedOrigin, |
| origin)); |
| } |
| |
| void DesktopNotificationService::Observe(NotificationType type, |
| const NotificationSource& source, |
| const NotificationDetails& details) { |
| DCHECK(NotificationType::PREF_CHANGED == type); |
| PrefService* prefs = profile_->GetPrefs(); |
| std::wstring* name = Details<std::wstring>(details).ptr(); |
| |
| if (0 == name->compare(prefs::kDesktopNotificationAllowedOrigins)) { |
| const ListValue* allowed_sites = |
| prefs->GetList(prefs::kDesktopNotificationAllowedOrigins); |
| std::vector<GURL> allowed_origins; |
| if (allowed_sites) { |
| NotificationsPrefsCache::ListValueToGurlVector(*allowed_sites, |
| &allowed_origins); |
| } |
| // Schedule a cache update on the IO thread. |
| ChromeThread::PostTask( |
| ChromeThread::IO, FROM_HERE, |
| NewRunnableMethod( |
| prefs_cache_.get(), |
| &NotificationsPrefsCache::SetCacheAllowedOrigins, |
| allowed_origins)); |
| } else if (0 == name->compare(prefs::kDesktopNotificationDeniedOrigins)) { |
| const ListValue* denied_sites = |
| prefs->GetList(prefs::kDesktopNotificationDeniedOrigins); |
| std::vector<GURL> denied_origins; |
| if (denied_sites) { |
| NotificationsPrefsCache::ListValueToGurlVector(*denied_sites, |
| &denied_origins); |
| } |
| // Schedule a cache update on the IO thread. |
| ChromeThread::PostTask( |
| ChromeThread::IO, FROM_HERE, |
| NewRunnableMethod( |
| prefs_cache_.get(), |
| &NotificationsPrefsCache::SetCacheDeniedOrigins, |
| denied_origins)); |
| } |
| } |
| |
| void DesktopNotificationService::PersistPermissionChange( |
| const GURL& origin, bool is_allowed) { |
| // Don't persist changes when off the record. |
| if (profile_->IsOffTheRecord()) |
| return; |
| |
| PrefService* prefs = profile_->GetPrefs(); |
| |
| StopObserving(); |
| |
| bool allowed_changed = false; |
| bool denied_changed = false; |
| |
| ListValue* allowed_sites = |
| prefs->GetMutableList(prefs::kDesktopNotificationAllowedOrigins); |
| ListValue* denied_sites = |
| prefs->GetMutableList(prefs::kDesktopNotificationDeniedOrigins); |
| { |
| // value is passed to the preferences list, or deleted. |
| StringValue* value = new StringValue(origin.spec()); |
| |
| // Remove from one list and add to the other. |
| if (is_allowed) { |
| // Remove from the denied list. |
| if (denied_sites->Remove(*value) != -1) |
| denied_changed = true; |
| |
| // Add to the allowed list. |
| if (allowed_sites->AppendIfNotPresent(value)) |
| allowed_changed = true; |
| else |
| delete value; |
| } else { |
| // Remove from the allowed list. |
| if (allowed_sites->Remove(*value) != -1) |
| allowed_changed = true; |
| |
| // Add to the denied list. |
| if (denied_sites->AppendIfNotPresent(value)) |
| denied_changed = true; |
| else |
| delete value; |
| } |
| } |
| |
| // Persist the pref if anthing changed, but only send updates for the |
| // list that changed. |
| if (allowed_changed || denied_changed) { |
| if (allowed_changed) { |
| ScopedPrefUpdate updateAllowed( |
| prefs, prefs::kDesktopNotificationAllowedOrigins); |
| } |
| if (denied_changed) { |
| ScopedPrefUpdate updateDenied( |
| prefs, prefs::kDesktopNotificationDeniedOrigins); |
| } |
| prefs->ScheduleSavePersistentPrefs(); |
| } |
| StartObserving(); |
| } |
| |
| void DesktopNotificationService::RequestPermission( |
| const GURL& origin, int process_id, int route_id, int callback_context, |
| TabContents* tab) { |
| DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); |
| if (!tab) |
| return; |
| // Show an info bar requesting permission. |
| std::wstring display_name = DisplayNameForOrigin(origin); |
| |
| tab->AddInfoBar(new NotificationPermissionInfoBarDelegate( |
| tab, origin, display_name, process_id, route_id, callback_context)); |
| } |
| |
| void DesktopNotificationService::ShowNotification( |
| const Notification& notification) { |
| ui_manager_->Add(notification, profile_); |
| } |
| |
| bool DesktopNotificationService::CancelDesktopNotification( |
| int process_id, int route_id, int notification_id) { |
| scoped_refptr<NotificationObjectProxy> proxy( |
| new NotificationObjectProxy(process_id, route_id, notification_id, |
| false)); |
| Notification notif(GURL(), GURL(), L"", proxy); |
| return ui_manager_->Cancel(notif); |
| } |
| |
| |
| bool DesktopNotificationService::ShowDesktopNotification( |
| const GURL& origin, const GURL& url, int process_id, int route_id, |
| DesktopNotificationSource source, int notification_id) { |
| DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); |
| NotificationObjectProxy* proxy = |
| new NotificationObjectProxy(process_id, route_id, |
| notification_id, |
| source == WorkerNotification); |
| Notification notif(origin, url, DisplayNameForOrigin(origin), proxy); |
| ShowNotification(notif); |
| return true; |
| } |
| |
| bool DesktopNotificationService::ShowDesktopNotificationText( |
| const GURL& origin, const GURL& icon, const string16& title, |
| const string16& text, int process_id, int route_id, |
| DesktopNotificationSource source, int notification_id) { |
| DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI)); |
| NotificationObjectProxy* proxy = |
| new NotificationObjectProxy(process_id, route_id, |
| notification_id, |
| source == WorkerNotification); |
| // "upconvert" the string parameters to a data: URL. |
| string16 data_url = CreateDataUrl(icon, title, text); |
| Notification notif( |
| origin, GURL(data_url), DisplayNameForOrigin(origin), proxy); |
| ShowNotification(notif); |
| return true; |
| } |
| |
| std::wstring DesktopNotificationService::DisplayNameForOrigin( |
| const GURL& origin) { |
| // If the source is an extension, lookup the display name. |
| if (origin.SchemeIs(chrome::kExtensionScheme)) { |
| ExtensionsService* ext_service = profile_->GetExtensionsService(); |
| if (ext_service) { |
| Extension* extension = ext_service->GetExtensionByURL(origin); |
| if (extension) |
| return UTF8ToWide(extension->name()); |
| } |
| } |
| return UTF8ToWide(origin.host()); |
| } |