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 #ifndef WebInputEvent_h 32 #define WebInputEvent_h 33 34 #include "../platform/WebCommon.h" 35 #include "../platform/WebGestureDevice.h" 36 #include "../platform/WebRect.h" 37 #include "WebTouchPoint.h" 38 39 #include <string.h> 40 41 namespace blink { 42 43 // The classes defined in this file are intended to be used with 44 // WebWidget's handleInputEvent method. These event types are cross- 45 // platform and correspond closely to WebCore's Platform*Event classes. 46 // 47 // WARNING! These classes must remain PODs (plain old data). They are 48 // intended to be "serializable" by copying their raw bytes, so they must 49 // not contain any non-bit-copyable member variables! 50 // 51 // Furthermore, the class members need to be packed so they are aligned 52 // properly and don't have paddings/gaps, otherwise memory check tools 53 // like Valgrind will complain about uninitialized memory usage when 54 // transferring these classes over the wire. 55 56 #pragma pack(push, 4) 57 58 // WebInputEvent -------------------------------------------------------------- 59 60 class WebInputEvent { 61 public: 62 // When we use an input method (or an input method editor), we receive 63 // two events for a keypress. The former event is a keydown, which 64 // provides a keycode, and the latter is a textinput, which provides 65 // a character processed by an input method. (The mapping from a 66 // keycode to a character code is not trivial for non-English 67 // keyboards.) 68 // To support input methods, Safari sends keydown events to WebKit for 69 // filtering. WebKit sends filtered keydown events back to Safari, 70 // which sends them to input methods. 71 // Unfortunately, it is hard to apply this design to Chrome because of 72 // our multiprocess architecture. An input method is running in a 73 // browser process. On the other hand, WebKit is running in a renderer 74 // process. So, this design results in increasing IPC messages. 75 // To support input methods without increasing IPC messages, Chrome 76 // handles keyboard events in a browser process and send asynchronous 77 // input events (to be translated to DOM events) to a renderer 78 // process. 79 // This design is mostly the same as the one of Windows and Mac Carbon. 80 // So, for what it's worth, our Linux and Mac front-ends emulate our 81 // Windows front-end. To emulate our Windows front-end, we can share 82 // our back-end code among Windows, Linux, and Mac. 83 // TODO(hbono): Issue 18064: remove the KeyDown type since it isn't 84 // used in Chrome any longer. 85 86 enum Type { 87 Undefined = -1, 88 TypeFirst = Undefined, 89 90 // WebMouseEvent 91 MouseDown, 92 MouseTypeFirst = MouseDown, 93 MouseUp, 94 MouseMove, 95 MouseEnter, 96 MouseLeave, 97 ContextMenu, 98 MouseTypeLast = ContextMenu, 99 100 // WebMouseWheelEvent 101 MouseWheel, 102 103 // WebKeyboardEvent 104 RawKeyDown, 105 KeyboardTypeFirst = RawKeyDown, 106 KeyDown, 107 KeyUp, 108 Char, 109 KeyboardTypeLast = Char, 110 111 // WebGestureEvent 112 GestureScrollBegin, 113 GestureTypeFirst = GestureScrollBegin, 114 GestureScrollEnd, 115 GestureScrollUpdate, 116 GestureScrollUpdateWithoutPropagation, 117 GestureFlingStart, 118 GestureFlingCancel, 119 GestureShowPress, 120 GestureTap, 121 GestureTapUnconfirmed, 122 GestureTapDown, 123 GestureTapCancel, 124 GestureDoubleTap, 125 GestureTwoFingerTap, 126 GestureLongPress, 127 GestureLongTap, 128 GesturePinchBegin, 129 GesturePinchEnd, 130 GesturePinchUpdate, 131 GestureTypeLast = GesturePinchUpdate, 132 133 // WebTouchEvent 134 TouchStart, 135 TouchTypeFirst = TouchStart, 136 TouchMove, 137 TouchEnd, 138 TouchCancel, 139 TouchTypeLast = TouchCancel, 140 141 TypeLast = TouchTypeLast 142 }; 143 144 enum Modifiers { 145 // modifiers for all events: 146 ShiftKey = 1 << 0, 147 ControlKey = 1 << 1, 148 AltKey = 1 << 2, 149 MetaKey = 1 << 3, 150 151 // modifiers for keyboard events: 152 IsKeyPad = 1 << 4, 153 IsAutoRepeat = 1 << 5, 154 155 // modifiers for mouse events: 156 LeftButtonDown = 1 << 6, 157 MiddleButtonDown = 1 << 7, 158 RightButtonDown = 1 << 8, 159 160 // Toggle modifiers for all events. Danger: these are not reflected 161 // into WebCore, so round-tripping from WebInputEvent to a WebCore 162 // event and back will not preserve these flags. 163 CapsLockOn = 1 << 9, 164 NumLockOn = 1 << 10, 165 166 // Left/right modifiers for keyboard events. 167 IsLeft = 1 << 11, 168 IsRight = 1 << 12, 169 170 // Last input event to be sent for the current vsync interval. If this 171 // flag is set, the sender guarantees that no more input events will be 172 // delivered until the next vsync and the receiver can schedule 173 // rendering accordingly. If it isn't set, the receiver should not make 174 // any assumptions about the delivery times of future input events 175 // w.r.t. vsync. 176 IsLastInputEventForCurrentVSync = 1 << 13, 177 }; 178 179 static const int InputModifiers = ShiftKey | ControlKey | AltKey | MetaKey; 180 181 double timeStampSeconds; // Seconds since epoch. 182 unsigned size; // The size of this structure, for serialization. 183 Type type; 184 int modifiers; 185 186 // Returns true if the WebInputEvent |type| is a mouse event. isMouseEventType(int type)187 static bool isMouseEventType(int type) 188 { 189 return MouseTypeFirst <= type && type <= MouseTypeLast; 190 } 191 192 // Returns true if the WebInputEvent |type| is a keyboard event. isKeyboardEventType(int type)193 static bool isKeyboardEventType(int type) 194 { 195 return KeyboardTypeFirst <= type && type <= KeyboardTypeLast; 196 } 197 198 // Returns true if the WebInputEvent |type| is a touch event. isTouchEventType(int type)199 static bool isTouchEventType(int type) 200 { 201 return TouchTypeFirst <= type && type <= TouchTypeLast; 202 } 203 204 // Returns true if the WebInputEvent is a gesture event. isGestureEventType(int type)205 static bool isGestureEventType(int type) 206 { 207 return GestureTypeFirst <= type && type <= GestureTypeLast; 208 } 209 210 protected: WebInputEvent(unsigned sizeParam)211 explicit WebInputEvent(unsigned sizeParam) 212 { 213 memset(this, 0, sizeParam); 214 timeStampSeconds = 0.0; 215 size = sizeParam; 216 type = Undefined; 217 modifiers = 0; 218 } 219 }; 220 221 // WebKeyboardEvent ----------------------------------------------------------- 222 223 class WebKeyboardEvent : public WebInputEvent { 224 public: 225 // Caps on string lengths so we can make them static arrays and keep 226 // them PODs. 227 static const size_t textLengthCap = 4; 228 229 // http://www.w3.org/TR/DOM-Level-3-Events/keyset.html lists the 230 // identifiers. The longest is 18 characters, so we round up to the 231 // next multiple of 4. 232 static const size_t keyIdentifierLengthCap = 20; 233 234 // |windowsKeyCode| is the Windows key code associated with this key 235 // event. Sometimes it's direct from the event (i.e. on Windows), 236 // sometimes it's via a mapping function. If you want a list, see 237 // WebCore/platform/chromium/KeyboardCodes* . Note that this should 238 // ALWAYS store the non-locational version of a keycode as this is 239 // what is returned by the Windows API. For example, it should 240 // store VK_SHIFT instead of VK_RSHIFT. The location information 241 // should be stored in |modifiers|. 242 int windowsKeyCode; 243 244 // The actual key code genenerated by the platform. The DOM spec runs 245 // on Windows-equivalent codes (thus |windowsKeyCode| above) but it 246 // doesn't hurt to have this one around. 247 int nativeKeyCode; 248 249 // This identifies whether this event was tagged by the system as being 250 // a "system key" event (see 251 // http://msdn.microsoft.com/en-us/library/ms646286(VS.85).aspx for 252 // details). Other platforms don't have this concept, but it's just 253 // easier to leave it always false than ifdef. 254 // See comment at the top of the file for why an int is used here. 255 bool isSystemKey; 256 257 // |text| is the text generated by this keystroke. |unmodifiedText| is 258 // |text|, but unmodified by an concurrently-held modifiers (except 259 // shift). This is useful for working out shortcut keys. Linux and 260 // Windows guarantee one character per event. The Mac does not, but in 261 // reality that's all it ever gives. We're generous, and cap it a bit 262 // longer. 263 WebUChar text[textLengthCap]; 264 WebUChar unmodifiedText[textLengthCap]; 265 266 // This is a string identifying the key pressed. 267 char keyIdentifier[keyIdentifierLengthCap]; 268 WebKeyboardEvent()269 WebKeyboardEvent() 270 : WebInputEvent(sizeof(WebKeyboardEvent)) 271 , windowsKeyCode(0) 272 , nativeKeyCode(0) 273 , isSystemKey(false) 274 { 275 memset(&text, 0, sizeof(text)); 276 memset(&unmodifiedText, 0, sizeof(unmodifiedText)); 277 memset(&keyIdentifier, 0, sizeof(keyIdentifier)); 278 } 279 280 // Sets keyIdentifier based on the value of windowsKeyCode. This is 281 // handy for generating synthetic keyboard events. 282 BLINK_EXPORT void setKeyIdentifierFromWindowsKeyCode(); 283 284 static int windowsKeyCodeWithoutLocation(int keycode); 285 static int locationModifiersFromWindowsKeyCode(int keycode); 286 }; 287 288 // WebMouseEvent -------------------------------------------------------------- 289 290 class WebMouseEvent : public WebInputEvent { 291 public: 292 enum Button { 293 ButtonNone = -1, 294 ButtonLeft, 295 ButtonMiddle, 296 ButtonRight 297 }; 298 299 Button button; 300 int x; 301 int y; 302 int windowX; 303 int windowY; 304 int globalX; 305 int globalY; 306 int movementX; 307 int movementY; 308 int clickCount; 309 WebMouseEvent()310 WebMouseEvent() 311 : WebInputEvent(sizeof(WebMouseEvent)) 312 , button(ButtonNone) 313 , x(0) 314 , y(0) 315 , windowX(0) 316 , windowY(0) 317 , globalX(0) 318 , globalY(0) 319 , movementX(0) 320 , movementY(0) 321 , clickCount(0) 322 { 323 } 324 325 protected: WebMouseEvent(unsigned sizeParam)326 explicit WebMouseEvent(unsigned sizeParam) 327 : WebInputEvent(sizeParam) 328 , button(ButtonNone) 329 , x(0) 330 , y(0) 331 , windowX(0) 332 , windowY(0) 333 , globalX(0) 334 , globalY(0) 335 , movementX(0) 336 , movementY(0) 337 , clickCount(0) 338 { 339 } 340 }; 341 342 // WebMouseWheelEvent --------------------------------------------------------- 343 344 class WebMouseWheelEvent : public WebMouseEvent { 345 public: 346 enum Phase { 347 PhaseNone = 0, 348 PhaseBegan = 1 << 0, 349 PhaseStationary = 1 << 1, 350 PhaseChanged = 1 << 2, 351 PhaseEnded = 1 << 3, 352 PhaseCancelled = 1 << 4, 353 PhaseMayBegin = 1 << 5, 354 }; 355 356 float deltaX; 357 float deltaY; 358 float wheelTicksX; 359 float wheelTicksY; 360 361 float accelerationRatioX; 362 float accelerationRatioY; 363 364 // See comment at the top of the file for why an int is used here. 365 int scrollByPage; 366 367 // See comment at the top of the file for why an int is used here. 368 int hasPreciseScrollingDeltas; 369 Phase phase; 370 Phase momentumPhase; 371 372 // See comment at the top of the file for why an int is used here. 373 // Rubberbanding is an OSX visual effect. When a user scrolls the content 374 // area with a track pad, and the content area is already at its limit in 375 // the direction being scrolled, the entire content area is allowed to 376 // scroll slightly off screen, revealing a grey background. When the user 377 // lets go, the content area snaps back into place. Blink is responsible 378 // for this rubberbanding effect, but the embedder may wish to disable 379 // rubber banding in the left or right direction, if the scroll should have 380 // an alternate effect. The common case is that a scroll in the left or 381 // right directions causes a back or forwards navigation, respectively. 382 // 383 // These flags prevent rubber banding from starting in a given direction, 384 // but have no effect on an ongoing rubber banding. A rubber banding that 385 // started in the vertical direction is allowed to continue in the right 386 // direction, even if canRubberbandRight is 0. 387 int canRubberbandLeft; 388 int canRubberbandRight; 389 WebMouseWheelEvent()390 WebMouseWheelEvent() 391 : WebMouseEvent(sizeof(WebMouseWheelEvent)) 392 , deltaX(0.0f) 393 , deltaY(0.0f) 394 , wheelTicksX(0.0f) 395 , wheelTicksY(0.0f) 396 , accelerationRatioX(1.0f) 397 , accelerationRatioY(1.0f) 398 , scrollByPage(false) 399 , hasPreciseScrollingDeltas(false) 400 , phase(PhaseNone) 401 , momentumPhase(PhaseNone) 402 , canRubberbandLeft(true) 403 , canRubberbandRight(true) 404 { 405 } 406 }; 407 408 // WebGestureEvent -------------------------------------------------------------- 409 410 class WebGestureEvent : public WebInputEvent { 411 public: 412 int x; 413 int y; 414 int globalX; 415 int globalY; 416 WebGestureDevice sourceDevice; 417 418 union { 419 // Tap information must be set for GestureTap, GestureTapUnconfirmed, 420 // and GestureDoubleTap events. 421 struct { 422 int tapCount; 423 float width; 424 float height; 425 } tap; 426 427 struct { 428 float width; 429 float height; 430 } tapDown; 431 432 struct { 433 float width; 434 float height; 435 } showPress; 436 437 struct { 438 float width; 439 float height; 440 } longPress; 441 442 struct { 443 float firstFingerWidth; 444 float firstFingerHeight; 445 } twoFingerTap; 446 447 struct { 448 // Initial motion that triggered the scroll. 449 // May be redundant with deltaX/deltaY in the first scrollUpdate. 450 float deltaXHint; 451 float deltaYHint; 452 } scrollBegin; 453 454 struct { 455 float deltaX; 456 float deltaY; 457 float velocityX; 458 float velocityY; 459 } scrollUpdate; 460 461 struct { 462 float velocityX; 463 float velocityY; 464 } flingStart; 465 466 struct { 467 float scale; 468 } pinchUpdate; 469 } data; 470 WebGestureEvent()471 WebGestureEvent() 472 : WebInputEvent(sizeof(WebGestureEvent)) 473 , x(0) 474 , y(0) 475 , globalX(0) 476 , globalY(0) 477 { 478 memset(&data, 0, sizeof(data)); 479 } 480 }; 481 482 // WebTouchEvent -------------------------------------------------------------- 483 484 class WebTouchEvent : public WebInputEvent { 485 public: 486 // Maximum number of simultaneous touches supported on 487 // Ash/Aura. 488 enum { touchesLengthCap = 12 }; 489 490 unsigned touchesLength; 491 // List of all touches, regardless of state. 492 WebTouchPoint touches[touchesLengthCap]; 493 494 // Whether the event can be canceled (with preventDefault). If true then the browser 495 // must wait for an ACK for this event. If false then no ACK IPC is expected. 496 // See comment at the top for why an int is used here instead of a bool. 497 int cancelable; 498 WebTouchEvent()499 WebTouchEvent() 500 : WebInputEvent(sizeof(WebTouchEvent)) 501 , touchesLength(0) 502 , cancelable(true) 503 { 504 } 505 }; 506 507 #pragma pack(pop) 508 509 } // namespace blink 510 511 #endif 512