| Index: content/renderer/notification_provider.cc
|
| diff --git a/content/renderer/notification_provider.cc b/content/renderer/notification_provider.cc
|
| index c4d869c2b1caea82e31f46e5997d5e64675c4808..5873edca3ac001aa29f4a72856cb4857a0919339 100644
|
| --- a/content/renderer/notification_provider.cc
|
| +++ b/content/renderer/notification_provider.cc
|
| @@ -5,13 +5,18 @@
|
| #include "content/renderer/notification_provider.h"
|
|
|
| #include "base/strings/string_util.h"
|
| +#include "content/child/image_decoder.h"
|
| #include "content/common/desktop_notification_messages.h"
|
| #include "content/common/frame_messages.h"
|
| #include "content/renderer/render_frame_impl.h"
|
| +#include "third_party/WebKit/public/platform/Platform.h"
|
| #include "third_party/WebKit/public/platform/WebURL.h"
|
| +#include "third_party/WebKit/public/platform/WebURLLoader.h"
|
| +#include "third_party/WebKit/public/platform/WebURLLoaderClient.h"
|
| #include "third_party/WebKit/public/web/WebDocument.h"
|
| #include "third_party/WebKit/public/web/WebLocalFrame.h"
|
| #include "third_party/WebKit/public/web/WebUserGestureIndicator.h"
|
| +#include "third_party/skia/include/core/SkBitmap.h"
|
|
|
| using blink::WebDocument;
|
| using blink::WebNotification;
|
| @@ -19,38 +24,178 @@ using blink::WebNotificationPresenter;
|
| using blink::WebSecurityOrigin;
|
| using blink::WebString;
|
| using blink::WebURL;
|
| +using blink::WebURLError;
|
| +using blink::WebURLLoader;
|
| +using blink::WebURLRequest;
|
| using blink::WebUserGestureIndicator;
|
|
|
| namespace content {
|
|
|
| -NotificationProvider::NotificationProvider(RenderFrame* render_frame)
|
| - : RenderFrameObserver(render_frame) {
|
| +// NotificationProvider::IconDownloader ----------------------------------------
|
| +
|
| +// Downloads the icon associated with a notification and decodes the received
|
| +// image. This must be completed before notifications are shown to the user.
|
| +class NotificationProvider::IconDownloader : public blink::WebURLLoaderClient {
|
| + typedef base::Callback<void(const WebNotification&,
|
| + const SkBitmap&)> DownloadCompletedCallback;
|
| +
|
| + public:
|
| + IconDownloader(const WebNotification& notification,
|
| + const DownloadCompletedCallback& callback);
|
| +
|
| + virtual ~IconDownloader();
|
| +
|
| + // Downloads |image_url| and invokes |callback| with the decoded SkBitmap when
|
| + // the download has succeeded, or invokes |callback| with an empty SkBitmap
|
| + // in case the download has failed.
|
| + void Start();
|
| +
|
| + // Cancels the current image download. The callback will not be invoked.
|
| + void Cancel();
|
| +
|
| + // blink::WebURLLoaderClient implementation.
|
| + virtual void didReceiveData(WebURLLoader* loader,
|
| + const char* data,
|
| + int data_length,
|
| + int encoded_data_length) OVERRIDE;
|
| + virtual void didFinishLoading(WebURLLoader* loader,
|
| + double finish_time,
|
| + int64_t total_encoded_data_length) OVERRIDE;
|
| + virtual void didFail(WebURLLoader* loader, const WebURLError& error) OVERRIDE;
|
| +
|
| + const WebNotification& notification() const {
|
| + return notification_;
|
| + }
|
| +
|
| + private:
|
| + WebNotification notification_;
|
| + DownloadCompletedCallback callback_;
|
| +
|
| + scoped_ptr<WebURLLoader> loader_;
|
| + bool completed_;
|
| +
|
| + std::string buffer_;
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(IconDownloader);
|
| +};
|
| +
|
| +NotificationProvider::IconDownloader::IconDownloader(
|
| + const WebNotification& notification,
|
| + const DownloadCompletedCallback& callback)
|
| + : notification_(notification),
|
| + callback_(callback),
|
| + completed_(false) {}
|
| +
|
| +NotificationProvider::IconDownloader::~IconDownloader() {}
|
| +
|
| +void NotificationProvider::IconDownloader::Start() {
|
| + DCHECK(!loader_);
|
| +
|
| + WebURLRequest request(notification_.iconURL());
|
| +
|
| + loader_.reset(blink::Platform::current()->createURLLoader());
|
| + loader_->loadAsynchronously(request, this);
|
| }
|
|
|
| -NotificationProvider::~NotificationProvider() {
|
| +void NotificationProvider::IconDownloader::Cancel() {
|
| + DCHECK(loader_);
|
| +
|
| + completed_ = true;
|
| + loader_->cancel();
|
| }
|
|
|
| +void NotificationProvider::IconDownloader::didReceiveData(
|
| + WebURLLoader* loader,
|
| + const char* data,
|
| + int data_length,
|
| + int encoded_data_length) {
|
| + DCHECK(!completed_);
|
| + DCHECK(data_length > 0);
|
| +
|
| + buffer_.append(data, data_length);
|
| +}
|
| +
|
| +void NotificationProvider::IconDownloader::didFinishLoading(
|
| + WebURLLoader* loader,
|
| + double finish_time,
|
| + int64_t total_encoded_data_length) {
|
| + DCHECK(!completed_);
|
| +
|
| + ImageDecoder decoder;
|
| + SkBitmap icon = decoder.Decode(
|
| + reinterpret_cast<const unsigned char*>(buffer_.data()), buffer_.size());
|
| +
|
| + callback_.Run(notification_, icon);
|
| + completed_ = true;
|
| +}
|
| +
|
| +void NotificationProvider::IconDownloader::didFail(
|
| + WebURLLoader* loader, const WebURLError& error) {
|
| + if (completed_)
|
| + return;
|
| +
|
| + callback_.Run(notification_, SkBitmap());
|
| + completed_ = true;
|
| +}
|
| +
|
| +// NotificationProvider --------------------------------------------------------
|
| +
|
| +NotificationProvider::NotificationProvider(RenderFrame* render_frame)
|
| + : RenderFrameObserver(render_frame) {}
|
| +
|
| +NotificationProvider::~NotificationProvider() {}
|
| +
|
| bool NotificationProvider::show(const WebNotification& notification) {
|
| + if (notification.iconURL().isEmpty()) {
|
| + DisplayNotification(notification, SkBitmap());
|
| + return true;
|
| + }
|
| +
|
| + scoped_ptr<IconDownloader> downloader(
|
| + new IconDownloader(notification,
|
| + base::Bind(&NotificationProvider::DisplayNotification,
|
| + base::Unretained(this))));
|
| +
|
| + downloader->Start();
|
| +
|
| + pending_notifications_.push_back(downloader.release());
|
| + return true;
|
| +}
|
| +
|
| +void NotificationProvider::DisplayNotification(
|
| + const WebNotification& notification, const SkBitmap& icon) {
|
| WebDocument document = render_frame()->GetWebFrame()->document();
|
| int notification_id = manager_.RegisterNotification(notification);
|
|
|
| + // TODO: Remove the downloaded notification from |pending_notifications_|
|
| + // if it was triggered through there.
|
| +
|
| ShowDesktopNotificationHostMsgParams params;
|
| params.origin = GURL(document.securityOrigin().toString());
|
| - params.icon_url = notification.iconURL();
|
| + params.icon = icon;
|
| params.title = notification.title();
|
| params.body = notification.body();
|
| params.direction = notification.direction();
|
| params.replace_id = notification.replaceId();
|
| - return Send(new DesktopNotificationHostMsg_Show(
|
| - routing_id(), notification_id, params));
|
| +
|
| + Send(new DesktopNotificationHostMsg_Show(routing_id(),
|
| + notification_id,
|
| + params));
|
| +}
|
| +
|
| +void NotificationProvider::CancelIconDownload(
|
| + const WebNotification& notification) {
|
| }
|
|
|
| void NotificationProvider::cancel(const WebNotification& notification) {
|
| int id;
|
| bool id_found = manager_.GetId(notification, id);
|
| - // Won't be found if the notification has already been closed by the user.
|
| + // Won't be found if the notification has already been closed by the user,
|
| + // or if the notification's icon is still being requested.
|
| if (id_found)
|
| Send(new DesktopNotificationHostMsg_Cancel(routing_id(), id));
|
| + else
|
| + CancelIconDownload(notification);
|
| }
|
|
|
| void NotificationProvider::objectDestroyed(
|
| @@ -60,6 +205,8 @@ void NotificationProvider::objectDestroyed(
|
| // Won't be found if the notification has already been closed by the user.
|
| if (id_found)
|
| manager_.UnregisterNotification(id);
|
| + else
|
| + CancelIconDownload(notification);
|
| }
|
|
|
| WebNotificationPresenter::Permission NotificationProvider::checkPermission(
|
|
|