• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 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.
4 
5 #include "content/renderer/pepper/pepper_webplugin_impl.h"
6 
7 #include <cmath>
8 
9 #include "base/debug/crash_logging.h"
10 #include "base/message_loop/message_loop.h"
11 #include "content/public/common/page_zoom.h"
12 #include "content/public/renderer/content_renderer_client.h"
13 #include "content/renderer/pepper/message_channel.h"
14 #include "content/renderer/pepper/npobject_var.h"
15 #include "content/renderer/pepper/pepper_plugin_instance_impl.h"
16 #include "content/renderer/pepper/plugin_module.h"
17 #include "content/renderer/render_view_impl.h"
18 #include "ppapi/shared_impl/ppapi_globals.h"
19 #include "ppapi/shared_impl/var_tracker.h"
20 #include "third_party/WebKit/public/platform/WebPoint.h"
21 #include "third_party/WebKit/public/platform/WebRect.h"
22 #include "third_party/WebKit/public/platform/WebSize.h"
23 #include "third_party/WebKit/public/platform/WebURLLoaderClient.h"
24 #include "third_party/WebKit/public/web/WebBindings.h"
25 #include "third_party/WebKit/public/web/WebDocument.h"
26 #include "third_party/WebKit/public/web/WebElement.h"
27 #include "third_party/WebKit/public/web/WebFrame.h"
28 #include "third_party/WebKit/public/web/WebPluginContainer.h"
29 #include "third_party/WebKit/public/web/WebPluginParams.h"
30 #include "third_party/WebKit/public/web/WebPrintParams.h"
31 #include "third_party/WebKit/public/web/WebPrintScalingOption.h"
32 #include "url/gurl.h"
33 
34 using ppapi::NPObjectVar;
35 using blink::WebCanvas;
36 using blink::WebPlugin;
37 using blink::WebPluginContainer;
38 using blink::WebPluginParams;
39 using blink::WebPoint;
40 using blink::WebPrintParams;
41 using blink::WebRect;
42 using blink::WebSize;
43 using blink::WebString;
44 using blink::WebURL;
45 using blink::WebVector;
46 
47 namespace content {
48 
49 struct PepperWebPluginImpl::InitData {
50   scoped_refptr<PluginModule> module;
51   base::WeakPtr<RenderViewImpl> render_view;
52   RenderFrame* render_frame;
53   std::vector<std::string> arg_names;
54   std::vector<std::string> arg_values;
55   GURL url;
56 };
57 
PepperWebPluginImpl(PluginModule * plugin_module,const WebPluginParams & params,const base::WeakPtr<RenderViewImpl> & render_view,RenderFrame * render_frame)58 PepperWebPluginImpl::PepperWebPluginImpl(
59     PluginModule* plugin_module,
60     const WebPluginParams& params,
61     const base::WeakPtr<RenderViewImpl>& render_view,
62     RenderFrame* render_frame)
63     : init_data_(new InitData()),
64       full_frame_(params.loadManually),
65       instance_object_(PP_MakeUndefined()),
66       container_(NULL) {
67   DCHECK(plugin_module);
68   init_data_->module = plugin_module;
69   init_data_->render_view = render_view;
70   init_data_->render_frame = render_frame;
71   for (size_t i = 0; i < params.attributeNames.size(); ++i) {
72     init_data_->arg_names.push_back(params.attributeNames[i].utf8());
73     init_data_->arg_values.push_back(params.attributeValues[i].utf8());
74   }
75   init_data_->url = params.url;
76 
77   // Set subresource URL for crash reporting.
78   base::debug::SetCrashKeyValue("subresource_url", init_data_->url.spec());
79 }
80 
~PepperWebPluginImpl()81 PepperWebPluginImpl::~PepperWebPluginImpl() {
82 }
83 
container() const84 blink::WebPluginContainer* PepperWebPluginImpl::container() const {
85   return container_;
86 }
87 
initialize(WebPluginContainer * container)88 bool PepperWebPluginImpl::initialize(WebPluginContainer* container) {
89   // The plugin delegate may have gone away.
90   instance_ = init_data_->module->CreateInstance(
91       init_data_->render_view->main_render_frame(), container, init_data_->url);
92   if (!instance_.get())
93     return false;
94 
95   // Enable script objects for this plugin.
96   container->allowScriptObjects();
97 
98   bool success = instance_->Initialize(init_data_->arg_names,
99                                        init_data_->arg_values,
100                                        full_frame_);
101   if (!success) {
102     instance_->Delete();
103     instance_ = NULL;
104 
105     blink::WebPlugin* replacement_plugin =
106         GetContentClient()->renderer()->CreatePluginReplacement(
107             init_data_->render_frame, init_data_->module->path());
108     if (!replacement_plugin || !replacement_plugin->initialize(container))
109       return false;
110 
111     container->setPlugin(replacement_plugin);
112     return true;
113   }
114 
115   init_data_.reset();
116   container_ = container;
117   return true;
118 }
119 
destroy()120 void PepperWebPluginImpl::destroy() {
121   // Tell |container_| to clear references to this plugin's script objects.
122   if (container_)
123     container_->clearScriptObjects();
124 
125   if (instance_.get()) {
126     ppapi::PpapiGlobals::Get()->GetVarTracker()->ReleaseVar(instance_object_);
127     instance_object_ = PP_MakeUndefined();
128     instance_->Delete();
129     instance_ = NULL;
130   }
131 
132   base::MessageLoop::current()->DeleteSoon(FROM_HERE, this);
133 }
134 
scriptableObject()135 NPObject* PepperWebPluginImpl::scriptableObject() {
136   // Call through the plugin to get its instance object. The plugin should pass
137   // us a reference which we release in destroy().
138   if (instance_object_.type == PP_VARTYPE_UNDEFINED)
139     instance_object_ = instance_->GetInstanceObject();
140   // GetInstanceObject talked to the plugin which may have removed the instance
141   // from the DOM, in which case instance_ would be NULL now.
142   if (!instance_.get())
143     return NULL;
144 
145   scoped_refptr<NPObjectVar> object(NPObjectVar::FromPPVar(instance_object_));
146   // If there's an InstanceObject, tell the Instance's MessageChannel to pass
147   // any non-postMessage calls to it.
148   if (object.get()) {
149     instance_->message_channel().SetPassthroughObject(object->np_object());
150   }
151   NPObject* message_channel_np_object(instance_->message_channel().np_object());
152   // The object is expected to be retained before it is returned.
153   blink::WebBindings::retainObject(message_channel_np_object);
154   return message_channel_np_object;
155 }
156 
pluginNPP()157 NPP PepperWebPluginImpl::pluginNPP() {
158   return instance_->instanceNPP();
159 }
160 
getFormValue(WebString & value)161 bool PepperWebPluginImpl::getFormValue(WebString& value) {
162   return false;
163 }
164 
paint(WebCanvas * canvas,const WebRect & rect)165 void PepperWebPluginImpl::paint(WebCanvas* canvas, const WebRect& rect) {
166   if (!instance_->FlashIsFullscreenOrPending())
167     instance_->Paint(canvas, plugin_rect_, rect);
168 }
169 
updateGeometry(const WebRect & window_rect,const WebRect & clip_rect,const WebVector<WebRect> & cut_outs_rects,bool is_visible)170 void PepperWebPluginImpl::updateGeometry(
171     const WebRect& window_rect,
172     const WebRect& clip_rect,
173     const WebVector<WebRect>& cut_outs_rects,
174     bool is_visible) {
175   plugin_rect_ = window_rect;
176   if (!instance_->FlashIsFullscreenOrPending()) {
177     std::vector<gfx::Rect> cut_outs;
178     for (size_t i = 0; i < cut_outs_rects.size(); ++i)
179       cut_outs.push_back(cut_outs_rects[i]);
180     instance_->ViewChanged(plugin_rect_, clip_rect, cut_outs);
181   }
182 }
183 
updateFocus(bool focused)184 void PepperWebPluginImpl::updateFocus(bool focused) {
185   instance_->SetWebKitFocus(focused);
186 }
187 
updateVisibility(bool visible)188 void PepperWebPluginImpl::updateVisibility(bool visible) {
189 }
190 
acceptsInputEvents()191 bool PepperWebPluginImpl::acceptsInputEvents() {
192   return true;
193 }
194 
handleInputEvent(const blink::WebInputEvent & event,blink::WebCursorInfo & cursor_info)195 bool PepperWebPluginImpl::handleInputEvent(const blink::WebInputEvent& event,
196                                            blink::WebCursorInfo& cursor_info) {
197   if (instance_->FlashIsFullscreenOrPending())
198     return false;
199   return instance_->HandleInputEvent(event, &cursor_info);
200 }
201 
didReceiveResponse(const blink::WebURLResponse & response)202 void PepperWebPluginImpl::didReceiveResponse(
203     const blink::WebURLResponse& response) {
204   DCHECK(!instance_->document_loader());
205   instance_->HandleDocumentLoad(response);
206 }
207 
didReceiveData(const char * data,int data_length)208 void PepperWebPluginImpl::didReceiveData(const char* data, int data_length) {
209   blink::WebURLLoaderClient* document_loader = instance_->document_loader();
210   if (document_loader)
211     document_loader->didReceiveData(NULL, data, data_length, 0);
212 }
213 
didFinishLoading()214 void PepperWebPluginImpl::didFinishLoading() {
215   blink::WebURLLoaderClient* document_loader = instance_->document_loader();
216   if (document_loader)
217     document_loader->didFinishLoading(NULL, 0.0);
218 }
219 
didFailLoading(const blink::WebURLError & error)220 void PepperWebPluginImpl::didFailLoading(const blink::WebURLError& error) {
221   blink::WebURLLoaderClient* document_loader = instance_->document_loader();
222   if (document_loader)
223     document_loader->didFail(NULL, error);
224 }
225 
didFinishLoadingFrameRequest(const blink::WebURL & url,void * notify_data)226 void PepperWebPluginImpl::didFinishLoadingFrameRequest(
227     const blink::WebURL& url,
228     void* notify_data) {
229 }
230 
didFailLoadingFrameRequest(const blink::WebURL & url,void * notify_data,const blink::WebURLError & error)231 void PepperWebPluginImpl::didFailLoadingFrameRequest(
232     const blink::WebURL& url,
233     void* notify_data,
234     const blink::WebURLError& error) {
235 }
236 
hasSelection() const237 bool PepperWebPluginImpl::hasSelection() const {
238   return !selectionAsText().isEmpty();
239 }
240 
selectionAsText() const241 WebString PepperWebPluginImpl::selectionAsText() const {
242   return instance_->GetSelectedText(false);
243 }
244 
selectionAsMarkup() const245 WebString PepperWebPluginImpl::selectionAsMarkup() const {
246   return instance_->GetSelectedText(true);
247 }
248 
linkAtPosition(const WebPoint & position) const249 WebURL PepperWebPluginImpl::linkAtPosition(const WebPoint& position) const {
250   return GURL(instance_->GetLinkAtPosition(position));
251 }
252 
setZoomLevel(double level,bool text_only)253 void PepperWebPluginImpl::setZoomLevel(double level, bool text_only) {
254   instance_->Zoom(content::ZoomLevelToZoomFactor(level), text_only);
255 }
256 
startFind(const blink::WebString & search_text,bool case_sensitive,int identifier)257 bool PepperWebPluginImpl::startFind(const blink::WebString& search_text,
258                                     bool case_sensitive,
259                                     int identifier) {
260   return instance_->StartFind(search_text, case_sensitive, identifier);
261 }
262 
selectFindResult(bool forward)263 void PepperWebPluginImpl::selectFindResult(bool forward) {
264   instance_->SelectFindResult(forward);
265 }
266 
stopFind()267 void PepperWebPluginImpl::stopFind() {
268   instance_->StopFind();
269 }
270 
supportsPaginatedPrint()271 bool PepperWebPluginImpl::supportsPaginatedPrint() {
272   return instance_->SupportsPrintInterface();
273 }
274 
isPrintScalingDisabled()275 bool PepperWebPluginImpl::isPrintScalingDisabled() {
276   return instance_->IsPrintScalingDisabled();
277 }
278 
printBegin(const WebPrintParams & print_params)279 int PepperWebPluginImpl::printBegin(const WebPrintParams& print_params) {
280   return instance_->PrintBegin(print_params);
281 }
282 
printPage(int page_number,blink::WebCanvas * canvas)283 bool PepperWebPluginImpl::printPage(int page_number,
284                                     blink::WebCanvas* canvas) {
285   return instance_->PrintPage(page_number, canvas);
286 }
287 
printEnd()288 void PepperWebPluginImpl::printEnd() {
289   return instance_->PrintEnd();
290 }
291 
canRotateView()292 bool PepperWebPluginImpl::canRotateView() {
293   return instance_->CanRotateView();
294 }
295 
rotateView(RotationType type)296 void PepperWebPluginImpl::rotateView(RotationType type) {
297   instance_->RotateView(type);
298 }
299 
isPlaceholder()300 bool PepperWebPluginImpl::isPlaceholder() {
301   return false;
302 }
303 
304 }  // namespace content
305