461 lines
17 KiB
C++
461 lines
17 KiB
C++
|
// Copyright (c) 2006-2008 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 "libcef/webwidget_host.h"
|
||
|
|
||
|
#include <cairo/cairo.h>
|
||
|
#include <gdk/gdkx.h>
|
||
|
#include <gtk/gtk.h>
|
||
|
|
||
|
#include "base/basictypes.h"
|
||
|
#include "base/logging.h"
|
||
|
#include "skia/ext/bitmap_platform_device.h"
|
||
|
#include "skia/ext/platform_canvas.h"
|
||
|
#include "skia/ext/platform_device.h"
|
||
|
#include "third_party/WebKit/Source/WebKit/chromium/public/gtk/WebInputEventFactory.h"
|
||
|
#include "third_party/WebKit/Source/WebKit/chromium/public/x11/WebScreenInfoFactory.h"
|
||
|
#include "third_party/WebKit/Source/WebKit/chromium/public/WebInputEvent.h"
|
||
|
#include "third_party/WebKit/Source/WebKit/chromium/public/WebPopupMenu.h"
|
||
|
#include "third_party/WebKit/Source/WebKit/chromium/public/WebScreenInfo.h"
|
||
|
#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebSize.h"
|
||
|
|
||
|
using WebKit::WebInputEventFactory;
|
||
|
using WebKit::WebKeyboardEvent;
|
||
|
using WebKit::WebMouseEvent;
|
||
|
using WebKit::WebMouseWheelEvent;
|
||
|
using WebKit::WebPopupMenu;
|
||
|
using WebKit::WebScreenInfo;
|
||
|
using WebKit::WebScreenInfoFactory;
|
||
|
using WebKit::WebSize;
|
||
|
using WebKit::WebWidgetClient;
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
// Used to store a backpointer to WebWidgetHost from our GtkWidget.
|
||
|
const char kWebWidgetHostKey[] = "webwidgethost";
|
||
|
|
||
|
// In response to an invalidation, we call into WebKit to do layout. On
|
||
|
// Windows, WM_PAINT is a virtual message so any extra invalidates that come up
|
||
|
// while it's doing layout are implicitly swallowed as soon as we actually do
|
||
|
// drawing via BeginPaint.
|
||
|
//
|
||
|
// Though GTK does know how to collapse multiple paint requests, it won't erase
|
||
|
// paint requests from the future when we start drawing. To avoid an infinite
|
||
|
// cycle of repaints, we track whether we're currently handling a redraw, and
|
||
|
// during that if we get told by WebKit that a region has become invalid, we
|
||
|
// still add that region to the local dirty rect but *don't* enqueue yet
|
||
|
// another "do a paint" message.
|
||
|
bool g_handling_expose = false;
|
||
|
|
||
|
// -----------------------------------------------------------------------------
|
||
|
// Callback functions to proxy to host...
|
||
|
|
||
|
// The web contents are completely drawn and handled by WebKit, except that
|
||
|
// windowed plugins are GtkSockets on top of it. We need to place the
|
||
|
// GtkSockets inside a GtkContainer. We use a GtkFixed container, and the
|
||
|
// GtkSocket objects override a little bit to manage their size (see the code
|
||
|
// in webplugin_delegate_impl_gtk.cc). We listen on a the events we're
|
||
|
// interested in and forward them on to the WebWidgetHost. This class is a
|
||
|
// collection of static methods, implementing the widget related code.
|
||
|
class WebWidgetHostGtkWidget {
|
||
|
public:
|
||
|
// This will create a new widget used for hosting the web contents. We use
|
||
|
// our GtkDrawingAreaContainer here, for the reasons mentioned above.
|
||
|
static GtkWidget* CreateNewWidget(GtkWidget* parent_view,
|
||
|
WebWidgetHost* host) {
|
||
|
GtkWidget* widget = gtk_fixed_new();
|
||
|
gtk_fixed_set_has_window(GTK_FIXED(widget), true);
|
||
|
|
||
|
gtk_box_pack_start(GTK_BOX(parent_view), widget, TRUE, TRUE, 0);
|
||
|
|
||
|
gtk_widget_add_events(widget, GDK_EXPOSURE_MASK |
|
||
|
GDK_POINTER_MOTION_MASK |
|
||
|
GDK_BUTTON_PRESS_MASK |
|
||
|
GDK_BUTTON_RELEASE_MASK |
|
||
|
GDK_KEY_PRESS_MASK |
|
||
|
GDK_KEY_RELEASE_MASK);
|
||
|
gtk_widget_set_can_focus(widget, TRUE);
|
||
|
g_signal_connect(widget, "size-request",
|
||
|
G_CALLBACK(&HandleSizeRequest), host);
|
||
|
g_signal_connect(widget, "size-allocate",
|
||
|
G_CALLBACK(&HandleSizeAllocate), host);
|
||
|
g_signal_connect(widget, "configure-event",
|
||
|
G_CALLBACK(&HandleConfigure), host);
|
||
|
g_signal_connect(widget, "expose-event",
|
||
|
G_CALLBACK(&HandleExpose), host);
|
||
|
g_signal_connect(widget, "key-press-event",
|
||
|
G_CALLBACK(&HandleKeyPress), host);
|
||
|
g_signal_connect(widget, "key-release-event",
|
||
|
G_CALLBACK(&HandleKeyRelease), host);
|
||
|
g_signal_connect(widget, "focus",
|
||
|
G_CALLBACK(&HandleFocus), host);
|
||
|
g_signal_connect(widget, "focus-in-event",
|
||
|
G_CALLBACK(&HandleFocusIn), host);
|
||
|
g_signal_connect(widget, "focus-out-event",
|
||
|
G_CALLBACK(&HandleFocusOut), host);
|
||
|
g_signal_connect(widget, "button-press-event",
|
||
|
G_CALLBACK(&HandleButtonPress), host);
|
||
|
g_signal_connect(widget, "button-release-event",
|
||
|
G_CALLBACK(&HandleButtonRelease), host);
|
||
|
g_signal_connect(widget, "motion-notify-event",
|
||
|
G_CALLBACK(&HandleMotionNotify), host);
|
||
|
g_signal_connect(widget, "scroll-event",
|
||
|
G_CALLBACK(&HandleScroll), host);
|
||
|
|
||
|
g_object_set_data(G_OBJECT(widget), kWebWidgetHostKey, host);
|
||
|
return widget;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
// Our size was requested. We let the GtkFixed do its normal calculation,
|
||
|
// after which this callback is called. The GtkFixed will come up with a
|
||
|
// requisition based on its children, which include plugin windows. Since
|
||
|
// we don't want to prevent resizing smaller than a plugin window, we need to
|
||
|
// control the size ourself.
|
||
|
static void HandleSizeRequest(GtkWidget* widget,
|
||
|
GtkRequisition* req,
|
||
|
WebWidgetHost* host) {
|
||
|
// This is arbitrary, but the WebKit scrollbars try to shrink themselves
|
||
|
// if the browser window is too small. Give them some space.
|
||
|
static const int kMinWidthHeight = 64;
|
||
|
|
||
|
req->width = kMinWidthHeight;
|
||
|
req->height = kMinWidthHeight;
|
||
|
}
|
||
|
|
||
|
// Our size has changed.
|
||
|
static void HandleSizeAllocate(GtkWidget* widget,
|
||
|
GtkAllocation* allocation,
|
||
|
WebWidgetHost* host) {
|
||
|
host->SetSize(allocation->width, allocation->height);
|
||
|
}
|
||
|
|
||
|
// Size, position, or stacking of the GdkWindow changed.
|
||
|
static gboolean HandleConfigure(GtkWidget* widget,
|
||
|
GdkEventConfigure* config,
|
||
|
WebWidgetHost* host) {
|
||
|
host->SetSize(config->width, config->height);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
// A portion of the GdkWindow needs to be redraw.
|
||
|
static gboolean HandleExpose(GtkWidget* widget,
|
||
|
GdkEventExpose* expose,
|
||
|
WebWidgetHost* host) {
|
||
|
// See comments above about what g_handling_expose is for.
|
||
|
g_handling_expose = true;
|
||
|
gfx::Rect rect(expose->area);
|
||
|
host->InvalidateRect(rect);
|
||
|
host->Paint();
|
||
|
g_handling_expose = false;
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
// Keyboard key pressed.
|
||
|
static gboolean HandleKeyPress(GtkWidget* widget,
|
||
|
GdkEventKey* event,
|
||
|
WebWidgetHost* host) {
|
||
|
host->KeyEvent(event);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
// Keyboard key released.
|
||
|
static gboolean HandleKeyRelease(GtkWidget* widget,
|
||
|
GdkEventKey* event,
|
||
|
WebWidgetHost* host) {
|
||
|
return HandleKeyPress(widget, event, host);
|
||
|
}
|
||
|
|
||
|
// This signal is called when arrow keys or tab is pressed. If we return
|
||
|
// true, we prevent focus from being moved to another widget. If we want to
|
||
|
// allow focus to be moved outside of web contents, we need to implement
|
||
|
// WebViewDelegate::TakeFocus in the test webview delegate.
|
||
|
static gboolean HandleFocus(GtkWidget* widget,
|
||
|
GdkEventFocus* focus,
|
||
|
WebWidgetHost* host) {
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
// Keyboard focus entered.
|
||
|
static gboolean HandleFocusIn(GtkWidget* widget,
|
||
|
GdkEventFocus* focus,
|
||
|
WebWidgetHost* host) {
|
||
|
host->webwidget()->setFocus(true);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
// Keyboard focus left.
|
||
|
static gboolean HandleFocusOut(GtkWidget* widget,
|
||
|
GdkEventFocus* focus,
|
||
|
WebWidgetHost* host) {
|
||
|
host->webwidget()->setFocus(false);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
// Mouse button down.
|
||
|
static gboolean HandleButtonPress(GtkWidget* widget,
|
||
|
GdkEventButton* event,
|
||
|
WebWidgetHost* host) {
|
||
|
if (!(event->button == 1 || event->button == 2 || event->button == 3))
|
||
|
return FALSE; // We do not forward any other buttons to the renderer.
|
||
|
if (event->type == GDK_2BUTTON_PRESS || event->type == GDK_3BUTTON_PRESS)
|
||
|
return FALSE;
|
||
|
|
||
|
gtk_widget_grab_focus(widget);
|
||
|
|
||
|
host->webwidget()->handleInputEvent(
|
||
|
WebInputEventFactory::mouseEvent(event));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
// Mouse button up.
|
||
|
static gboolean HandleButtonRelease(GtkWidget* widget,
|
||
|
GdkEventButton* event,
|
||
|
WebWidgetHost* host) {
|
||
|
return HandleButtonPress(widget, event, host);
|
||
|
}
|
||
|
|
||
|
// Mouse pointer movements.
|
||
|
static gboolean HandleMotionNotify(GtkWidget* widget,
|
||
|
GdkEventMotion* event,
|
||
|
WebWidgetHost* host) {
|
||
|
host->webwidget()->handleInputEvent(
|
||
|
WebInputEventFactory::mouseEvent(event));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
// Mouse scroll wheel.
|
||
|
static gboolean HandleScroll(GtkWidget* widget,
|
||
|
GdkEventScroll* event,
|
||
|
WebWidgetHost* host) {
|
||
|
host->webwidget()->handleInputEvent(
|
||
|
WebInputEventFactory::mouseWheelEvent(event));
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
DISALLOW_IMPLICIT_CONSTRUCTORS(WebWidgetHostGtkWidget);
|
||
|
};
|
||
|
|
||
|
// Return an Xlib Display pointer for the given widget.
|
||
|
Display* GtkWidgetGetDisplay(GtkWidget* widget) {
|
||
|
GdkDisplay* gdk_display = gtk_widget_get_display(widget);
|
||
|
return gdk_x11_display_get_xdisplay(gdk_display);
|
||
|
}
|
||
|
|
||
|
// Return the screen number for the given widget.
|
||
|
int GtkWidgetGetScreenNum(GtkWidget* widget) {
|
||
|
GdkDisplay* gdk_display = gtk_widget_get_display(widget);
|
||
|
GdkScreen* gdk_screen = gdk_display_get_default_screen(gdk_display);
|
||
|
return gdk_x11_screen_get_screen_number(gdk_screen);
|
||
|
}
|
||
|
|
||
|
} // namespace
|
||
|
|
||
|
// This is provided so that the webview can reuse the custom GTK window code.
|
||
|
// static
|
||
|
gfx::NativeView WebWidgetHost::CreateWidget(
|
||
|
gfx::NativeView parent_view, WebWidgetHost* host) {
|
||
|
return WebWidgetHostGtkWidget::CreateNewWidget(parent_view, host);
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
WebWidgetHost* WebWidgetHost::Create(GtkWidget* parent_view,
|
||
|
WebWidgetClient* client,
|
||
|
PaintDelegate* paint_delegate) {
|
||
|
WebWidgetHost* host = new WebWidgetHost();
|
||
|
host->view_ = CreateWidget(parent_view, host);
|
||
|
host->webwidget_ = WebPopupMenu::create(client);
|
||
|
// We manage our own double buffering because we need to be able to update
|
||
|
// the expose area in an ExposeEvent within the lifetime of the event handler.
|
||
|
gtk_widget_set_double_buffered(GTK_WIDGET(host->view_), false);
|
||
|
|
||
|
return host;
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::ScrollRect(int dx, int dy, const gfx::Rect& clip_rect) {
|
||
|
// This is used for optimizing painting when the renderer is scrolled. We're
|
||
|
// currently not doing any optimizations so just invalidate the region.
|
||
|
InvalidateRect(clip_rect);
|
||
|
}
|
||
|
|
||
|
WebWidgetHost::WebWidgetHost()
|
||
|
: view_(NULL),
|
||
|
paint_delegate_(NULL),
|
||
|
webwidget_(NULL),
|
||
|
canvas_w_(0),
|
||
|
canvas_h_(0),
|
||
|
popup_(false),
|
||
|
timer_executing_(false),
|
||
|
timer_wanted_(false),
|
||
|
frame_delay_(1000 / kDefaultFrameRate) {
|
||
|
set_painting(false);
|
||
|
}
|
||
|
|
||
|
WebWidgetHost::~WebWidgetHost() {
|
||
|
// We may be deleted before the view_. Clear out the signals so that we don't
|
||
|
// attempt to invoke something on a deleted object.
|
||
|
g_object_set_data(G_OBJECT(view_), kWebWidgetHostKey, NULL);
|
||
|
g_signal_handlers_disconnect_matched(view_,
|
||
|
G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, this);
|
||
|
// TODO(port): Figure out why we can't do this without crashing.
|
||
|
// webwidget_->close();
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::InvalidateWindow() {
|
||
|
int width, height;
|
||
|
GetSize(width, height);
|
||
|
const gfx::Rect client_rect(width, height);
|
||
|
InvalidateWindowRect(client_rect);
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::InvalidateWindowRect(const gfx::Rect& rect) {
|
||
|
DCHECK(view_);
|
||
|
if (!g_handling_expose) {
|
||
|
gtk_widget_queue_draw_area(GTK_WIDGET(view_), rect.x(),
|
||
|
rect.y(), rect.width(), rect.height());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::Paint() {
|
||
|
int width, height;
|
||
|
GetSize(width, height);
|
||
|
gfx::Rect client_rect(width, height);
|
||
|
|
||
|
// Number of pixels that the canvas is allowed to differ from the client area.
|
||
|
const int kCanvasGrowSize = 128;
|
||
|
|
||
|
if (!canvas_.get() ||
|
||
|
canvas_w_ < client_rect.width() ||
|
||
|
canvas_h_ < client_rect.height() ||
|
||
|
canvas_w_ > client_rect.width() + kCanvasGrowSize * 2 ||
|
||
|
canvas_h_ > client_rect.height() + kCanvasGrowSize * 2) {
|
||
|
paint_rect_ = client_rect;
|
||
|
|
||
|
// Resize the canvas to be within a reasonable size of the client area.
|
||
|
canvas_w_ = client_rect.width() + kCanvasGrowSize;
|
||
|
canvas_h_ = client_rect.height() + kCanvasGrowSize;
|
||
|
canvas_.reset(skia::CreatePlatformCanvas(canvas_w_, canvas_h_, true));
|
||
|
if (!canvas_.get()) {
|
||
|
// memory allocation failed, we can't paint.
|
||
|
LOG(ERROR) << "Failed to allocate memory for " << width << "x" << height;
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
webwidget_->animate(0.0);
|
||
|
|
||
|
// This may result in more invalidation
|
||
|
webwidget_->layout();
|
||
|
|
||
|
// Paint the canvas if necessary. Allow painting to generate extra rects the
|
||
|
// first time we call it. This is necessary because some WebCore rendering
|
||
|
// objects update their layout only when painted.
|
||
|
// Store the total area painted in total_paint. Then tell the gdk window
|
||
|
// to update that area after we're done painting it.
|
||
|
gfx::Rect total_paint;
|
||
|
for (int i = 0; i < 2; ++i) {
|
||
|
paint_rect_.Intersect(client_rect);
|
||
|
if (!paint_rect_.IsEmpty()) {
|
||
|
gfx::Rect rect(paint_rect_);
|
||
|
paint_rect_ = gfx::Rect();
|
||
|
|
||
|
DLOG_IF(WARNING, i == 1) << "painting caused additional invalidations";
|
||
|
PaintRect(rect);
|
||
|
total_paint.Union(rect);
|
||
|
}
|
||
|
}
|
||
|
// DCHECK(paint_rect_.IsEmpty());
|
||
|
|
||
|
// Invalidate the paint region on the widget's underlying gdk window. Note
|
||
|
// that gdk_window_invalidate_* will generate extra expose events, which
|
||
|
// we wish to avoid. So instead we use calls to begin_paint/end_paint.
|
||
|
GdkRectangle grect = {
|
||
|
total_paint.x(),
|
||
|
total_paint.y(),
|
||
|
total_paint.width(),
|
||
|
total_paint.height(),
|
||
|
};
|
||
|
GdkWindow* window = view_->window;
|
||
|
gdk_window_begin_paint_rect(window, &grect);
|
||
|
|
||
|
// BitBlit to the gdk window.
|
||
|
skia::ScopedPlatformPaint scoped_platform_paint(canvas_.get());
|
||
|
cairo_t* source_surface = scoped_platform_paint.GetPlatformSurface();
|
||
|
cairo_t* cairo_drawable = gdk_cairo_create(window);
|
||
|
cairo_set_source_surface(cairo_drawable, cairo_get_target(source_surface),
|
||
|
0, 0);
|
||
|
cairo_paint(cairo_drawable);
|
||
|
cairo_destroy(cairo_drawable);
|
||
|
|
||
|
gdk_window_end_paint(window);
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::SetTooltipText(const CefString& tooltip_text) {
|
||
|
// TODO(port): Implement this method as part of tooltip support.
|
||
|
}
|
||
|
|
||
|
WebScreenInfo WebWidgetHost::GetScreenInfo() {
|
||
|
Display* display = GtkWidgetGetDisplay(view_);
|
||
|
int screen_num = GtkWidgetGetScreenNum(view_);
|
||
|
return WebScreenInfoFactory::screenInfo(display, screen_num);
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::SendKeyEvent(cef_key_type_t type,
|
||
|
const cef_key_info_t& keyInfo,
|
||
|
int modifiers) {
|
||
|
// TODO(port): Implement this method as part of off-screen rendering support.
|
||
|
NOTIMPLEMENTED();
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::SendMouseClickEvent(int x, int y,
|
||
|
cef_mouse_button_type_t type,
|
||
|
bool mouseUp, int clickCount) {
|
||
|
// TODO(port): Implement this method as part of off-screen rendering support.
|
||
|
NOTIMPLEMENTED();
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::SendMouseMoveEvent(int x, int y, bool mouseLeave) {
|
||
|
// TODO(port): Implement this method as part of off-screen rendering support.
|
||
|
NOTIMPLEMENTED();
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::SendMouseWheelEvent(int x, int y, int deltaX, int deltaY) {
|
||
|
// TODO(port): Implement this method as part of off-screen rendering support.
|
||
|
NOTIMPLEMENTED();
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::SendFocusEvent(bool setFocus) {
|
||
|
// TODO(port): Implement this method as part of off-screen rendering support.
|
||
|
NOTIMPLEMENTED();
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::SendCaptureLostEvent() {
|
||
|
// TODO(port): Implement this method as part of off-screen rendering support.
|
||
|
NOTIMPLEMENTED();
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::EnsureTooltip() {
|
||
|
// TODO(port): Implement this method as part of tooltip support.
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::ResetTooltip() {
|
||
|
// TODO(port): Implement this method as part of tooltip support.
|
||
|
}
|
||
|
|
||
|
void WebWidgetHost::KeyEvent(GdkEventKey* event) {
|
||
|
WebKeyboardEvent keyboard_event(WebInputEventFactory::keyboardEvent(event));
|
||
|
last_key_event_ = keyboard_event;
|
||
|
webwidget()->handleInputEvent(keyboard_event);
|
||
|
|
||
|
// In the browser we do a ton of work with IMEs. This is some minimal
|
||
|
// code to make basic text work in test_shell, but doesn't cover IME.
|
||
|
// This is a copy of the logic in ProcessUnfilteredKeyPressEvent in
|
||
|
// render_widget_host_view_gtk.cc .
|
||
|
if (event->type == GDK_KEY_PRESS && keyboard_event.text[0]) {
|
||
|
keyboard_event.type = WebKit::WebInputEvent::Char;
|
||
|
last_key_event_ = keyboard_event;
|
||
|
webwidget()->handleInputEvent(keyboard_event);
|
||
|
}
|
||
|
}
|