1 /*
2 * Copyright (C) 2009 Google Inc. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
6 * met:
7 *
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above
11 * copyright notice, this list of conditions and the following disclaimer
12 * in the documentation and/or other materials provided with the
13 * distribution.
14 * * Neither the name of Google Inc. nor the names of its
15 * contributors may be used to endorse or promote products derived from
16 * this software without specific prior written permission.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 */
30
31 #include "config.h"
32 #include "web/WebPopupMenuImpl.h"
33
34 #include "core/frame/FrameView.h"
35 #include "platform/Cursor.h"
36 #include "platform/NotImplemented.h"
37 #include "platform/PlatformGestureEvent.h"
38 #include "platform/PlatformKeyboardEvent.h"
39 #include "platform/PlatformMouseEvent.h"
40 #include "platform/PlatformWheelEvent.h"
41 #include "platform/geometry/IntRect.h"
42 #include "platform/graphics/GraphicsContext.h"
43 #include "platform/graphics/skia/SkiaUtils.h"
44 #include "platform/scroll/FramelessScrollView.h"
45 #include "public/platform/Platform.h"
46 #include "public/platform/WebCompositorSupport.h"
47 #include "public/platform/WebContentLayer.h"
48 #include "public/platform/WebFloatRect.h"
49 #include "public/platform/WebLayerTreeView.h"
50 #include "public/platform/WebRect.h"
51 #include "public/web/WebInputEvent.h"
52 #include "public/web/WebRange.h"
53 #include "public/web/WebViewClient.h"
54 #include "public/web/WebWidgetClient.h"
55 #include "web/PopupContainer.h"
56 #include "web/PopupMenuChromium.h"
57 #include "web/WebInputEventConversion.h"
58 #include <skia/ext/platform_canvas.h>
59
60 using namespace WebCore;
61
62 namespace blink {
63
64 // WebPopupMenu ---------------------------------------------------------------
65
create(WebWidgetClient * client)66 WebPopupMenu* WebPopupMenu::create(WebWidgetClient* client)
67 {
68 // Pass the WebPopupMenuImpl's self-reference to the caller.
69 return adoptRef(new WebPopupMenuImpl(client)).leakRef();
70 }
71
72 // WebWidget ------------------------------------------------------------------
73
WebPopupMenuImpl(WebWidgetClient * client)74 WebPopupMenuImpl::WebPopupMenuImpl(WebWidgetClient* client)
75 : m_client(client)
76 , m_layerTreeView(0)
77 // Set to impossible point so we always get the first mouse position.
78 , m_lastMousePosition(WebPoint(-1, -1))
79 , m_widget(0)
80 {
81 }
82
~WebPopupMenuImpl()83 WebPopupMenuImpl::~WebPopupMenuImpl()
84 {
85 if (m_widget)
86 m_widget->setClient(0);
87 }
88
willCloseLayerTreeView()89 void WebPopupMenuImpl::willCloseLayerTreeView()
90 {
91 m_layerTreeView = 0;
92 }
93
initialize(FramelessScrollView * widget,const WebRect & bounds)94 void WebPopupMenuImpl::initialize(FramelessScrollView* widget, const WebRect& bounds)
95 {
96 m_widget = widget;
97 m_widget->setClient(this);
98
99 if (!m_client)
100 return;
101 m_client->setWindowRect(bounds);
102 m_client->show(WebNavigationPolicy()); // Policy is ignored.
103
104 m_client->initializeLayerTreeView();
105 m_layerTreeView = m_client->layerTreeView();
106 if (m_layerTreeView) {
107 m_layerTreeView->setVisible(true);
108 m_layerTreeView->setDeviceScaleFactor(m_client->deviceScaleFactor());
109 m_rootLayer = adoptPtr(Platform::current()->compositorSupport()->createContentLayer(this));
110 m_rootLayer->layer()->setBounds(m_size);
111 m_layerTreeView->setRootLayer(*m_rootLayer->layer());
112 }
113 }
114
handleMouseMove(const WebMouseEvent & event)115 void WebPopupMenuImpl::handleMouseMove(const WebMouseEvent& event)
116 {
117 // Don't send mouse move messages if the mouse hasn't moved.
118 if (event.x != m_lastMousePosition.x || event.y != m_lastMousePosition.y) {
119 m_lastMousePosition = WebPoint(event.x, event.y);
120 m_widget->handleMouseMoveEvent(PlatformMouseEventBuilder(m_widget, event));
121
122 // We cannot call setToolTipText() in PopupContainer, because PopupContainer is in WebCore, and we cannot refer to WebKit from Webcore.
123 PopupContainer* container = static_cast<PopupContainer*>(m_widget);
124 client()->setToolTipText(container->getSelectedItemToolTip(), container->menuStyle().textDirection() == WebCore::RTL ? WebTextDirectionRightToLeft : WebTextDirectionLeftToRight);
125 }
126 }
127
handleMouseLeave(const WebMouseEvent & event)128 void WebPopupMenuImpl::handleMouseLeave(const WebMouseEvent& event)
129 {
130 m_widget->handleMouseMoveEvent(PlatformMouseEventBuilder(m_widget, event));
131 }
132
handleMouseDown(const WebMouseEvent & event)133 void WebPopupMenuImpl::handleMouseDown(const WebMouseEvent& event)
134 {
135 m_widget->handleMouseDownEvent(PlatformMouseEventBuilder(m_widget, event));
136 }
137
handleMouseUp(const WebMouseEvent & event)138 void WebPopupMenuImpl::handleMouseUp(const WebMouseEvent& event)
139 {
140 mouseCaptureLost();
141 m_widget->handleMouseReleaseEvent(PlatformMouseEventBuilder(m_widget, event));
142 }
143
handleMouseWheel(const WebMouseWheelEvent & event)144 void WebPopupMenuImpl::handleMouseWheel(const WebMouseWheelEvent& event)
145 {
146 m_widget->handleWheelEvent(PlatformWheelEventBuilder(m_widget, event));
147 }
148
handleGestureEvent(const WebGestureEvent & event)149 bool WebPopupMenuImpl::handleGestureEvent(const WebGestureEvent& event)
150 {
151 return m_widget->handleGestureEvent(PlatformGestureEventBuilder(m_widget, event));
152 }
153
handleTouchEvent(const WebTouchEvent & event)154 bool WebPopupMenuImpl::handleTouchEvent(const WebTouchEvent& event)
155 {
156
157 PlatformTouchEventBuilder touchEventBuilder(m_widget, event);
158 bool defaultPrevented(m_widget->handleTouchEvent(touchEventBuilder));
159 return defaultPrevented;
160 }
161
handleKeyEvent(const WebKeyboardEvent & event)162 bool WebPopupMenuImpl::handleKeyEvent(const WebKeyboardEvent& event)
163 {
164 return m_widget->handleKeyEvent(PlatformKeyboardEventBuilder(event));
165 }
166
167 // WebWidget -------------------------------------------------------------------
168
close()169 void WebPopupMenuImpl::close()
170 {
171 if (m_widget)
172 m_widget->hide();
173
174 m_client = 0;
175
176 deref(); // Balances ref() from WebPopupMenu::create.
177 }
178
willStartLiveResize()179 void WebPopupMenuImpl::willStartLiveResize()
180 {
181 }
182
resize(const WebSize & newSize)183 void WebPopupMenuImpl::resize(const WebSize& newSize)
184 {
185 if (m_size == newSize)
186 return;
187 m_size = newSize;
188
189 if (m_widget) {
190 IntRect newGeometry(0, 0, m_size.width, m_size.height);
191 m_widget->setFrameRect(newGeometry);
192 }
193
194 if (m_client) {
195 WebRect damagedRect(0, 0, m_size.width, m_size.height);
196 m_client->didInvalidateRect(damagedRect);
197 }
198
199 if (m_rootLayer)
200 m_rootLayer->layer()->setBounds(newSize);
201 }
202
willEndLiveResize()203 void WebPopupMenuImpl::willEndLiveResize()
204 {
205 }
206
animate(double)207 void WebPopupMenuImpl::animate(double)
208 {
209 }
210
layout()211 void WebPopupMenuImpl::layout()
212 {
213 }
214
paintContents(WebCanvas * canvas,const WebRect & rect,bool,WebFloatRect &,WebContentLayerClient::GraphicsContextStatus contextStatus)215 void WebPopupMenuImpl::paintContents(WebCanvas* canvas, const WebRect& rect, bool, WebFloatRect&,
216 WebContentLayerClient::GraphicsContextStatus contextStatus)
217 {
218 if (!m_widget)
219 return;
220
221 if (!rect.isEmpty()) {
222 GraphicsContext context(canvas,
223 contextStatus == WebContentLayerClient::GraphicsContextEnabled ? GraphicsContext::NothingDisabled : GraphicsContext::FullyDisabled);
224 m_widget->paint(&context, rect);
225 }
226 }
227
paint(WebCanvas * canvas,const WebRect & rect)228 void WebPopupMenuImpl::paint(WebCanvas* canvas, const WebRect& rect)
229 {
230 if (!m_widget)
231 return;
232
233 if (!rect.isEmpty()) {
234 GraphicsContext context(canvas);
235 context.applyDeviceScaleFactor(m_client->deviceScaleFactor());
236 m_widget->paint(&context, rect);
237 }
238 }
239
themeChanged()240 void WebPopupMenuImpl::themeChanged()
241 {
242 notImplemented();
243 }
244
handleInputEvent(const WebInputEvent & inputEvent)245 bool WebPopupMenuImpl::handleInputEvent(const WebInputEvent& inputEvent)
246 {
247 if (!m_widget)
248 return false;
249
250 // FIXME: WebKit seems to always return false on mouse events methods. For
251 // now we'll assume it has processed them (as we are only interested in
252 // whether keyboard events are processed).
253 switch (inputEvent.type) {
254 case WebInputEvent::MouseMove:
255 handleMouseMove(*static_cast<const WebMouseEvent*>(&inputEvent));
256 return true;
257
258 case WebInputEvent::MouseLeave:
259 handleMouseLeave(*static_cast<const WebMouseEvent*>(&inputEvent));
260 return true;
261
262 case WebInputEvent::MouseWheel:
263 handleMouseWheel(*static_cast<const WebMouseWheelEvent*>(&inputEvent));
264 return true;
265
266 case WebInputEvent::MouseDown:
267 handleMouseDown(*static_cast<const WebMouseEvent*>(&inputEvent));
268 return true;
269
270 case WebInputEvent::MouseUp:
271 handleMouseUp(*static_cast<const WebMouseEvent*>(&inputEvent));
272 return true;
273
274 // In Windows, RawKeyDown only has information about the physical key, but
275 // for "selection", we need the information about the character the key
276 // translated into. For English, the physical key value and the character
277 // value are the same, hence, "selection" works for English. But for other
278 // languages, such as Hebrew, the character value is different from the
279 // physical key value. Thus, without accepting Char event type which
280 // contains the key's character value, the "selection" won't work for
281 // non-English languages, such as Hebrew.
282 case WebInputEvent::RawKeyDown:
283 case WebInputEvent::KeyDown:
284 case WebInputEvent::KeyUp:
285 case WebInputEvent::Char:
286 return handleKeyEvent(*static_cast<const WebKeyboardEvent*>(&inputEvent));
287
288 case WebInputEvent::TouchStart:
289 case WebInputEvent::TouchMove:
290 case WebInputEvent::TouchEnd:
291 case WebInputEvent::TouchCancel:
292 return handleTouchEvent(*static_cast<const WebTouchEvent*>(&inputEvent));
293
294 case WebInputEvent::GestureScrollBegin:
295 case WebInputEvent::GestureScrollEnd:
296 case WebInputEvent::GestureScrollUpdate:
297 case WebInputEvent::GestureScrollUpdateWithoutPropagation:
298 case WebInputEvent::GestureFlingStart:
299 case WebInputEvent::GestureFlingCancel:
300 case WebInputEvent::GestureTap:
301 case WebInputEvent::GestureTapUnconfirmed:
302 case WebInputEvent::GestureTapDown:
303 case WebInputEvent::GestureShowPress:
304 case WebInputEvent::GestureTapCancel:
305 case WebInputEvent::GestureDoubleTap:
306 case WebInputEvent::GestureTwoFingerTap:
307 case WebInputEvent::GestureLongPress:
308 case WebInputEvent::GestureLongTap:
309 case WebInputEvent::GesturePinchBegin:
310 case WebInputEvent::GesturePinchEnd:
311 case WebInputEvent::GesturePinchUpdate:
312 return handleGestureEvent(*static_cast<const WebGestureEvent*>(&inputEvent));
313
314 case WebInputEvent::Undefined:
315 case WebInputEvent::MouseEnter:
316 case WebInputEvent::ContextMenu:
317 return false;
318 }
319 return false;
320 }
321
mouseCaptureLost()322 void WebPopupMenuImpl::mouseCaptureLost()
323 {
324 }
325
setFocus(bool)326 void WebPopupMenuImpl::setFocus(bool)
327 {
328 }
329
setComposition(const WebString &,const WebVector<WebCompositionUnderline> &,int,int)330 bool WebPopupMenuImpl::setComposition(const WebString&, const WebVector<WebCompositionUnderline>&, int, int)
331 {
332 return false;
333 }
334
confirmComposition()335 bool WebPopupMenuImpl::confirmComposition()
336 {
337 return false;
338 }
339
confirmComposition(ConfirmCompositionBehavior)340 bool WebPopupMenuImpl::confirmComposition(ConfirmCompositionBehavior)
341 {
342 return false;
343 }
344
confirmComposition(const WebString &)345 bool WebPopupMenuImpl::confirmComposition(const WebString&)
346 {
347 return false;
348 }
349
compositionRange(size_t * location,size_t * length)350 bool WebPopupMenuImpl::compositionRange(size_t* location, size_t* length)
351 {
352 *location = 0;
353 *length = 0;
354 return false;
355 }
356
caretOrSelectionRange(size_t * location,size_t * length)357 bool WebPopupMenuImpl::caretOrSelectionRange(size_t* location, size_t* length)
358 {
359 *location = 0;
360 *length = 0;
361 return false;
362 }
363
setTextDirection(WebTextDirection)364 void WebPopupMenuImpl::setTextDirection(WebTextDirection)
365 {
366 }
367
368
369 //-----------------------------------------------------------------------------
370 // WebCore::HostWindow
371
invalidateContentsAndRootView(const IntRect & paintRect)372 void WebPopupMenuImpl::invalidateContentsAndRootView(const IntRect& paintRect)
373 {
374 if (paintRect.isEmpty())
375 return;
376 if (m_client)
377 m_client->didInvalidateRect(paintRect);
378 if (m_rootLayer)
379 m_rootLayer->layer()->invalidateRect(FloatRect(paintRect));
380 }
381
invalidateContentsForSlowScroll(const IntRect & updateRect)382 void WebPopupMenuImpl::invalidateContentsForSlowScroll(const IntRect& updateRect)
383 {
384 invalidateContentsAndRootView(updateRect);
385 }
386
scheduleAnimation()387 void WebPopupMenuImpl::scheduleAnimation()
388 {
389 }
390
scroll(const IntSize & scrollDelta,const IntRect & scrollRect,const IntRect & clipRect)391 void WebPopupMenuImpl::scroll(const IntSize& scrollDelta, const IntRect& scrollRect, const IntRect& clipRect)
392 {
393 if (m_client) {
394 int dx = scrollDelta.width();
395 int dy = scrollDelta.height();
396 m_client->didScrollRect(dx, dy, clipRect);
397 }
398 if (m_rootLayer)
399 m_rootLayer->layer()->invalidateRect(FloatRect(clipRect));
400 }
401
rootViewToScreen(const IntRect & rect) const402 IntRect WebPopupMenuImpl::rootViewToScreen(const IntRect& rect) const
403 {
404 notImplemented();
405 return IntRect();
406 }
407
screenInfo() const408 WebScreenInfo WebPopupMenuImpl::screenInfo() const
409 {
410 return WebScreenInfo();
411 }
412
413 //-----------------------------------------------------------------------------
414 // WebCore::FramelessScrollViewClient
415
popupClosed(FramelessScrollView * widget)416 void WebPopupMenuImpl::popupClosed(FramelessScrollView* widget)
417 {
418 ASSERT(widget == m_widget);
419 if (m_widget) {
420 m_widget->setClient(0);
421 m_widget = 0;
422 }
423 if (m_client)
424 m_client->closeWidgetSoon();
425 }
426
427 } // namespace blink
428