// Copyright (c) 2012 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. #include "content/renderer/render_widget_fullscreen_pepper.h" #include #include "base/bind.h" #include "base/command_line.h" #include "base/message_loop/message_loop.h" #include "content/common/gpu/client/gpu_channel_host.h" #include "content/common/view_messages.h" #include "content/public/common/content_switches.h" #include "content/renderer/gpu/render_widget_compositor.h" #include "content/renderer/pepper/pepper_platform_context_3d.h" #include "content/renderer/pepper/pepper_plugin_instance_impl.h" #include "content/renderer/render_thread_impl.h" #include "gpu/command_buffer/client/gles2_implementation.h" #include "skia/ext/platform_canvas.h" #include "third_party/WebKit/public/platform/WebCanvas.h" #include "third_party/WebKit/public/platform/WebGraphicsContext3D.h" #include "third_party/WebKit/public/platform/WebLayer.h" #include "third_party/WebKit/public/platform/WebSize.h" #include "third_party/WebKit/public/web/WebCursorInfo.h" #include "third_party/WebKit/public/web/WebWidget.h" #include "ui/gfx/size_conversions.h" #include "ui/gl/gpu_preference.h" using WebKit::WebCanvas; using WebKit::WebCompositionUnderline; using WebKit::WebCursorInfo; using WebKit::WebGestureEvent; using WebKit::WebInputEvent; using WebKit::WebMouseEvent; using WebKit::WebMouseWheelEvent; using WebKit::WebPoint; using WebKit::WebRect; using WebKit::WebSize; using WebKit::WebString; using WebKit::WebTextDirection; using WebKit::WebTextInputType; using WebKit::WebVector; using WebKit::WebWidget; using WebKit::WGC3Dintptr; namespace content { namespace { // See third_party/WebKit/Source/WebCore/dom/WheelEvent.h. const float kTickDivisor = 120.0f; class FullscreenMouseLockDispatcher : public MouseLockDispatcher { public: explicit FullscreenMouseLockDispatcher(RenderWidgetFullscreenPepper* widget); virtual ~FullscreenMouseLockDispatcher(); private: // MouseLockDispatcher implementation. virtual void SendLockMouseRequest(bool unlocked_by_target) OVERRIDE; virtual void SendUnlockMouseRequest() OVERRIDE; RenderWidgetFullscreenPepper* widget_; DISALLOW_COPY_AND_ASSIGN(FullscreenMouseLockDispatcher); }; WebMouseEvent WebMouseEventFromGestureEvent(const WebGestureEvent& gesture) { WebMouseEvent mouse; switch (gesture.type) { case WebInputEvent::GestureScrollBegin: mouse.type = WebInputEvent::MouseDown; break; case WebInputEvent::GestureScrollUpdate: mouse.type = WebInputEvent::MouseMove; break; case WebInputEvent::GestureFlingStart: if (gesture.sourceDevice == WebGestureEvent::Touchscreen) { // A scroll gesture on the touchscreen may end with a GestureScrollEnd // when there is no velocity, or a GestureFlingStart when it has a // velocity. In both cases, it should end the drag that was initiated by // the GestureScrollBegin (and subsequent GestureScrollUpdate) events. mouse.type = WebInputEvent::MouseUp; break; } else { return mouse; } case WebInputEvent::GestureScrollEnd: mouse.type = WebInputEvent::MouseUp; break; default: break; } if (mouse.type == WebInputEvent::Undefined) return mouse; mouse.timeStampSeconds = gesture.timeStampSeconds; mouse.modifiers = gesture.modifiers | WebInputEvent::LeftButtonDown; mouse.button = WebMouseEvent::ButtonLeft; mouse.clickCount = (mouse.type == WebInputEvent::MouseDown || mouse.type == WebInputEvent::MouseUp); mouse.x = gesture.x; mouse.y = gesture.y; mouse.windowX = gesture.globalX; mouse.windowY = gesture.globalY; mouse.globalX = gesture.globalX; mouse.globalY = gesture.globalY; return mouse; } FullscreenMouseLockDispatcher::FullscreenMouseLockDispatcher( RenderWidgetFullscreenPepper* widget) : widget_(widget) { } FullscreenMouseLockDispatcher::~FullscreenMouseLockDispatcher() { } void FullscreenMouseLockDispatcher::SendLockMouseRequest( bool unlocked_by_target) { widget_->Send(new ViewHostMsg_LockMouse(widget_->routing_id(), false, unlocked_by_target, true)); } void FullscreenMouseLockDispatcher::SendUnlockMouseRequest() { widget_->Send(new ViewHostMsg_UnlockMouse(widget_->routing_id())); } // WebWidget that simply wraps the pepper plugin. class PepperWidget : public WebWidget { public: explicit PepperWidget(RenderWidgetFullscreenPepper* widget) : widget_(widget) { } virtual ~PepperWidget() {} // WebWidget API virtual void close() { delete this; } virtual WebSize size() { return size_; } virtual void willStartLiveResize() { } virtual void resize(const WebSize& size) { if (!widget_->plugin()) return; size_ = size; WebRect plugin_rect(0, 0, size_.width, size_.height); widget_->plugin()->ViewChanged(plugin_rect, plugin_rect, std::vector()); widget_->Invalidate(); } virtual void willEndLiveResize() { } virtual void animate(double frameBeginTime) { } virtual void layout() { } virtual void paint(WebCanvas* canvas, const WebRect& rect, PaintOptions) { if (!widget_->plugin()) return; SkAutoCanvasRestore auto_restore(canvas, true); float canvas_scale = widget_->deviceScaleFactor(); canvas->scale(canvas_scale, canvas_scale); WebRect plugin_rect(0, 0, size_.width, size_.height); widget_->plugin()->Paint(canvas, plugin_rect, rect); } virtual void setCompositorSurfaceReady() { } virtual void composite(bool finish) { } virtual void themeChanged() { NOTIMPLEMENTED(); } virtual bool handleInputEvent(const WebInputEvent& event) { if (!widget_->plugin()) return false; // This cursor info is ignored, we always set the cursor directly from // RenderWidgetFullscreenPepper::DidChangeCursor. WebCursorInfo cursor; // Pepper plugins do not accept gesture events. So do not send the gesture // events directly to the plugin. Instead, try to convert them to equivalent // mouse events, and then send to the plugin. if (WebInputEvent::isGestureEventType(event.type)) { bool result = false; const WebGestureEvent* gesture_event = static_cast(&event); switch (event.type) { case WebInputEvent::GestureTap: { WebMouseEvent mouse; mouse.timeStampSeconds = gesture_event->timeStampSeconds; mouse.type = WebInputEvent::MouseMove; mouse.modifiers = gesture_event->modifiers; mouse.x = gesture_event->x; mouse.y = gesture_event->y; mouse.windowX = gesture_event->globalX; mouse.windowY = gesture_event->globalY; mouse.globalX = gesture_event->globalX; mouse.globalY = gesture_event->globalY; mouse.movementX = 0; mouse.movementY = 0; result |= widget_->plugin()->HandleInputEvent(mouse, &cursor); mouse.type = WebInputEvent::MouseDown; mouse.button = WebMouseEvent::ButtonLeft; mouse.clickCount = gesture_event->data.tap.tapCount; result |= widget_->plugin()->HandleInputEvent(mouse, &cursor); mouse.type = WebInputEvent::MouseUp; result |= widget_->plugin()->HandleInputEvent(mouse, &cursor); break; } default: { WebMouseEvent mouse = WebMouseEventFromGestureEvent(*gesture_event); if (mouse.type != WebInputEvent::Undefined) result |= widget_->plugin()->HandleInputEvent(mouse, &cursor); break; } } return result; } bool result = widget_->plugin()->HandleInputEvent(event, &cursor); // For normal web pages, WebViewImpl does input event translations and // generates context menu events. Since we don't have a WebView, we need to // do the necessary translation ourselves. if (WebInputEvent::isMouseEventType(event.type)) { const WebMouseEvent& mouse_event = reinterpret_cast(event); bool send_context_menu_event = false; // On Mac/Linux, we handle it on mouse down. // On Windows, we handle it on mouse up. #if defined(OS_WIN) send_context_menu_event = mouse_event.type == WebInputEvent::MouseUp && mouse_event.button == WebMouseEvent::ButtonRight; #elif defined(OS_MACOSX) send_context_menu_event = mouse_event.type == WebInputEvent::MouseDown && (mouse_event.button == WebMouseEvent::ButtonRight || (mouse_event.button == WebMouseEvent::ButtonLeft && mouse_event.modifiers & WebMouseEvent::ControlKey)); #else send_context_menu_event = mouse_event.type == WebInputEvent::MouseDown && mouse_event.button == WebMouseEvent::ButtonRight; #endif if (send_context_menu_event) { WebMouseEvent context_menu_event(mouse_event); context_menu_event.type = WebInputEvent::ContextMenu; widget_->plugin()->HandleInputEvent(context_menu_event, &cursor); } } return result; } virtual void mouseCaptureLost() { } virtual void setFocus(bool focus) { } // TODO(piman): figure out IME and implement these if necessary. virtual bool setComposition( const WebString& text, const WebVector& underlines, int selectionStart, int selectionEnd) { return false; } virtual bool confirmComposition() { return false; } virtual bool compositionRange(size_t* location, size_t* length) { return false; } virtual bool confirmComposition(const WebString& text) { return false; } virtual WebTextInputType textInputType() { return WebKit::WebTextInputTypeNone; } virtual WebRect caretOrSelectionBounds() { return WebRect(); } virtual bool selectionRange(WebPoint& start, WebPoint& end) const { return false; } virtual bool caretOrSelectionRange(size_t* location, size_t* length) { return false; } virtual void setTextDirection(WebTextDirection) { } virtual bool isAcceleratedCompositingActive() const { return widget_->plugin() && widget_->is_compositing(); } private: RenderWidgetFullscreenPepper* widget_; WebSize size_; DISALLOW_COPY_AND_ASSIGN(PepperWidget); }; } // anonymous namespace // static RenderWidgetFullscreenPepper* RenderWidgetFullscreenPepper::Create( int32 opener_id, PepperPluginInstanceImpl* plugin, const GURL& active_url, const WebKit::WebScreenInfo& screen_info) { DCHECK_NE(MSG_ROUTING_NONE, opener_id); scoped_refptr widget( new RenderWidgetFullscreenPepper(plugin, active_url, screen_info)); widget->Init(opener_id); widget->AddRef(); return widget.get(); } RenderWidgetFullscreenPepper::RenderWidgetFullscreenPepper( PepperPluginInstanceImpl* plugin, const GURL& active_url, const WebKit::WebScreenInfo& screen_info) : RenderWidgetFullscreen(screen_info), active_url_(active_url), plugin_(plugin), layer_(NULL), mouse_lock_dispatcher_(new FullscreenMouseLockDispatcher( this)) { } RenderWidgetFullscreenPepper::~RenderWidgetFullscreenPepper() { } void RenderWidgetFullscreenPepper::Invalidate() { InvalidateRect(gfx::Rect(size_.width(), size_.height())); } void RenderWidgetFullscreenPepper::InvalidateRect(const WebKit::WebRect& rect) { didInvalidateRect(rect); } void RenderWidgetFullscreenPepper::ScrollRect( int dx, int dy, const WebKit::WebRect& rect) { didScrollRect(dx, dy, rect); } void RenderWidgetFullscreenPepper::Destroy() { // This function is called by the plugin instance as it's going away, so reset // plugin_ to NULL to avoid calling into a dangling pointer e.g. on Close(). plugin_ = NULL; // After calling Destroy(), the plugin instance assumes that the layer is not // used by us anymore, so it may destroy the layer before this object goes // away. SetLayer(NULL); Send(new ViewHostMsg_Close(routing_id_)); Release(); } void RenderWidgetFullscreenPepper::DidChangeCursor( const WebKit::WebCursorInfo& cursor) { didChangeCursor(cursor); } void RenderWidgetFullscreenPepper::SetLayer(WebKit::WebLayer* layer) { layer_ = layer; bool compositing = !!layer_; if (compositing != is_accelerated_compositing_active_) { if (compositing) { initializeLayerTreeView(); if (!layerTreeView()) return; layer_->setBounds(WebKit::WebSize(size())); layer_->setDrawsContent(true); compositor_->setDeviceScaleFactor(device_scale_factor_); compositor_->setRootLayer(*layer_); didActivateCompositor(-1); } else { didDeactivateCompositor(); } } } bool RenderWidgetFullscreenPepper::OnMessageReceived(const IPC::Message& msg) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(RenderWidgetFullscreenPepper, msg) IPC_MESSAGE_FORWARD(ViewMsg_LockMouse_ACK, mouse_lock_dispatcher_.get(), MouseLockDispatcher::OnLockMouseACK) IPC_MESSAGE_FORWARD(ViewMsg_MouseLockLost, mouse_lock_dispatcher_.get(), MouseLockDispatcher::OnMouseLockLost) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() if (handled) return true; return RenderWidgetFullscreen::OnMessageReceived(msg); } void RenderWidgetFullscreenPepper::WillInitiatePaint() { if (plugin_) plugin_->ViewWillInitiatePaint(); } void RenderWidgetFullscreenPepper::DidInitiatePaint() { if (plugin_) plugin_->ViewInitiatedPaint(); } void RenderWidgetFullscreenPepper::DidFlushPaint() { if (plugin_) plugin_->ViewFlushedPaint(); } void RenderWidgetFullscreenPepper::Close() { // If the fullscreen window is closed (e.g. user pressed escape), reset to // normal mode. if (plugin_) plugin_->FlashSetFullscreen(false, false); // Call Close on the base class to destroy the WebWidget instance. RenderWidget::Close(); } PepperPluginInstanceImpl* RenderWidgetFullscreenPepper::GetBitmapForOptimizedPluginPaint( const gfx::Rect& paint_bounds, TransportDIB** dib, gfx::Rect* location, gfx::Rect* clip, float* scale_factor) { if (plugin_ && plugin_->GetBitmapForOptimizedPluginPaint( paint_bounds, dib, location, clip, scale_factor)) { return plugin_; } return NULL; } void RenderWidgetFullscreenPepper::OnResize( const ViewMsg_Resize_Params& params) { if (layer_) layer_->setBounds(WebKit::WebSize(params.new_size)); RenderWidget::OnResize(params); } WebWidget* RenderWidgetFullscreenPepper::CreateWebWidget() { return new PepperWidget(this); } GURL RenderWidgetFullscreenPepper::GetURLForGraphicsContext3D() { return active_url_; } void RenderWidgetFullscreenPepper::SetDeviceScaleFactor( float device_scale_factor) { RenderWidget::SetDeviceScaleFactor(device_scale_factor); if (compositor_) compositor_->setDeviceScaleFactor(device_scale_factor); } } // namespace content