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 "components/plugins/renderer/plugin_placeholder.h"
7 #include "base/strings/string_util.h"
8 #include "content/public/common/web_preferences.h"
9 #include "content/public/renderer/render_frame.h"
10 #include "content/public/renderer/render_thread.h"
11 #include "gin/object_template_builder.h"
12 #include "third_party/WebKit/public/web/WebElement.h"
13 #include "third_party/WebKit/public/web/WebPluginContainer.h"
14 #include "third_party/re2/re2/re2.h"
18 // The placeholder is loaded in normal web renderer processes, so it should not
19 // have a chrome:// scheme that might let it be confused with a WebUI page.
20 const char kPluginPlaceholderDataURL
[] = "data:text/html,pluginplaceholderdata";
22 PluginPlaceholderBase::PluginPlaceholderBase(
23 content::RenderFrame
* render_frame
,
24 blink::WebLocalFrame
* frame
,
25 const blink::WebPluginParams
& params
,
26 const std::string
& html_data
)
27 : content::RenderFrameObserver(render_frame
),
29 plugin_params_(params
),
30 plugin_(WebViewPlugin::Create(this,
32 ? render_frame
->GetWebkitPreferences()
33 : content::WebPreferences(),
35 GURL(kPluginPlaceholderDataURL
))),
39 PluginPlaceholderBase::~PluginPlaceholderBase() {
42 const blink::WebPluginParams
& PluginPlaceholderBase::GetPluginParams() const {
43 return plugin_params_
;
46 void PluginPlaceholderBase::ShowContextMenu(const blink::WebMouseEvent
& event
) {
47 // Does nothing by default. Will be overridden if a specific browser wants
52 void PluginPlaceholderBase::PluginDestroyed() {
56 v8::Local
<v8::Object
> PluginPlaceholderBase::GetV8ScriptableObject(
57 v8::Isolate
* isolate
) const {
58 return v8::Local
<v8::Object
>();
61 void PluginPlaceholderBase::HidePlugin() {
65 blink::WebPluginContainer
* container
= plugin()->container();
66 blink::WebElement element
= container
->element();
67 element
.setAttribute("style", "display: none;");
68 // If we have a width and height, search for a parent (often <div>) with the
69 // same dimensions. If we find such a parent, hide that as well.
70 // This makes much more uncovered page content usable (including clickable)
71 // as opposed to merely visible.
72 // TODO(cevans) -- it's a foul heuristic but we're going to tolerate it for
73 // now for these reasons:
74 // 1) Makes the user experience better.
75 // 2) Foulness is encapsulated within this single function.
76 // 3) Confidence in no fasle positives.
77 // 4) Seems to have a good / low false negative rate at this time.
78 if (element
.hasAttribute("width") && element
.hasAttribute("height")) {
79 std::string
width_str("width:[\\s]*");
80 width_str
+= element
.getAttribute("width").utf8().data();
81 if (EndsWith(width_str
, "px", false)) {
82 width_str
= width_str
.substr(0, width_str
.length() - 2);
84 base::TrimWhitespace(width_str
, base::TRIM_TRAILING
, &width_str
);
85 width_str
+= "[\\s]*px";
86 std::string
height_str("height:[\\s]*");
87 height_str
+= element
.getAttribute("height").utf8().data();
88 if (EndsWith(height_str
, "px", false)) {
89 height_str
= height_str
.substr(0, height_str
.length() - 2);
91 base::TrimWhitespace(height_str
, base::TRIM_TRAILING
, &height_str
);
92 height_str
+= "[\\s]*px";
93 blink::WebNode parent
= element
;
94 while (!parent
.parentNode().isNull()) {
95 parent
= parent
.parentNode();
96 if (!parent
.isElementNode())
98 element
= parent
.toConst
<blink::WebElement
>();
99 if (element
.hasAttribute("style")) {
100 std::string style_str
= element
.getAttribute("style").utf8();
101 if (RE2::PartialMatch(style_str
, width_str
) &&
102 RE2::PartialMatch(style_str
, height_str
))
103 element
.setAttribute("style", "display: none;");
109 void PluginPlaceholderBase::HideCallback() {
110 content::RenderThread::Get()->RecordAction(
111 base::UserMetricsAction("Plugin_Hide_Click"));
115 void PluginPlaceholderBase::OnDestruct() {
119 blink::WebLocalFrame
* PluginPlaceholderBase::GetFrame() {
124 gin::WrapperInfo
PluginPlaceholder::kWrapperInfo
= {gin::kEmbedderNativeGin
};
126 PluginPlaceholder::PluginPlaceholder(content::RenderFrame
* render_frame
,
127 blink::WebLocalFrame
* frame
,
128 const blink::WebPluginParams
& params
,
129 const std::string
& html_data
)
130 : PluginPlaceholderBase(render_frame
, frame
, params
, html_data
) {
133 PluginPlaceholder::~PluginPlaceholder() {
136 v8::Local
<v8::Value
> PluginPlaceholder::GetV8Handle(v8::Isolate
* isolate
) {
137 return gin::CreateHandle(isolate
, this).ToV8();
140 gin::ObjectTemplateBuilder
PluginPlaceholder::GetObjectTemplateBuilder(
141 v8::Isolate
* isolate
) {
142 return gin::Wrappable
<PluginPlaceholder
>::GetObjectTemplateBuilder(isolate
)
143 .SetMethod
<void (plugins::PluginPlaceholder::*)()>(
144 "hide", &PluginPlaceholder::HideCallback
);
147 } // namespace plugins