WebKit briefly shows wrong webpage after swiping back (gigaom.com, or any site on...
[WebKit-https.git] / Source / WebKit2 / UIProcess / API / gtk / PageClientImpl.cpp
1 /*
2  * Copyright (C) 2010 Apple Inc. All rights reserved.
3  * Portions Copyright (c) 2010 Motorola Mobility, Inc.  All rights reserved.
4  * Copyright (C) 2011 Igalia S.L.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
16  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
17  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
19  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
20  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
21  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
22  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
23  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
24  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
25  * THE POSSIBILITY OF SUCH DAMAGE.
26  */
27
28 #include "config.h"
29 #include "PageClientImpl.h"
30
31 #include "DrawingAreaProxyImpl.h"
32 #include "NativeWebKeyboardEvent.h"
33 #include "NativeWebMouseEvent.h"
34 #include "NotImplemented.h"
35 #include "WebColorPickerGtk.h"
36 #include "WebContextMenuProxyGtk.h"
37 #include "WebEventFactory.h"
38 #include "WebKitColorChooser.h"
39 #include "WebKitWebView.h"
40 #include "WebKitWebViewBasePrivate.h"
41 #include "WebPageProxy.h"
42 #include "WebPopupMenuProxyGtk.h"
43 #include "WebProcessPool.h"
44 #include <WebCore/Cursor.h>
45 #include <WebCore/EventNames.h>
46 #include <WebCore/GtkUtilities.h>
47 #include <wtf/text/CString.h>
48 #include <wtf/text/WTFString.h>
49
50 using namespace WebCore;
51
52 namespace WebKit {
53
54 PageClientImpl::PageClientImpl(GtkWidget* viewWidget)
55     : m_viewWidget(viewWidget)
56 {
57 }
58
59 void PageClientImpl::getEditorCommandsForKeyEvent(const NativeWebKeyboardEvent& event, const AtomicString& eventType, Vector<WTF::String>& commandList)
60 {
61     ASSERT(eventType == eventNames().keydownEvent || eventType == eventNames().keypressEvent);
62
63     KeyBindingTranslator::EventType type = eventType == eventNames().keydownEvent ?
64         KeyBindingTranslator::KeyDown : KeyBindingTranslator::KeyPress;
65     m_keyBindingTranslator.getEditorCommandsForKeyEvent(const_cast<GdkEventKey*>(&event.nativeEvent()->key), type, commandList);
66 }
67
68 // PageClient's pure virtual functions
69 std::unique_ptr<DrawingAreaProxy> PageClientImpl::createDrawingAreaProxy()
70 {
71     return std::make_unique<DrawingAreaProxyImpl>(*webkitWebViewBaseGetPage(WEBKIT_WEB_VIEW_BASE(m_viewWidget)));
72 }
73
74 void PageClientImpl::setViewNeedsDisplay(const WebCore::IntRect& rect)
75 {
76     gtk_widget_queue_draw_area(m_viewWidget, rect.x(), rect.y(), rect.width(), rect.height());
77 }
78
79 void PageClientImpl::displayView()
80 {
81     notImplemented();
82 }
83
84 void PageClientImpl::scrollView(const WebCore::IntRect& scrollRect, const WebCore::IntSize& /* scrollOffset */)
85 {
86     setViewNeedsDisplay(scrollRect);
87 }
88
89 void PageClientImpl::requestScroll(const WebCore::FloatPoint&, bool)
90 {
91     notImplemented();
92 }
93
94 WebCore::IntSize PageClientImpl::viewSize()
95 {
96     if (!gtk_widget_get_realized(m_viewWidget))
97         return IntSize();
98     GtkAllocation allocation;
99     gtk_widget_get_allocation(m_viewWidget, &allocation);
100     return IntSize(allocation.width, allocation.height);
101 }
102
103 bool PageClientImpl::isViewWindowActive()
104 {
105     return webkitWebViewBaseIsInWindowActive(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
106 }
107
108 bool PageClientImpl::isViewFocused()
109 {
110     return webkitWebViewBaseIsFocused(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
111 }
112
113 bool PageClientImpl::isViewVisible()
114 {
115     return webkitWebViewBaseIsVisible(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
116 }
117
118 bool PageClientImpl::isViewInWindow()
119 {
120     return webkitWebViewBaseIsInWindow(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
121 }
122
123 void PageClientImpl::PageClientImpl::processDidExit()
124 {
125     notImplemented();
126 }
127
128 void PageClientImpl::didRelaunchProcess()
129 {
130     notImplemented();
131 }
132
133 void PageClientImpl::toolTipChanged(const String&, const String& newToolTip)
134 {
135     webkitWebViewBaseSetTooltipText(WEBKIT_WEB_VIEW_BASE(m_viewWidget), newToolTip.utf8().data());
136 }
137
138 void PageClientImpl::setCursor(const Cursor& cursor)
139 {
140     if (!gtk_widget_get_realized(m_viewWidget))
141         return;
142
143     // [GTK] Widget::setCursor() gets called frequently
144     // http://bugs.webkit.org/show_bug.cgi?id=16388
145     // Setting the cursor may be an expensive operation in some backends,
146     // so don't re-set the cursor if it's already set to the target value.
147     GdkWindow* window = gtk_widget_get_window(m_viewWidget);
148     GdkCursor* currentCursor = gdk_window_get_cursor(window);
149     GdkCursor* newCursor = cursor.platformCursor().get();
150     if (currentCursor != newCursor)
151         gdk_window_set_cursor(window, newCursor);
152 }
153
154 void PageClientImpl::setCursorHiddenUntilMouseMoves(bool /* hiddenUntilMouseMoves */)
155 {
156     notImplemented();
157 }
158
159 void PageClientImpl::didChangeViewportProperties(const WebCore::ViewportAttributes&)
160 {
161     notImplemented();
162 }
163
164 void PageClientImpl::registerEditCommand(PassRefPtr<WebEditCommandProxy> command, WebPageProxy::UndoOrRedo undoOrRedo)
165 {
166     m_undoController.registerEditCommand(command, undoOrRedo);
167 }
168
169 void PageClientImpl::clearAllEditCommands()
170 {
171     m_undoController.clearAllEditCommands();
172 }
173
174 bool PageClientImpl::canUndoRedo(WebPageProxy::UndoOrRedo undoOrRedo)
175 {
176     return m_undoController.canUndoRedo(undoOrRedo);
177 }
178
179 void PageClientImpl::executeUndoRedo(WebPageProxy::UndoOrRedo undoOrRedo)
180 {
181     m_undoController.executeUndoRedo(undoOrRedo);
182 }
183
184 FloatRect PageClientImpl::convertToDeviceSpace(const FloatRect& viewRect)
185 {
186     notImplemented();
187     return viewRect;
188 }
189
190 FloatRect PageClientImpl::convertToUserSpace(const FloatRect& viewRect)
191 {
192     notImplemented();
193     return viewRect;
194 }
195
196 IntPoint PageClientImpl::screenToRootView(const IntPoint& point)
197 {
198     IntPoint widgetPositionOnScreen = convertWidgetPointToScreenPoint(m_viewWidget, IntPoint());
199     IntPoint result(point);
200     result.move(-widgetPositionOnScreen.x(), -widgetPositionOnScreen.y());
201     return result;
202 }
203
204 IntRect PageClientImpl::rootViewToScreen(const IntRect& rect)
205 {
206     return IntRect(convertWidgetPointToScreenPoint(m_viewWidget, rect.location()), rect.size());
207 }
208
209 void PageClientImpl::doneWithKeyEvent(const NativeWebKeyboardEvent& event, bool wasEventHandled)
210 {
211     if (wasEventHandled)
212         return;
213     if (event.isFakeEventForComposition())
214         return;
215
216     WebKitWebViewBase* webkitWebViewBase = WEBKIT_WEB_VIEW_BASE(m_viewWidget);
217     webkitWebViewBaseForwardNextKeyEvent(webkitWebViewBase);
218     gtk_main_do_event(event.nativeEvent());
219 }
220
221 PassRefPtr<WebPopupMenuProxy> PageClientImpl::createPopupMenuProxy(WebPageProxy* page)
222 {
223     return WebPopupMenuProxyGtk::create(m_viewWidget, page);
224 }
225
226 PassRefPtr<WebContextMenuProxy> PageClientImpl::createContextMenuProxy(WebPageProxy* page)
227 {
228     return WebContextMenuProxyGtk::create(m_viewWidget, page);
229 }
230
231 PassRefPtr<WebColorPicker> PageClientImpl::createColorPicker(WebPageProxy* page, const WebCore::Color& color, const WebCore::IntRect& rect)
232 {
233     if (WEBKIT_IS_WEB_VIEW(m_viewWidget))
234         return WebKitColorChooser::create(*page, color, rect);
235     return WebColorPickerGtk::create(*page, color, rect);
236 }
237
238 void PageClientImpl::setTextIndicator(PassRefPtr<WebCore::TextIndicator>, bool /* fadeOut */)
239 {
240     notImplemented();
241 }
242
243 void PageClientImpl::setTextIndicatorAnimationProgress(float)
244 {
245     notImplemented();
246 }
247
248 void PageClientImpl::enterAcceleratedCompositingMode(const LayerTreeContext&)
249 {
250     webkitWebViewBaseEnterAcceleratedCompositingMode(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
251 }
252
253 void PageClientImpl::exitAcceleratedCompositingMode()
254 {
255     webkitWebViewBaseExitAcceleratedCompositingMode(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
256 }
257
258 void PageClientImpl::updateAcceleratedCompositingMode(const LayerTreeContext&)
259 {
260     notImplemented();
261 }
262
263 void PageClientImpl::pageClosed()
264 {
265     notImplemented();
266 }
267
268 void PageClientImpl::preferencesDidChange()
269 {
270     notImplemented();
271 }
272
273 void PageClientImpl::updateTextInputState()
274 {
275     webkitWebViewBaseUpdateTextInputState(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
276 }
277
278 #if ENABLE(DRAG_SUPPORT)
279 void PageClientImpl::startDrag(const WebCore::DragData& dragData, PassRefPtr<ShareableBitmap> dragImage)
280 {
281     WebKitWebViewBase* webView = WEBKIT_WEB_VIEW_BASE(m_viewWidget);
282     webkitWebViewBaseDragAndDropHandler(webView).startDrag(dragData, dragImage);
283
284     // A drag starting should prevent a double-click from happening. This might
285     // happen if a drag is followed very quickly by another click (like in the WTR).
286     webkitWebViewBaseResetClickCounter(webView);
287 }
288 #endif
289
290 void PageClientImpl::handleDownloadRequest(DownloadProxy* download)
291 {
292     webkitWebViewBaseHandleDownloadRequest(WEBKIT_WEB_VIEW_BASE(m_viewWidget), download);
293 }
294
295 void PageClientImpl::didCommitLoadForMainFrame(const String& /* mimeType */, bool /* useCustomContentProvider */ )
296 {
297     webkitWebViewBaseResetClickCounter(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
298 }
299
300 #if ENABLE(FULLSCREEN_API)
301 WebFullScreenManagerProxyClient& PageClientImpl::fullScreenManagerProxyClient()
302 {
303     return *this;
304 }
305
306 void PageClientImpl::closeFullScreenManager()
307 {
308     notImplemented();
309 }
310
311 bool PageClientImpl::isFullScreen()
312 {
313     notImplemented();
314     return false;
315 }
316
317 void PageClientImpl::enterFullScreen()
318 {
319     if (!m_viewWidget)
320         return;
321
322     webkitWebViewBaseEnterFullScreen(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
323 }
324
325 void PageClientImpl::exitFullScreen()
326 {
327     if (!m_viewWidget)
328         return;
329
330     webkitWebViewBaseExitFullScreen(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
331 }
332
333 void PageClientImpl::beganEnterFullScreen(const IntRect& /* initialFrame */, const IntRect& /* finalFrame */)
334 {
335     notImplemented();
336 }
337
338 void PageClientImpl::beganExitFullScreen(const IntRect& /* initialFrame */, const IntRect& /* finalFrame */)
339 {
340     notImplemented();
341 }
342
343 #endif // ENABLE(FULLSCREEN_API)
344
345 void PageClientImpl::doneWithTouchEvent(const NativeWebTouchEvent& event, bool wasEventHandled)
346 {
347     if (wasEventHandled)
348         return;
349
350 #if HAVE(GTK_GESTURES)
351     GestureController& gestureController = webkitWebViewBaseGestureController(WEBKIT_WEB_VIEW_BASE(m_viewWidget));
352     if (gestureController.handleEvent(event.nativeEvent()))
353         return;
354 #endif
355
356     // Emulate pointer events if unhandled.
357     const GdkEvent* touchEvent = event.nativeEvent();
358
359     if (!touchEvent->touch.emulating_pointer)
360         return;
361
362     GUniquePtr<GdkEvent> pointerEvent;
363
364     if (touchEvent->type == GDK_TOUCH_UPDATE) {
365         pointerEvent.reset(gdk_event_new(GDK_MOTION_NOTIFY));
366         pointerEvent->motion.time = touchEvent->touch.time;
367         pointerEvent->motion.x = touchEvent->touch.x;
368         pointerEvent->motion.y = touchEvent->touch.y;
369         pointerEvent->motion.x_root = touchEvent->touch.x_root;
370         pointerEvent->motion.y_root = touchEvent->touch.y_root;
371         pointerEvent->motion.state = touchEvent->touch.state | GDK_BUTTON1_MASK;
372     } else {
373         switch (touchEvent->type) {
374         case GDK_TOUCH_END:
375             pointerEvent.reset(gdk_event_new(GDK_BUTTON_RELEASE));
376             pointerEvent->button.state = touchEvent->touch.state | GDK_BUTTON1_MASK;
377             break;
378         case GDK_TOUCH_BEGIN:
379             pointerEvent.reset(gdk_event_new(GDK_BUTTON_PRESS));
380             break;
381         default:
382             ASSERT_NOT_REACHED();
383         }
384
385         pointerEvent->button.button = 1;
386         pointerEvent->button.time = touchEvent->touch.time;
387         pointerEvent->button.x = touchEvent->touch.x;
388         pointerEvent->button.y = touchEvent->touch.y;
389         pointerEvent->button.x_root = touchEvent->touch.x_root;
390         pointerEvent->button.y_root = touchEvent->touch.y_root;
391     }
392
393     gdk_event_set_device(pointerEvent.get(), gdk_event_get_device(touchEvent));
394     gdk_event_set_source_device(pointerEvent.get(), gdk_event_get_source_device(touchEvent));
395     pointerEvent->any.window = GDK_WINDOW(g_object_ref(touchEvent->any.window));
396     pointerEvent->any.send_event = TRUE;
397
398     gtk_widget_event(m_viewWidget, pointerEvent.get());
399 }
400
401 void PageClientImpl::didFinishLoadingDataForCustomContentProvider(const String&, const IPC::DataReference&)
402 {
403 }
404
405 void PageClientImpl::navigationGestureDidBegin()
406 {
407 }
408
409 void PageClientImpl::navigationGestureWillEnd(bool, WebBackForwardListItem&)
410 {
411 }
412
413 void PageClientImpl::navigationGestureDidEnd(bool, WebBackForwardListItem&)
414 {
415 }
416
417 void PageClientImpl::willRecordNavigationSnapshot(WebBackForwardListItem&)
418 {
419 }
420
421 void PageClientImpl::didFirstVisuallyNonEmptyLayoutForMainFrame()
422 {
423 }
424
425 void PageClientImpl::didFinishLoadForMainFrame()
426 {
427 }
428
429 void PageClientImpl::didSameDocumentNavigationForMainFrame(SameDocumentNavigationType)
430 {
431 }
432
433 void PageClientImpl::didChangeBackgroundColor()
434 {
435 }
436
437 } // namespace WebKit