1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
|
// Copyright (c) 2011 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.
#ifndef WEBKIT_PLUGINS_PPAPI_PPAPI_PLUGIN_INSTANCE_H_
#define WEBKIT_PLUGINS_PPAPI_PPAPI_PLUGIN_INSTANCE_H_
#include <map>
#include <set>
#include <string>
#include <vector>
#include "base/basictypes.h"
#include "base/memory/ref_counted.h"
#include "base/memory/scoped_ptr.h"
#include "base/string16.h"
#include "googleurl/src/gurl.h"
#include "ppapi/c/dev/pp_cursor_type_dev.h"
#include "ppapi/c/dev/ppp_graphics_3d_dev.h"
// TODO(dmichael): Remove the 0.3 printing interface and remove the following
// #define.
#define PPP_PRINTING_DEV_USE_0_4 1
#include "ppapi/c/dev/ppp_printing_dev.h"
#include "ppapi/c/pp_instance.h"
#include "ppapi/c/pp_resource.h"
#include "ppapi/c/ppp_instance.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "third_party/skia/include/core/SkRefCnt.h"
#include "third_party/WebKit/Source/WebKit/chromium/public/WebCanvas.h"
#include "ui/gfx/rect.h"
#include "webkit/plugins/ppapi/plugin_delegate.h"
typedef struct NPObject NPObject;
struct PP_Var;
struct PPB_Instance;
struct PPB_Instance_Private;
struct PPB_Fullscreen_Dev;
struct PPB_Messaging;
struct PPB_Zoom_Dev;
struct PPP_Find_Dev;
struct PPP_Instance_Private;
struct PPP_Messaging;
struct PPP_Pdf;
struct PPP_Selection_Dev;
struct PPP_Zoom_Dev;
class SkBitmap;
class TransportDIB;
namespace gfx {
class Rect;
}
namespace WebKit {
struct WebCursorInfo;
class WebInputEvent;
class WebPluginContainer;
}
namespace webkit {
namespace ppapi {
class FullscreenContainer;
class MessageChannel;
class ObjectVar;
class PluginDelegate;
class PluginModule;
class PluginObject;
class PPB_Graphics2D_Impl;
class PPB_ImageData_Impl;
class PPB_Surface3D_Impl;
class PPB_URLLoader_Impl;
class PPB_URLRequestInfo_Impl;
class Resource;
// Represents one time a plugin appears on one web page.
//
// Note: to get from a PP_Instance to a PluginInstance*, use the
// ResourceTracker.
class PluginInstance : public base::RefCounted<PluginInstance> {
public:
struct PPP_Instance_Combined;
PluginInstance(PluginDelegate* delegate,
PluginModule* module,
PPP_Instance_Combined* instance_interface);
// Delete should be called by the WebPlugin before this destructor.
~PluginInstance();
// Return a PPB_Instance interface compatible with the given interface name,
// if one is available. Returns NULL if the requested interface is
// not supported.
static const void* GetInterface(const char* if_name);
static const PPB_Instance_Private* GetPrivateInterface();
// Returns a pointer to the interface implementing PPB_Find that is
// exposed to the plugin.
static const PPB_Fullscreen_Dev* GetFullscreenInterface();
static const PPB_Messaging* GetMessagingInterface();
static const PPB_Zoom_Dev* GetZoomInterface();
PluginDelegate* delegate() const { return delegate_; }
PluginModule* module() const { return module_.get(); }
MessageChannel& message_channel() { return *message_channel_; }
WebKit::WebPluginContainer* container() const { return container_; }
const gfx::Rect& position() const { return position_; }
const gfx::Rect& clip() const { return clip_; }
int find_identifier() const { return find_identifier_; }
void set_always_on_top(bool on_top) { always_on_top_ = on_top; }
// Returns the PP_Instance uniquely identifying this instance. Guaranteed
// nonzero.
PP_Instance pp_instance() const { return pp_instance_; }
// Does some pre-destructor cleanup on the instance. This is necessary
// because some cleanup depends on the plugin instance still existing (like
// calling the plugin's DidDestroy function). This function is called from
// the WebPlugin implementation when WebKit is about to remove the plugin.
void Delete();
// Paints the current backing store to the web page.
void Paint(WebKit::WebCanvas* canvas,
const gfx::Rect& plugin_rect,
const gfx::Rect& paint_rect);
// Schedules a paint of the page for the given region. The coordinates are
// relative to the top-left of the plugin. This does nothing if the plugin
// has not yet been positioned. You can supply an empty gfx::Rect() to
// invalidate the entire plugin.
void InvalidateRect(const gfx::Rect& rect);
// Schedules a scroll of the plugin. This uses optimized scrolling only for
// full-frame plugins, as otherwise there could be other elements on top. The
// slow path can also be triggered if there is an overlapping frame.
void ScrollRect(int dx, int dy, const gfx::Rect& rect);
// If the plugin instance is backed by a texture, return its texture ID in the
// compositor's namespace. Otherwise return 0. Returns 0 by default.
unsigned GetBackingTextureId();
// Commit the backing texture to the screen once the side effects some
// rendering up to an offscreen SwapBuffers are visible.
void CommitBackingTexture();
// Called when the out-of-process plugin implementing this instance crashed.
void InstanceCrashed();
// PPB_Instance and PPB_Instance_Private implementation.
PP_Var GetWindowObject();
PP_Var GetOwnerElementObject();
bool BindGraphics(PP_Resource graphics_id);
const GURL& plugin_url() const { return plugin_url_; }
bool full_frame() const { return full_frame_; }
// If |type| is not PP_CURSORTYPE_CUSTOM, |custom_image| and |hot_spot| are
// ignored.
bool SetCursor(PP_CursorType_Dev type,
PP_Resource custom_image,
const PP_Point* hot_spot);
PP_Var ExecuteScript(PP_Var script, PP_Var* exception);
// PPP_Instance and PPP_Instance_Private pass-through.
bool Initialize(WebKit::WebPluginContainer* container,
const std::vector<std::string>& arg_names,
const std::vector<std::string>& arg_values,
const GURL& plugin_url,
bool full_frame);
bool HandleDocumentLoad(PPB_URLLoader_Impl* loader);
bool HandleInputEvent(const WebKit::WebInputEvent& event,
WebKit::WebCursorInfo* cursor_info);
PP_Var GetInstanceObject();
void ViewChanged(const gfx::Rect& position, const gfx::Rect& clip);
// Notifications about focus changes, see has_webkit_focus_ below.
void SetWebKitFocus(bool has_focus);
void SetContentAreaFocus(bool has_focus);
// Notifications that the view has rendered the page and that it has been
// flushed to the screen. These messages are used to send Flush callbacks to
// the plugin for DeviceContext2D.
void ViewInitiatedPaint();
void ViewFlushedPaint();
// If this plugin can be painted merely by copying the backing store to the
// screen, and the plugin bounds encloses the given paint bounds, returns
// true. In this case, the location, clipping, and ID of the backing store
// will be filled into the given output parameters.
bool GetBitmapForOptimizedPluginPaint(
const gfx::Rect& paint_bounds,
TransportDIB** dib,
gfx::Rect* dib_bounds,
gfx::Rect* clip);
// Tracks all live PluginObjects.
void AddPluginObject(PluginObject* plugin_object);
void RemovePluginObject(PluginObject* plugin_object);
string16 GetSelectedText(bool html);
string16 GetLinkAtPosition(const gfx::Point& point);
void Zoom(double factor, bool text_only);
bool StartFind(const string16& search_text,
bool case_sensitive,
int identifier);
void SelectFindResult(bool forward);
void StopFind();
bool SupportsPrintInterface();
int PrintBegin(const gfx::Rect& printable_area, int printer_dpi);
bool PrintPage(int page_number, WebKit::WebCanvas* canvas);
void PrintEnd();
void Graphics3DContextLost();
// Implementation of PPB_Fullscreen_Dev.
// Because going to fullscreen is asynchronous (but going out is not), there
// are 3 states:
// - normal (fullscreen_container_ == NULL)
// - fullscreen pending (fullscreen_container_ != NULL, fullscreen_ == false)
// - fullscreen (fullscreen_container_ != NULL, fullscreen_ = true)
//
// In normal state, events come from webkit and painting goes back to it.
// In fullscreen state, events come from the fullscreen container, and
// painting goes back to it
// In pending state, events from webkit are ignored, and as soon as we receive
// events from the fullscreen container, we go to the fullscreen state.
bool IsFullscreen();
bool IsFullscreenOrPending();
// Switches between fullscreen and normal mode. If |delay_report| is set to
// false, it may report the new state through DidChangeView immediately. If
// true, it will delay it. When called from the plugin, delay_report should be
// true to avoid re-entrancy.
void SetFullscreen(bool fullscreen, bool delay_report);
// Implementation of PPB_Flash.
int32_t Navigate(PPB_URLRequestInfo_Impl* request,
const char* target,
bool from_user_action);
// Implementation of PPB_Messaging and PPP_Messaging.
void PostMessage(PP_Var message);
void HandleMessage(PP_Var message);
PluginDelegate::PlatformContext3D* CreateContext3D();
// Tracks all live ObjectVar. This is so we can map between PluginModule +
// NPObject and get the ObjectVar corresponding to it. This Add/Remove
// function should be called by the ObjectVar when it is created and
// destroyed.
void AddNPObjectVar(ObjectVar* object_var);
void RemoveNPObjectVar(ObjectVar* object_var);
// Looks up a previously registered ObjectVar for the given NPObject and
// module. Returns NULL if there is no ObjectVar corresponding to the given
// NPObject for the given module. See AddNPObjectVar above.
ObjectVar* ObjectVarForNPObject(NPObject* np_object) const;
// Returns true iff the plugin is a full-page plugin (i.e. not in an iframe or
// embedded in a page).
bool IsFullPagePlugin() const;
FullscreenContainer* fullscreen_container() const {
return fullscreen_container_;
}
// TODO(dmichael): Remove this when all plugins are ported to use scripting
// from private interfaces.
struct PPP_Instance_Combined : public PPP_Instance_0_5 {
PPP_Instance_Combined(const PPP_Instance_0_5& instance_if);
PPP_Instance_Combined(const PPP_Instance_0_4& instance_if);
struct PP_Var (*GetInstanceObject_0_4)(PP_Instance instance);
};
template <class InterfaceType>
static PPP_Instance_Combined* new_instance_interface(
const void* interface_object) {
return new PPP_Instance_Combined(
*static_cast<const InterfaceType*>(interface_object));
}
private:
bool LoadFindInterface();
bool LoadMessagingInterface();
bool LoadPdfInterface();
bool LoadSelectionInterface();
bool LoadPrintInterface();
bool LoadPrivateInterface();
bool LoadZoomInterface();
// Determines if we think the plugin has focus, both content area and webkit
// (see has_webkit_focus_ below).
bool PluginHasFocus() const;
// Reports the current plugin geometry to the plugin by calling
// DidChangeView.
void ReportGeometry();
// Queries the plugin for supported print formats and sets |format| to the
// best format to use. Returns false if the plugin does not support any
// print format that we can handle (we can handle raster and PDF).
bool GetPreferredPrintOutputFormat(PP_PrintOutputFormat_Dev* format);
bool PrintPDFOutput(PP_Resource print_output, WebKit::WebCanvas* canvas);
bool PrintRasterOutput(PP_Resource print_output, WebKit::WebCanvas* canvas);
#if defined(OS_WIN)
bool DrawJPEGToPlatformDC(const SkBitmap& bitmap,
const gfx::Rect& printable_area,
WebKit::WebCanvas* canvas);
#elif defined(OS_MACOSX) && !defined(USE_SKIA)
// Draws the given kARGB_8888_Config bitmap to the specified canvas starting
// at the specified destination rect.
void DrawSkBitmapToCanvas(const SkBitmap& bitmap, WebKit::WebCanvas* canvas,
const gfx::Rect& dest_rect, int canvas_height);
#endif // OS_MACOSX
// Get the bound graphics context as a concrete 2D graphics context or returns
// null if the context is not 2D.
PPB_Graphics2D_Impl* bound_graphics_2d() const;
// Get the bound 3D graphics surface.
// Returns NULL if bound graphics is not a 3D surface.
PPB_Surface3D_Impl* bound_graphics_3d() const;
// Sets the id of the texture that the plugin draws to. The id is in the
// compositor space so it can use it to composite with rest of the page.
// A value of zero indicates the plugin is not backed by a texture.
void setBackingTextureId(unsigned int id);
// Internal helper function for PrintPage().
bool PrintPageHelper(PP_PrintPageNumberRange_Dev* page_ranges,
int num_ranges,
WebKit::WebCanvas* canvas);
PluginDelegate* delegate_;
scoped_refptr<PluginModule> module_;
scoped_ptr<PPP_Instance_Combined> instance_interface_;
PP_Instance pp_instance_;
// NULL until we have been initialized.
WebKit::WebPluginContainer* container_;
// Plugin URL.
GURL plugin_url_;
// Indicates whether this is a full frame instance, which means it represents
// an entire document rather than an embed tag.
bool full_frame_;
// Position in the viewport (which moves as the page is scrolled) of this
// plugin. This will be a 0-sized rectangle if the plugin has not yet been
// laid out.
gfx::Rect position_;
// Current clip rect. This will be empty if the plugin is not currently
// visible. This is in the plugin's coordinate system, so fully visible will
// be (0, 0, w, h) regardless of scroll position.
gfx::Rect clip_;
// The current device context for painting in 2D or 3D.
scoped_refptr<Resource> bound_graphics_;
// We track two types of focus, one from WebKit, which is the focus among
// all elements of the page, one one from the browser, which is whether the
// tab/window has focus. We tell the plugin it has focus only when both of
// these values are set to true.
bool has_webkit_focus_;
bool has_content_area_focus_;
// The id of the current find operation, or -1 if none is in process.
int find_identifier_;
// The plugin-provided interfaces.
const PPP_Find_Dev* plugin_find_interface_;
const PPP_Messaging* plugin_messaging_interface_;
const PPP_Pdf* plugin_pdf_interface_;
const PPP_Instance_Private* plugin_private_interface_;
const PPP_Selection_Dev* plugin_selection_interface_;
const PPP_Zoom_Dev* plugin_zoom_interface_;
// A flag to indicate whether we have asked this plugin instance for its
// messaging interface, so that we can ask only once.
bool checked_for_plugin_messaging_interface_;
// This is only valid between a successful PrintBegin call and a PrintEnd
// call.
PP_PrintSettings_Dev current_print_settings_;
#if defined(OS_MACOSX)
// On the Mac, when we draw the bitmap to the PDFContext, it seems necessary
// to keep the pixels valid until CGContextEndPage is called. We use this
// variable to hold on to the pixels.
scoped_refptr<PPB_ImageData_Impl> last_printed_page_;
#endif // defined(OS_MACOSX)
#if WEBKIT_USING_SKIA
// When printing to PDF (print preview, Linux) the entire document goes into
// one metafile. However, when users print only a subset of all the pages,
// it is impossible to know if a call to PrintPage() is the last call.
// Thus in PrintPage(), just store the page number in |ranges_|.
// The hack is in PrintEnd(), where a valid |canvas_| is preserved in
// PrintWebViewHelper::PrintPages. This makes it possible to generate the
// entire PDF given the variables below:
//
// The most recently used WebCanvas, guaranteed to be valid.
SkRefPtr<WebKit::WebCanvas> canvas_;
// An array of page ranges.
std::vector<PP_PrintPageNumberRange_Dev> ranges_;
#endif // WEBKIT_USING_SKIA
// The plugin print interface. This nested struct adds functions needed for
// backwards compatibility.
struct PPP_Printing_Dev_Combined : public PPP_Printing_Dev {
// Conversion constructor for the most current interface. Sets all old
// functions to NULL, so we know not to try to use them.
PPP_Printing_Dev_Combined(const PPP_Printing_Dev& base_if)
: PPP_Printing_Dev(base_if),
QuerySupportedFormats_0_3(NULL),
Begin_0_3(NULL) {}
// Conversion constructor for version 0.3. Sets unsupported functions to
// NULL, so we know not to try to use them.
PPP_Printing_Dev_Combined(const PPP_Printing_Dev_0_3& old_if)
: PPP_Printing_Dev(), // NOTE: The parens are important, to zero-
// initialize the struct.
// Except older version of g++ doesn't!
// So do it explicitly in the ctor.
QuerySupportedFormats_0_3(old_if.QuerySupportedFormats),
Begin_0_3(old_if.Begin) {
QuerySupportedFormats = NULL;
Begin = NULL;
PrintPages = old_if.PrintPages;
End = old_if.End;
}
// The 0.3 version of 'QuerySupportedFormats'.
PP_PrintOutputFormat_Dev_0_3* (*QuerySupportedFormats_0_3)(
PP_Instance instance, uint32_t* format_count);
// The 0.3 version of 'Begin'.
int32_t (*Begin_0_3)(PP_Instance instance,
const struct PP_PrintSettings_Dev_0_3* print_settings);
};
scoped_ptr<PPP_Printing_Dev_Combined> plugin_print_interface_;
// The plugin 3D interface.
const PPP_Graphics3D_Dev* plugin_graphics_3d_interface_;
// Contains the cursor if it's set by the plugin.
scoped_ptr<WebKit::WebCursorInfo> cursor_;
// Set to true if this plugin thinks it will always be on top. This allows us
// to use a more optimized painting path in some cases.
bool always_on_top_;
// Plugin container for fullscreen mode. NULL if not in fullscreen mode. Note:
// there is a transition state where fullscreen_container_ is non-NULL but
// fullscreen_ is false (see above).
FullscreenContainer* fullscreen_container_;
// True if we are in fullscreen mode. Note: it is false during the transition.
bool fullscreen_;
// The MessageChannel used to implement bidirectional postMessage for the
// instance.
scoped_ptr<MessageChannel> message_channel_;
// Bitmap for crashed plugin. Lazily initialized, non-owning pointer.
SkBitmap* sad_plugin_;
typedef std::set<PluginObject*> PluginObjectSet;
PluginObjectSet live_plugin_objects_;
// Tracks all live ObjectVars used by this module so we can map NPObjects to
// the corresponding object. These are non-owning references.
typedef std::map<NPObject*, ObjectVar*> NPObjectToObjectVarMap;
NPObjectToObjectVarMap np_object_to_object_var_;
DISALLOW_COPY_AND_ASSIGN(PluginInstance);
};
} // namespace ppapi
} // namespace webkit
#endif // WEBKIT_PLUGINS_PPAPI_PPAPI_PLUGIN_INSTANCE_H_
|