// Copyright (c) 2010 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_GLUE_PLUGINS_WEBPLUGIN_DELEGATE_IMPL_H_ #define WEBKIT_GLUE_PLUGINS_WEBPLUGIN_DELEGATE_IMPL_H_ #include "build/build_config.h" #include #include #include #include "base/ref_counted.h" #include "base/task.h" #include "base/time.h" #include "base/timer.h" #include "gfx/native_widget_types.h" #include "gfx/rect.h" #include "third_party/npapi/bindings/npapi.h" #include "webkit/glue/plugins/webplugin_delegate.h" #include "webkit/glue/webcursor.h" #if defined(OS_MACOSX) #include "app/surface/accelerated_surface_mac.h" #endif #if defined(USE_X11) #include "app/x11_util.h" typedef struct _GdkDrawable GdkPixmap; #endif class FilePath; namespace NPAPI { class PluginInstance; } namespace WebKit { class WebMouseEvent; } #if defined(OS_MACOSX) class ExternalDragTracker; #ifndef NP_NO_QUICKDRAW class QuickDrawDrawingManager; #endif #ifdef __OBJC__ @class CALayer; @class CARenderer; #else class CALayer; class CARenderer; #endif #endif // An implementation of WebPluginDelegate that runs in the plugin process, // proxied from the renderer by WebPluginDelegateProxy. class WebPluginDelegateImpl : public webkit_glue::WebPluginDelegate { public: enum PluginQuirks { PLUGIN_QUIRK_SETWINDOW_TWICE = 1, // Win32 PLUGIN_QUIRK_THROTTLE_WM_USER_PLUS_ONE = 2, // Win32 PLUGIN_QUIRK_DONT_CALL_WND_PROC_RECURSIVELY = 4, // Win32 PLUGIN_QUIRK_DONT_SET_NULL_WINDOW_HANDLE_ON_DESTROY = 8, // Win32 PLUGIN_QUIRK_DONT_ALLOW_MULTIPLE_INSTANCES = 16, // Win32 PLUGIN_QUIRK_DIE_AFTER_UNLOAD = 32, // Win32 PLUGIN_QUIRK_PATCH_SETCURSOR = 64, // Win32 PLUGIN_QUIRK_BLOCK_NONSTANDARD_GETURL_REQUESTS = 128, // Win32 PLUGIN_QUIRK_WINDOWLESS_OFFSET_WINDOW_TO_DRAW = 256, // Linux PLUGIN_QUIRK_WINDOWLESS_INVALIDATE_AFTER_SET_WINDOW = 512, // Linux PLUGIN_QUIRK_NO_WINDOWLESS = 1024, // Windows PLUGIN_QUIRK_PATCH_REGENUMKEYEXW = 2048, // Windows PLUGIN_QUIRK_ALWAYS_NOTIFY_SUCCESS = 4096, // Windows PLUGIN_QUIRK_ALLOW_FASTER_QUICKDRAW_PATH = 8192, // Mac PLUGIN_QUIRK_HANDLE_MOUSE_CAPTURE = 16384, // Windows PLUGIN_QUIRK_WINDOWLESS_NO_RIGHT_CLICK = 32768, // Linux PLUGIN_QUIRK_IGNORE_FIRST_SETWINDOW_CALL = 65536, // Windows. }; static WebPluginDelegateImpl* Create(const FilePath& filename, const std::string& mime_type, gfx::PluginWindowHandle containing_view); static bool IsPluginDelegateWindow(gfx::NativeWindow window); static bool GetPluginNameFromWindow(gfx::NativeWindow window, std::wstring *plugin_name); // Returns true if the window handle passed in is that of the dummy // activation window for windowless plugins. static bool IsDummyActivationWindow(gfx::NativeWindow window); // WebPluginDelegate implementation virtual bool Initialize(const GURL& url, const std::vector& arg_names, const std::vector& arg_values, webkit_glue::WebPlugin* plugin, bool load_manually); virtual void PluginDestroyed(); virtual void UpdateGeometry(const gfx::Rect& window_rect, const gfx::Rect& clip_rect); virtual void Paint(WebKit::WebCanvas* canvas, const gfx::Rect& rect); virtual void Print(gfx::NativeDrawingContext context); virtual void SetFocus(bool focused); virtual bool HandleInputEvent(const WebKit::WebInputEvent& event, WebKit::WebCursorInfo* cursor_info); virtual NPObject* GetPluginScriptableObject(); virtual void DidFinishLoadWithReason( const GURL& url, NPReason reason, int notify_id); virtual int GetProcessId(); virtual void SendJavaScriptStream(const GURL& url, const std::string& result, bool success, int notify_id); virtual void DidReceiveManualResponse(const GURL& url, const std::string& mime_type, const std::string& headers, uint32 expected_length, uint32 last_modified); virtual void DidReceiveManualData(const char* buffer, int length); virtual void DidFinishManualLoading(); virtual void DidManualLoadFail(); virtual void InstallMissingPlugin(); virtual webkit_glue::WebPluginResourceClient* CreateResourceClient( unsigned long resource_id, const GURL& url, int notify_id); virtual webkit_glue::WebPluginResourceClient* CreateSeekableResourceClient( unsigned long resource_id, int range_request_id); // End of WebPluginDelegate implementation. bool IsWindowless() const { return windowless_ ; } gfx::Rect GetRect() const { return window_rect_; } gfx::Rect GetClipRect() const { return clip_rect_; } // Returns the path for the library implementing this plugin. FilePath GetPluginPath(); // Returns a combination of PluginQuirks. int GetQuirks() const { return quirks_; } #if defined(OS_MACOSX) // Informs the plugin that the geometry has changed, as with UpdateGeometry, // but also includes the new buffer context for that new geometry. void UpdateGeometryAndContext(const gfx::Rect& window_rect, const gfx::Rect& clip_rect, gfx::NativeDrawingContext context); // Informs the delegate that the plugin called NPN_Invalidate*. Used as a // trigger for Core Animation drawing. void PluginDidInvalidate(); // Returns the delegate currently processing events. static WebPluginDelegateImpl* GetActiveDelegate(); // Informs the plugin that the window it is in has gained or lost focus. void SetWindowHasFocus(bool has_focus); // Informs the plugin that the view it is in has gained or lost first // responder status. void SetContentAreaHasFocus(bool has_focus); // Returns whether or not the window the plugin is in has focus. bool GetWindowHasFocus() const { return containing_window_has_focus_; } // Informs the plugin that its tab or window has been hidden or shown. void SetContainerVisibility(bool is_visible); // Informs the plugin that its containing window's frame has changed. // Frames are in screen coordinates. void WindowFrameChanged(gfx::Rect window_frame, gfx::Rect view_frame); // Informs the delegate that the plugin set a Carbon ThemeCursor. void SetThemeCursor(ThemeCursor cursor); // Informs the delegate that the plugin set a Carbon Cursor. void SetCursor(const Cursor* cursor); // Informs the delegate that the plugin set a Cocoa NSCursor. void SetNSCursor(NSCursor* cursor); #ifndef NP_NO_CARBON // Indicates that it's time to send the plugin a null event. void FireIdleEvent(); #endif #endif // OS_MACOSX gfx::PluginWindowHandle windowed_handle() const { return windowed_handle_; } #if defined(OS_MACOSX) // Allow setting a "fake" window handle to associate this plug-in with // an IOSurface in the browser. Used for accelerated drawing surfaces. void set_windowed_handle(gfx::PluginWindowHandle handle); #endif #if defined(USE_X11) void SetWindowlessShmPixmap(XID shm_pixmap) { windowless_shm_pixmap_ = shm_pixmap; } #endif private: friend class DeleteTask; friend class webkit_glue::WebPluginDelegate; WebPluginDelegateImpl(gfx::PluginWindowHandle containing_view, NPAPI::PluginInstance *instance); ~WebPluginDelegateImpl(); // Called by Initialize() for platform-specific initialization. // If this returns false, the plugin shouldn't be started--see Initialize(). bool PlatformInitialize(); // Called by DestroyInstance(), used for platform-specific destruction. void PlatformDestroyInstance(); //-------------------------- // used for windowed plugins void WindowedUpdateGeometry(const gfx::Rect& window_rect, const gfx::Rect& clip_rect); // Create the native window. // Returns true if the window is created (or already exists). // Returns false if unable to create the window. bool WindowedCreatePlugin(); // Destroy the native window. void WindowedDestroyWindow(); // Reposition the native window to be in sync with the given geometry. // Returns true if the native window has moved or been clipped differently. bool WindowedReposition(const gfx::Rect& window_rect, const gfx::Rect& clip_rect); // Tells the plugin about the current state of the window. // See NPAPI NPP_SetWindow for more information. void WindowedSetWindow(); #if defined(OS_WIN) // Registers the window class for our window ATOM RegisterNativeWindowClass(); // Our WndProc functions. static LRESULT CALLBACK DummyWindowProc( HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam); static LRESULT CALLBACK NativeWndProc( HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam); static LRESULT CALLBACK FlashWindowlessWndProc( HWND hwnd, UINT message, WPARAM wparam, LPARAM lparam); // Used for throttling Flash messages. static void ClearThrottleQueueForWindow(HWND window); static void OnThrottleMessage(); static void ThrottleMessage(WNDPROC proc, HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam); #endif //---------------------------- // used for windowless plugins void WindowlessUpdateGeometry(const gfx::Rect& window_rect, const gfx::Rect& clip_rect); void WindowlessPaint(gfx::NativeDrawingContext hdc, const gfx::Rect& rect); // Tells the plugin about the current state of the window. // See NPAPI NPP_SetWindow for more information. void WindowlessSetWindow(); //----------------------------------------- // used for windowed and windowless plugins // Does platform-specific event handling. Arguments and return are identical // to HandleInputEvent. bool PlatformHandleInputEvent(const WebKit::WebInputEvent& event, WebKit::WebCursorInfo* cursor_info); NPAPI::PluginInstance* instance() { return instance_.get(); } // Closes down and destroys our plugin instance. void DestroyInstance(); // used for windowed plugins // Note: on Mac OS X, the only time the windowed handle is non-zero // is the case of accelerated rendering, which uses a fake window handle to // identify itself back to the browser. It still performs all of its // work offscreen. gfx::PluginWindowHandle windowed_handle_; gfx::Rect windowed_last_pos_; bool windowed_did_set_window_; // used by windowed and windowless plugins bool windowless_; webkit_glue::WebPlugin* plugin_; scoped_refptr instance_; #if defined(OS_WIN) // Original wndproc before we subclassed. WNDPROC plugin_wnd_proc_; // Used to throttle WM_USER+1 messages in Flash. uint32 last_message_; bool is_calling_wndproc; // The current keyboard layout of this process and the main thread ID of the // browser process. These variables are used for synchronizing the keyboard // layout of this process with the one of the browser process. HKL keyboard_layout_; int parent_thread_id_; #endif // defined(OS_WIN) #if defined(USE_X11) // The SHM pixmap for a windowless plugin. XID windowless_shm_pixmap_; // The pixmap we're drawing into, for a windowless plugin. GdkPixmap* pixmap_; double first_event_time_; // On Linux some plugins assume that the GtkSocket container is in the same // process. So we create a GtkPlug to plug into the browser's container, and // a GtkSocket to hold the plugin. We then send the GtkPlug to the browser // process. GtkWidget* plug_; GtkWidget* socket_; // Ensure pixmap_ exists and is at least width by height pixels. void EnsurePixmapAtLeastSize(int width, int height); #endif gfx::PluginWindowHandle parent_; NPWindow window_; gfx::Rect window_rect_; gfx::Rect clip_rect_; int quirks_; #if defined(OS_WIN) // Windowless plugins don't have keyboard focus causing issues with the // plugin not receiving keyboard events if the plugin enters a modal // loop like TrackPopupMenuEx or MessageBox, etc. // This is a basic issue with windows activation and focus arising due to // the fact that these windows are created by different threads. Activation // and focus are thread specific states, and if the browser has focus, // the plugin may not have focus. // To fix a majority of these activation issues we create a dummy visible // child window to which we set focus whenever the windowless plugin // receives a WM_LBUTTONDOWN/WM_RBUTTONDOWN message via NPP_HandleEvent. HWND dummy_window_for_activation_; bool CreateDummyWindowForActivation(); // Returns true if the event passed in needs to be tracked for a potential // modal loop. static bool ShouldTrackEventForModalLoops(NPEvent* event); // The message filter hook procedure, which tracks modal loops entered by // a plugin in the course of a NPP_HandleEvent call. static LRESULT CALLBACK HandleEventMessageFilterHook(int code, WPARAM wParam, LPARAM lParam); // TrackPopupMenu interceptor. Parameters are the same as the Win32 function // TrackPopupMenu. static BOOL WINAPI TrackPopupMenuPatch(HMENU menu, unsigned int flags, int x, int y, int reserved, HWND window, const RECT* rect); // SetCursor interceptor for windowless plugins. static HCURSOR WINAPI SetCursorPatch(HCURSOR cursor); // RegEnumKeyExW interceptor. static LONG WINAPI RegEnumKeyExWPatch( HKEY key, DWORD index, LPWSTR name, LPDWORD name_size, LPDWORD reserved, LPWSTR class_name, LPDWORD class_size, PFILETIME last_write_time); // The mouse hook proc which handles mouse capture in windowed plugins. static LRESULT CALLBACK MouseHookProc(int code, WPARAM wParam, LPARAM lParam); // Calls SetCapture/ReleaseCapture based on the message type. static void HandleCaptureForMessage(HWND window, UINT message); #elif defined(OS_MACOSX) // Sets window_rect_ to |rect| void SetPluginRect(const gfx::Rect& rect); // Sets content_area_origin to |origin| void SetContentAreaOrigin(const gfx::Point& origin); // Updates everything that depends on the plugin's absolute screen location. void PluginScreenLocationChanged(); // Informs the plugin that it has gained or lost keyboard focus (i.e., window // first responder status). void SetPluginHasFocus(bool has_focus); // Returns the apparent zoom ratio for the given event, as inferred from our // current knowledge about about where on screen the plugin is. // This is a temporary workaround for ; once that is // fixed we should have correct event coordinates (or an explicit // notification of zoom level). float ApparentEventZoomLevel(const WebKit::WebMouseEvent& event); // Informs the browser about the updated accelerated drawing surface. void UpdateAcceleratedSurface(); // Updates anything that depends on plugin visibility. void PluginVisibilityChanged(); // Uses a CARenderer to draw the plug-in's layer in our OpenGL surface. void DrawLayerInSurface(); #ifndef NP_NO_CARBON // Moves our dummy window to match the current screen location of the plugin. void UpdateDummyWindowBounds(const gfx::Point& plugin_origin); #ifndef NP_NO_QUICKDRAW // Sets the mode used for QuickDraw plugin drawing. If enabled is true the // plugin draws into a GWorld that's not connected to a window (the faster // path), otherwise the plugin draws into our invisible dummy window (which is // slower, since the call we use to scrape the window contents is much more // expensive than copying between GWorlds). void SetQuickDrawFastPathEnabled(bool enabled); #endif // Adjusts the idle event rate for a Carbon plugin based on its current // visibility. void UpdateIdleEventRate(); #endif // !NP_NO_CARBON CGContextRef buffer_context_; // Weak ref. #ifndef NP_NO_CARBON NP_CGContext np_cg_context_; #endif #ifndef NP_NO_QUICKDRAW NP_Port qd_port_; scoped_ptr qd_manager_; base::TimeTicks fast_path_enable_tick_; #endif CALayer* layer_; // Used for CA drawing mode. Weak, retained by plug-in. AcceleratedSurface* surface_; CARenderer* renderer_; // Renders layer_ to surface_. scoped_ptr > redraw_timer_; // The upper-left corner of the web content area in screen coordinates, // relative to an upper-left (0,0). gfx::Point content_area_origin_; // True if the plugin thinks it has keyboard focus bool plugin_has_focus_; // True if the plugin element has focus within the page, regardless of whether // its containing view is currently the first responder for the window. bool has_webkit_focus_; // True if the containing view is the window's first responder. bool containing_view_has_focus_; bool containing_window_has_focus_; bool initial_window_focus_; bool container_is_visible_; bool have_called_set_window_; gfx::Rect cached_clip_rect_; scoped_ptr external_drag_tracker_; #endif // OS_MACOSX // Called by the message filter hook when the plugin enters a modal loop. void OnModalLoopEntered(); // Returns true if the message passed in corresponds to a user gesture. static bool IsUserGesture(const WebKit::WebInputEvent& event); // The url with which the plugin was instantiated. std::string plugin_url_; #if defined(OS_WIN) // Indicates the end of a user gesture period. void OnUserGestureEnd(); // Handle to the message filter hook HHOOK handle_event_message_filter_hook_; // Event which is set when the plugin enters a modal loop in the course // of a NPP_HandleEvent call. HANDLE handle_event_pump_messages_event_; // This flag indicates whether we started tracking a user gesture message. bool user_gesture_message_posted_; // Runnable Method Factory used to invoke the OnUserGestureEnd method // asynchronously. ScopedRunnableMethodFactory user_gesture_msg_factory_; // Handle to the mouse hook installed for certain windowed plugins like // flash. HHOOK mouse_hook_; #endif // Holds the depth of the HandleEvent callstack. int handle_event_depth_; // Holds the current cursor set by the windowless plugin. WebCursor current_windowless_cursor_; // Set to true initially and indicates if this is the first npp_setwindow // call received by the plugin. bool first_set_window_call_; DISALLOW_COPY_AND_ASSIGN(WebPluginDelegateImpl); }; #endif // WEBKIT_GLUE_PLUGINS_WEBPLUGIN_DELEGATE_IMPL_H_