1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chrome/renderer/chrome_render_frame_observer.h"
11 #include "base/command_line.h"
12 #include "base/strings/utf_string_conversions.h"
13 #include "chrome/common/chrome_switches.h"
14 #include "chrome/common/prerender_messages.h"
15 #include "chrome/common/render_messages.h"
16 #include "chrome/renderer/prerender/prerender_helper.h"
17 #include "components/printing/common/print_messages.h"
18 #include "components/printing/renderer/print_web_view_helper.h"
19 #include "content/public/renderer/render_frame.h"
20 #include "content/public/renderer/render_view.h"
21 #include "net/base/net_util.h"
22 #include "skia/ext/image_operations.h"
23 #include "third_party/WebKit/public/platform/WebImage.h"
24 #include "third_party/WebKit/public/platform/modules/app_banner/WebAppBannerPromptReply.h"
25 #include "third_party/WebKit/public/web/WebDataSource.h"
26 #include "third_party/WebKit/public/web/WebDocument.h"
27 #include "third_party/WebKit/public/web/WebElement.h"
28 #include "third_party/WebKit/public/web/WebFrame.h"
29 #include "third_party/WebKit/public/web/WebLocalFrame.h"
30 #include "third_party/WebKit/public/web/WebNode.h"
31 #include "third_party/WebKit/public/web/WebSecurityPolicy.h"
32 #include "third_party/skia/include/core/SkBitmap.h"
33 #include "ui/gfx/codec/jpeg_codec.h"
36 using blink::WebDataSource
;
37 using blink::WebElement
;
39 using content::SSLStatus
;
43 // If the source image is null or occupies less area than
44 // |thumbnail_min_area_pixels|, we return the image unmodified. Otherwise, we
45 // scale down the image so that the width and height do not exceed
46 // |thumbnail_max_size_pixels|, preserving the original aspect ratio.
47 SkBitmap
Downscale(const blink::WebImage
& image
,
48 int thumbnail_min_area_pixels
,
49 const gfx::Size
& thumbnail_max_size_pixels
) {
53 gfx::Size image_size
= image
.size();
55 if (image_size
.GetArea() < thumbnail_min_area_pixels
)
56 return image
.getSkBitmap();
58 if (image_size
.width() <= thumbnail_max_size_pixels
.width() &&
59 image_size
.height() <= thumbnail_max_size_pixels
.height())
60 return image
.getSkBitmap();
62 gfx::SizeF scaled_size
= image_size
;
64 if (scaled_size
.width() > thumbnail_max_size_pixels
.width()) {
65 scaled_size
.Scale(thumbnail_max_size_pixels
.width() / scaled_size
.width());
68 if (scaled_size
.height() > thumbnail_max_size_pixels
.height()) {
70 thumbnail_max_size_pixels
.height() / scaled_size
.height());
73 return skia::ImageOperations::Resize(image
.getSkBitmap(),
74 skia::ImageOperations::RESIZE_GOOD
,
75 static_cast<int>(scaled_size
.width()),
76 static_cast<int>(scaled_size
.height()));
81 ChromeRenderFrameObserver::ChromeRenderFrameObserver(
82 content::RenderFrame
* render_frame
)
83 : content::RenderFrameObserver(render_frame
) {
86 ChromeRenderFrameObserver::~ChromeRenderFrameObserver() {
89 bool ChromeRenderFrameObserver::OnMessageReceived(const IPC::Message
& message
) {
92 IPC_BEGIN_MESSAGE_MAP(ChromeRenderFrameObserver
, message
)
93 IPC_MESSAGE_HANDLER(PrerenderMsg_SetIsPrerendering
, OnSetIsPrerendering
)
94 IPC_MESSAGE_UNHANDLED(handled
= false)
99 IPC_BEGIN_MESSAGE_MAP(ChromeRenderFrameObserver
, message
)
100 IPC_MESSAGE_HANDLER(ChromeViewMsg_RequestReloadImageForContextNode
,
101 OnRequestReloadImageForContextNode
)
102 IPC_MESSAGE_HANDLER(ChromeViewMsg_RequestThumbnailForContextNode
,
103 OnRequestThumbnailForContextNode
)
104 IPC_MESSAGE_HANDLER(PrintMsg_PrintNodeUnderContextMenu
,
105 OnPrintNodeUnderContextMenu
)
106 IPC_MESSAGE_HANDLER(ChromeViewMsg_AppBannerPromptRequest
,
107 OnAppBannerPromptRequest
)
108 IPC_MESSAGE_HANDLER(ChromeViewMsg_AppBannerDebugMessageRequest
,
109 OnAppBannerDebugMessageRequest
)
110 IPC_MESSAGE_UNHANDLED(handled
= false)
111 IPC_END_MESSAGE_MAP()
116 void ChromeRenderFrameObserver::OnSetIsPrerendering(bool is_prerendering
) {
117 if (is_prerendering
) {
118 // If the PrerenderHelper for this frame already exists, don't create it. It
119 // can already be created for subframes during handling of
120 // RenderFrameCreated, if the parent frame was prerendering at time of
121 // subframe creation.
122 if (prerender::PrerenderHelper::Get(render_frame()))
125 // The PrerenderHelper will destroy itself either after recording histograms
126 // or on destruction of the RenderView.
127 new prerender::PrerenderHelper(render_frame());
131 void ChromeRenderFrameObserver::OnRequestReloadImageForContextNode() {
132 WebNode context_node
= render_frame()->GetContextMenuNode();
133 if (!context_node
.isNull() && context_node
.isElementNode() &&
134 render_frame()->GetWebFrame()) {
135 render_frame()->GetWebFrame()->reloadImage(context_node
);
139 void ChromeRenderFrameObserver::OnRequestThumbnailForContextNode(
140 int thumbnail_min_area_pixels
,
141 const gfx::Size
& thumbnail_max_size_pixels
) {
142 WebNode context_node
= render_frame()->GetContextMenuNode();
144 gfx::Size original_size
;
145 if (!context_node
.isNull() && context_node
.isElementNode()) {
146 blink::WebImage image
= context_node
.to
<WebElement
>().imageContents();
147 original_size
= image
.size();
148 thumbnail
= Downscale(image
,
149 thumbnail_min_area_pixels
,
150 thumbnail_max_size_pixels
);
154 if (thumbnail
.colorType() == kN32_SkColorType
)
157 thumbnail
.copyTo(&bitmap
, kN32_SkColorType
);
159 std::string thumbnail_data
;
160 SkAutoLockPixels
lock(bitmap
);
161 if (bitmap
.getPixels()) {
162 const int kDefaultQuality
= 90;
163 std::vector
<unsigned char> data
;
164 if (gfx::JPEGCodec::Encode(
165 reinterpret_cast<unsigned char*>(bitmap
.getAddr32(0, 0)),
166 gfx::JPEGCodec::FORMAT_SkBitmap
, bitmap
.width(), bitmap
.height(),
167 static_cast<int>(bitmap
.rowBytes()), kDefaultQuality
, &data
))
168 thumbnail_data
= std::string(data
.begin(), data
.end());
171 Send(new ChromeViewHostMsg_RequestThumbnailForContextNode_ACK(
172 routing_id(), thumbnail_data
, original_size
));
175 void ChromeRenderFrameObserver::OnPrintNodeUnderContextMenu() {
176 printing::PrintWebViewHelper
* helper
=
177 printing::PrintWebViewHelper::Get(render_frame()->GetRenderView());
179 helper
->PrintNode(render_frame()->GetContextMenuNode());
182 void ChromeRenderFrameObserver::DidFinishDocumentLoad() {
183 // If the navigation is to a localhost URL (and the flag is set to
184 // allow localhost SSL misconfigurations), print a warning to the
185 // console telling the developer to check their SSL configuration
186 // before going to production.
187 bool allow_localhost
= base::CommandLine::ForCurrentProcess()->HasSwitch(
188 switches::kAllowInsecureLocalhost
);
189 WebDataSource
* ds
= render_frame()->GetWebFrame()->dataSource();
191 if (allow_localhost
) {
192 SSLStatus ssl_status
= render_frame()->GetRenderView()->GetSSLStatusOfFrame(
193 render_frame()->GetWebFrame());
194 bool is_cert_error
= net::IsCertStatusError(ssl_status
.cert_status
) &&
195 !net::IsCertStatusMinorError(ssl_status
.cert_status
);
196 bool is_localhost
= net::IsLocalhost(GURL(ds
->request().url()).host());
198 if (is_cert_error
&& is_localhost
) {
199 render_frame()->GetWebFrame()->addMessageToConsole(
200 blink::WebConsoleMessage(
201 blink::WebConsoleMessage::LevelWarning
,
203 "This site does not have a valid SSL "
204 "certificate! Without SSL, your site's and "
205 "visitors' data is vulnerable to theft and "
206 "tampering. Get a valid SSL certificate before"
207 " releasing your website to the public.")));
212 void ChromeRenderFrameObserver::OnAppBannerPromptRequest(
214 const std::string
& platform
) {
215 // App banner prompt requests are handled in the general chrome render frame
216 // observer, not the AppBannerClient, as the AppBannerClient is created lazily
217 // by blink and may not exist when the request is sent.
218 blink::WebAppBannerPromptReply reply
= blink::WebAppBannerPromptReply::None
;
219 blink::WebString
web_platform(base::UTF8ToUTF16(platform
));
220 blink::WebVector
<blink::WebString
> web_platforms(&web_platform
, 1);
222 blink::WebLocalFrame
* frame
= render_frame()->GetWebFrame();
223 frame
->willShowInstallBannerPrompt(request_id
, web_platforms
, &reply
);
225 // Extract the referrer header for this site according to its referrer policy.
226 // Pass in an empty URL as the destination so that it is always treated
227 // as a cross-origin request.
228 std::string referrer
= blink::WebSecurityPolicy::generateReferrerHeader(
229 frame
->document().referrerPolicy(), GURL(),
230 frame
->document().outgoingReferrer()).utf8();
232 Send(new ChromeViewHostMsg_AppBannerPromptReply(
233 routing_id(), request_id
, reply
, referrer
));
236 void ChromeRenderFrameObserver::OnAppBannerDebugMessageRequest(
237 const std::string
& message
) {
238 render_frame()->GetWebFrame()->addMessageToConsole(blink::WebConsoleMessage(
239 blink::WebConsoleMessage::LevelDebug
, base::UTF8ToUTF16(message
)));