| Index: chrome/renderer/favicon_helper.cc
|
| diff --git a/chrome/renderer/favicon_helper.cc b/chrome/renderer/favicon_helper.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..8df4839af873e75cd68422073a3a0bc33416c9e2
|
| --- /dev/null
|
| +++ b/chrome/renderer/favicon_helper.cc
|
| @@ -0,0 +1,172 @@
|
| +// Copyright (c) 2012 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/renderer/favicon_helper.h"
|
| +
|
| +#include "base/bind.h"
|
| +#include "base/message_loop.h"
|
| +#include "chrome/common/chrome_constants.h"
|
| +#include "chrome/common/favicon_url.h"
|
| +#include "chrome/common/icon_messages.h"
|
| +#include "content/public/renderer/render_view.h"
|
| +#include "net/base/data_url.h"
|
| +#include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h"
|
| +#include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h"
|
| +#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURLRequest.h"
|
| +#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebVector.h"
|
| +#include "ui/gfx/favicon_size.h"
|
| +#include "ui/gfx/size.h"
|
| +#include "ui/gfx/skbitmap_operations.h"
|
| +#include "webkit/glue/image_decoder.h"
|
| +#include "webkit/glue/multi_resolution_image_resource_fetcher.h"
|
| +#include "webkit/glue/webkit_glue.h"
|
| +
|
| +using WebKit::WebFrame;
|
| +using WebKit::WebIconURL;
|
| +using WebKit::WebVector;
|
| +using WebKit::WebURL;
|
| +using WebKit::WebURLRequest;
|
| +using webkit_glue::MultiResolutionImageResourceFetcher;
|
| +
|
| +static FaviconURL::IconType ToFaviconType(WebIconURL::Type type) {
|
| + switch (type) {
|
| + case WebIconURL::TypeFavicon:
|
| + return FaviconURL::FAVICON;
|
| + case WebIconURL::TypeTouch:
|
| + return FaviconURL::TOUCH_ICON;
|
| + case WebIconURL::TypeTouchPrecomposed:
|
| + return FaviconURL::TOUCH_PRECOMPOSED_ICON;
|
| + case WebIconURL::TypeInvalid:
|
| + return FaviconURL::INVALID_ICON;
|
| + }
|
| + return FaviconURL::INVALID_ICON;
|
| +}
|
| +
|
| +FaviconHelper::FaviconHelper(content::RenderView* render_view)
|
| + : content::RenderViewObserver(render_view) {
|
| +}
|
| +
|
| +FaviconHelper::~FaviconHelper() {
|
| +}
|
| +
|
| +void FaviconHelper::OnDownloadFavicon(int id,
|
| + const GURL& image_url,
|
| + int image_size) {
|
| + std::vector<SkBitmap> result_images;
|
| + if (image_url.SchemeIs("data")) {
|
| + SkBitmap data_image = ImageFromDataUrl(image_url);
|
| + if (!data_image.empty())
|
| + result_images.push_back(data_image);
|
| + } else {
|
| + if (DownloadFavicon(id, image_url, image_size)) {
|
| + // Will complete asynchronously via FaviconHelper::DidDownloadFavicon
|
| + return;
|
| + }
|
| + }
|
| +
|
| + Send(new IconHostMsg_DidDownloadFavicon(
|
| + routing_id(), id, image_url, true, image_size, result_images));
|
| +}
|
| +
|
| +bool FaviconHelper::DownloadFavicon(int id,
|
| + const GURL& image_url,
|
| + int image_size) {
|
| + // Make sure webview was not shut down.
|
| + if (!render_view()->GetWebView())
|
| + return false;
|
| + // Create an image resource fetcher and assign it with a call back object.
|
| + image_fetchers_.push_back(new MultiResolutionImageResourceFetcher(
|
| + image_url, render_view()->GetWebView()->mainFrame(), id,
|
| + WebURLRequest::TargetIsFavicon,
|
| + base::Bind(&FaviconHelper::DidDownloadFavicon,
|
| + base::Unretained(this), image_size)));
|
| + return true;
|
| +}
|
| +
|
| +void FaviconHelper::DidDownloadFavicon(
|
| + int requested_size,
|
| + MultiResolutionImageResourceFetcher* fetcher,
|
| + const std::vector<SkBitmap>& images) {
|
| + // Notify requester of image download status.
|
| + Send(new IconHostMsg_DidDownloadFavicon(routing_id(),
|
| + fetcher->id(),
|
| + fetcher->image_url(),
|
| + images.empty(),
|
| + requested_size,
|
| + images));
|
| +
|
| + // Remove the image fetcher from our pending list. We're in the callback from
|
| + // MultiResolutionImageResourceFetcher, best to delay deletion.
|
| + ImageResourceFetcherList::iterator iter =
|
| + std::find(image_fetchers_.begin(), image_fetchers_.end(), fetcher);
|
| + if (iter != image_fetchers_.end()) {
|
| + image_fetchers_.weak_erase(iter);
|
| + MessageLoop::current()->DeleteSoon(FROM_HERE, fetcher);
|
| + }
|
| +}
|
| +
|
| +SkBitmap FaviconHelper::ImageFromDataUrl(const GURL& url) const {
|
| + std::string mime_type, char_set, data;
|
| + if (net::DataURL::Parse(url, &mime_type, &char_set, &data) && !data.empty()) {
|
| + // Decode the favicon using WebKit's image decoder.
|
| + webkit_glue::ImageDecoder decoder(
|
| + gfx::Size(gfx::kFaviconSize, gfx::kFaviconSize));
|
| + const unsigned char* src_data =
|
| + reinterpret_cast<const unsigned char*>(&data[0]);
|
| +
|
| + return decoder.Decode(src_data, data.size());
|
| + }
|
| + return SkBitmap();
|
| +}
|
| +
|
| +void FaviconHelper::SendUpdateFaviconURL(int32 routing_id,
|
| + int32 page_id,
|
| + const std::vector<FaviconURL>& urls) {
|
| + if (!urls.empty())
|
| + Send(new IconHostMsg_UpdateFaviconURL(routing_id, page_id, urls));
|
| +}
|
| +
|
| +bool FaviconHelper::OnMessageReceived(const IPC::Message& message) {
|
| + bool handled = true;
|
| + IPC_BEGIN_MESSAGE_MAP(FaviconHelper, message)
|
| + IPC_MESSAGE_HANDLER(IconMsg_DownloadFavicon, OnDownloadFavicon)
|
| + IPC_MESSAGE_UNHANDLED(handled = false)
|
| + IPC_END_MESSAGE_MAP()
|
| +
|
| + return handled;
|
| +}
|
| +
|
| +void FaviconHelper::DidStopLoading() {
|
| + int icon_types = WebIconURL::TypeFavicon;
|
| + if (chrome::kEnableTouchIcon)
|
| + icon_types |= WebIconURL::TypeTouchPrecomposed | WebIconURL::TypeTouch;
|
| +
|
| + WebVector<WebIconURL> icon_urls =
|
| + render_view()->GetWebView()->mainFrame()->iconURLs(icon_types);
|
| + std::vector<FaviconURL> urls;
|
| + for (size_t i = 0; i < icon_urls.size(); i++) {
|
| + WebURL url = icon_urls[i].iconURL();
|
| + if (!url.isEmpty())
|
| + urls.push_back(FaviconURL(url, ToFaviconType(icon_urls[i].iconType())));
|
| + }
|
| + SendUpdateFaviconURL(routing_id(), render_view()->GetPageId(), urls);
|
| +}
|
| +
|
| +void FaviconHelper::DidChangeIcon(WebKit::WebFrame* frame,
|
| + WebKit::WebIconURL::Type icon_type) {
|
| + if (frame->parent())
|
| + return;
|
| +
|
| + if (!chrome::kEnableTouchIcon &&
|
| + icon_type != WebIconURL::TypeFavicon)
|
| + return;
|
| +
|
| + WebVector<WebIconURL> icon_urls = frame->iconURLs(icon_type);
|
| + std::vector<FaviconURL> urls;
|
| + for (size_t i = 0; i < icon_urls.size(); i++) {
|
| + urls.push_back(FaviconURL(icon_urls[i].iconURL(),
|
| + ToFaviconType(icon_urls[i].iconType())));
|
| + }
|
| + SendUpdateFaviconURL(routing_id(), render_view()->GetPageId(), urls);
|
| +}
|
|
|