Home | History | Annotate | Download | only in ntp4
      1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 /**
      6  * @fileoverview Touch Handler. Class that handles all touch events and
      7  * uses them to interpret higher level gestures and behaviors. TouchEvent is a
      8  * built in mobile safari type:
      9  * http://developer.apple.com/safari/library/documentation/UserExperience/Reference/TouchEventClassReference/TouchEvent/TouchEvent.html.
     10  * This class is intended to work with all webkit browsers, tested on Chrome and
     11  * iOS.
     12  *
     13  * The following types of gestures are currently supported.  See the definition
     14  * of TouchHandler.EventType for details.
     15  *
     16  * Single Touch:
     17  *      This provides simple single-touch events.  Any secondary touch is
     18  *      ignored.
     19  *
     20  * Drag:
     21  *      A single touch followed by some movement. This behavior will handle all
     22  *      of the required events and report the properties of the drag to you
     23  *      while the touch is happening and at the end of the drag sequence. This
     24  *      behavior will NOT perform the actual dragging (redrawing the element)
     25  *      for you, this responsibility is left to the client code.
     26  *
     27  * Long press:
     28  *     When your element is touched and held without any drag occuring, the
     29  *     LONG_PRESS event will fire.
     30  */
     31 
     32 // Use an anonymous function to enable strict mode just for this file (which
     33 // will be concatenated with other files when embedded in Chrome)
     34 var TouchHandler = (function() {
     35   'use strict';
     36 
     37   /**
     38    * A TouchHandler attaches to an Element, listents for low-level touch (or
     39    * mouse) events and dispatching higher-level events on the element.
     40    * @param {!Element} element The element to listen on and fire events
     41    * for.
     42    * @constructor
     43    */
     44   function TouchHandler(element) {
     45     /**
     46      * @type {!Element}
     47      * @private
     48      */
     49     this.element_ = element;
     50 
     51     /**
     52      * The absolute sum of all touch y deltas.
     53      * @type {number}
     54      * @private
     55      */
     56     this.totalMoveY_ = 0;
     57 
     58     /**
     59      * The absolute sum of all touch x deltas.
     60      * @type {number}
     61      * @private
     62      */
     63     this.totalMoveX_ = 0;
     64 
     65     /**
     66      * An array of tuples where the first item is the horizontal component of a
     67      * recent relevant touch and the second item is the touch's time stamp. Old
     68      * touches are removed based on the max tracking time and when direction
     69      * changes.
     70       * @type {!Array.<number>}
     71       * @private
     72       */
     73     this.recentTouchesX_ = [];
     74 
     75     /**
     76      * An array of tuples where the first item is the vertical component of a
     77      * recent relevant touch and the second item is the touch's time stamp. Old
     78      * touches are removed based on the max tracking time and when direction
     79      * changes.
     80      * @type {!Array.<number>}
     81      * @private
     82      */
     83     this.recentTouchesY_ = [];
     84 
     85     /**
     86      * Used to keep track of all events we subscribe to so we can easily clean
     87      * up
     88      * @type {EventTracker}
     89      * @private
     90      */
     91     this.events_ = new EventTracker();
     92   }
     93 
     94 
     95   /**
     96    * DOM Events that may be fired by the TouchHandler at the element
     97    */
     98   TouchHandler.EventType = {
     99     // Fired whenever the element is touched as the only touch to the device.
    100     // enableDrag defaults to false, set to true to permit dragging.
    101     TOUCH_START: 'touchHandler:touch_start',
    102 
    103     // Fired when an element is held for a period of time.  Prevents dragging
    104     // from occuring (even if enableDrag was set to true).
    105     LONG_PRESS: 'touchHandler:long_press',
    106 
    107     // If enableDrag was set to true at TOUCH_START, DRAG_START will fire when
    108     // the touch first moves sufficient distance.  enableDrag is set to true but
    109     // can be reset to false to cancel the drag.
    110     DRAG_START: 'touchHandler:drag_start',
    111 
    112     // If enableDrag was true after DRAG_START, DRAG_MOVE will fire whenever the
    113     // touch is moved.
    114     DRAG_MOVE: 'touchHandler:drag_move',
    115 
    116     // Fired just before TOUCH_END when a drag is released.  Correlates 1:1 with
    117     // a DRAG_START.
    118     DRAG_END: 'touchHandler:drag_end',
    119 
    120     // Fired whenever a touch that is being tracked has been released.
    121     // Correlates 1:1 with a TOUCH_START.
    122     TOUCH_END: 'touchHandler:touch_end'
    123   };
    124 
    125 
    126   /**
    127    * The type of event sent by TouchHandler
    128    * @constructor
    129    * @param {string} type The type of event (one of Grabber.EventType).
    130    * @param {boolean} bubbles Whether or not the event should bubble.
    131    * @param {number} clientX The X location of the touch.
    132    * @param {number} clientY The Y location of the touch.
    133    * @param {!Element} touchedElement The element at the current location of the
    134    *        touch.
    135    */
    136   TouchHandler.Event = function(type, bubbles, clientX, clientY,
    137       touchedElement) {
    138     var event = document.createEvent('Event');
    139     event.initEvent(type, bubbles, true);
    140     event.__proto__ = TouchHandler.Event.prototype;
    141 
    142     /**
    143      * The X location of the touch affected
    144      * @type {number}
    145      */
    146     event.clientX = clientX;
    147 
    148     /**
    149      * The Y location of the touch affected
    150      * @type {number}
    151      */
    152     event.clientY = clientY;
    153 
    154     /**
    155      * The element at the current location of the touch.
    156      * @type {!Element}
    157      */
    158     event.touchedElement = touchedElement;
    159 
    160     return event;
    161   };
    162 
    163   TouchHandler.Event.prototype = {
    164     __proto__: Event.prototype,
    165 
    166     /**
    167      * For TOUCH_START and DRAG START events, set to true to enable dragging or
    168      * false to disable dragging.
    169      * @type {boolean|undefined}
    170      */
    171     enableDrag: undefined,
    172 
    173     /**
    174      * For DRAG events, provides the horizontal component of the
    175      * drag delta. Drag delta is defined as the delta of the start touch
    176      * position and the current drag position.
    177      * @type {number|undefined}
    178      */
    179     dragDeltaX: undefined,
    180 
    181     /**
    182      * For DRAG events, provides the vertical component of the
    183      * drag delta.
    184      * @type {number|undefined}
    185      */
    186     dragDeltaY: undefined
    187   };
    188 
    189   /**
    190    * Minimum movement of touch required to be considered a drag.
    191    * @type {number}
    192    * @private
    193    */
    194   TouchHandler.MIN_TRACKING_FOR_DRAG_ = 8;
    195 
    196 
    197   /**
    198    * The maximum number of ms to track a touch event. After an event is older
    199    * than this value, it will be ignored in velocity calculations.
    200    * @type {number}
    201    * @private
    202    */
    203   TouchHandler.MAX_TRACKING_TIME_ = 250;
    204 
    205 
    206   /**
    207    * The maximum number of touches to track.
    208    * @type {number}
    209    * @private
    210    */
    211   TouchHandler.MAX_TRACKING_TOUCHES_ = 5;
    212 
    213 
    214   /**
    215    * The maximum velocity to return, in pixels per millisecond, that is used
    216    * to guard against errors in calculating end velocity of a drag. This is a
    217    * very fast drag velocity.
    218    * @type {number}
    219    * @private
    220    */
    221   TouchHandler.MAXIMUM_VELOCITY_ = 5;
    222 
    223 
    224   /**
    225    * The velocity to return, in pixel per millisecond, when the time stamps on
    226    * the events are erroneous. The browser can return bad time stamps if the
    227    * thread is blocked for the duration of the drag. This is a low velocity to
    228    * prevent the content from moving quickly after a slow drag. It is less
    229    * jarring if the content moves slowly after a fast drag.
    230    * @type {number}
    231    * @private
    232    */
    233   TouchHandler.VELOCITY_FOR_INCORRECT_EVENTS_ = 1;
    234 
    235   /**
    236    * The time, in milliseconds, that a touch must be held to be considered
    237    * 'long'.
    238    * @type {number}
    239    * @private
    240    */
    241   TouchHandler.TIME_FOR_LONG_PRESS_ = 500;
    242 
    243   TouchHandler.prototype = {
    244     /**
    245      * If defined, the identifer of the single touch that is active.  Note that
    246      * 0 is a valid touch identifier - it should not be treated equivalently to
    247      * undefined.
    248      * @type {number|undefined}
    249      * @private
    250      */
    251     activeTouch_: undefined,
    252 
    253     /**
    254      * @type {boolean|undefined}
    255      * @private
    256      */
    257     tracking_: undefined,
    258 
    259     /**
    260      * @type {number|undefined}
    261      * @private
    262      */
    263     startTouchX_: undefined,
    264 
    265     /**
    266      * @type {number|undefined}
    267      * @private
    268      */
    269     startTouchY_: undefined,
    270 
    271     /**
    272      * @type {number|undefined}
    273      * @private
    274      */
    275     endTouchX_: undefined,
    276 
    277     /**
    278      * @type {number|undefined}
    279      * @private
    280      */
    281     endTouchY_: undefined,
    282 
    283     /**
    284      * Time of the touchstart event.
    285      * @type {number|undefined}
    286      * @private
    287      */
    288     startTime_: undefined,
    289 
    290     /**
    291      * The time of the touchend event.
    292      * @type {number|undefined}
    293      * @private
    294      */
    295     endTime_: undefined,
    296 
    297     /**
    298      * @type {number|undefined}
    299      * @private
    300      */
    301     lastTouchX_: undefined,
    302 
    303     /**
    304      * @type {number|undefined}
    305      * @private
    306      */
    307     lastTouchY_: undefined,
    308 
    309     /**
    310      * @type {number|undefined}
    311      * @private
    312      */
    313     lastMoveX_: undefined,
    314 
    315     /**
    316      * @type {number|undefined}
    317      * @private
    318      */
    319     lastMoveY_: undefined,
    320 
    321     /**
    322      * @type {number|undefined}
    323      * @private
    324      */
    325     longPressTimeout_: undefined,
    326 
    327     /**
    328      * If defined and true, the next click event should be swallowed
    329      * @type {boolean|undefined}
    330      * @private
    331      */
    332     swallowNextClick_: undefined,
    333 
    334     /**
    335      * Start listenting for events.
    336      * @param {boolean=} opt_capture True if the TouchHandler should listen to
    337      *      during the capture phase.
    338      */
    339     enable: function(opt_capture) {
    340       var capture = !!opt_capture;
    341 
    342       // Just listen to start events for now. When a touch is occuring we'll
    343       // want to be subscribed to move and end events on the document, but we
    344       // don't want to incur the cost of lots of no-op handlers on the document.
    345       this.events_.add(this.element_, 'touchstart', this.onStart_.bind(this),
    346                        capture);
    347       this.events_.add(this.element_, 'mousedown',
    348                        this.mouseToTouchCallback_(this.onStart_.bind(this)),
    349                        capture);
    350 
    351       // If the element is long-pressed, we may need to swallow a click
    352       this.events_.add(this.element_, 'click', this.onClick_.bind(this), true);
    353     },
    354 
    355     /**
    356      * Stop listening to all events.
    357      */
    358     disable: function() {
    359       this.stopTouching_();
    360       this.events_.removeAll();
    361     },
    362 
    363     /**
    364      * Wraps a callback with translations of mouse events to touch events.
    365      * NOTE: These types really should be function(Event) but then we couldn't
    366      * use this with bind (which operates on any type of function).  Doesn't
    367      * JSDoc support some sort of polymorphic types?
    368      * @param {Function} callback The event callback.
    369      * @return {Function} The wrapping callback.
    370      * @private
    371      */
    372     mouseToTouchCallback_: function(callback) {
    373       return function(e) {
    374         // Note that there may be synthesizes mouse events caused by touch
    375         // events (a mouseDown after a touch-click).  We leave it up to the
    376         // client to worry about this if it matters to them (typically a short
    377         // mouseDown/mouseUp without a click is no big problem and it's not
    378         // obvious how we identify such synthesized events in a general way).
    379         var touch = {
    380           // any fixed value will do for the identifier - there will only
    381           // ever be a single active 'touch' when using the mouse.
    382           identifier: 0,
    383           clientX: e.clientX,
    384           clientY: e.clientY,
    385           target: e.target
    386         };
    387         e.touches = [];
    388         e.targetTouches = [];
    389         e.changedTouches = [touch];
    390         if (e.type != 'mouseup') {
    391           e.touches[0] = touch;
    392           e.targetTouches[0] = touch;
    393         }
    394         callback(e);
    395       };
    396     },
    397 
    398     /**
    399      * Begin tracking the touchable element, it is eligible for dragging.
    400      * @private
    401      */
    402     beginTracking_: function() {
    403       this.tracking_ = true;
    404     },
    405 
    406     /**
    407      * Stop tracking the touchable element, it is no longer dragging.
    408      * @private
    409      */
    410     endTracking_: function() {
    411       this.tracking_ = false;
    412       this.dragging_ = false;
    413       this.totalMoveY_ = 0;
    414       this.totalMoveX_ = 0;
    415     },
    416 
    417     /**
    418      * Reset the touchable element as if we never saw the touchStart
    419      * Doesn't dispatch any end events - be careful of existing listeners.
    420      */
    421     cancelTouch: function() {
    422       this.stopTouching_();
    423       this.endTracking_();
    424       // If clients needed to be aware of this, we could fire a cancel event
    425       // here.
    426     },
    427 
    428     /**
    429      * Record that touching has stopped
    430      * @private
    431      */
    432     stopTouching_: function() {
    433       // Mark as no longer being touched
    434       this.activeTouch_ = undefined;
    435 
    436       // If we're waiting for a long press, stop
    437       window.clearTimeout(this.longPressTimeout_);
    438 
    439       // Stop listening for move/end events until there's another touch.
    440       // We don't want to leave handlers piled up on the document.
    441       // Note that there's no harm in removing handlers that weren't added, so
    442       // rather than track whether we're using mouse or touch we do both.
    443       this.events_.remove(document, 'touchmove');
    444       this.events_.remove(document, 'touchend');
    445       this.events_.remove(document, 'touchcancel');
    446       this.events_.remove(document, 'mousemove');
    447       this.events_.remove(document, 'mouseup');
    448     },
    449 
    450     /**
    451      * Touch start handler.
    452      * @param {!TouchEvent} e The touchstart event.
    453      * @private
    454      */
    455     onStart_: function(e) {
    456       // Only process single touches.  If there is already a touch happening, or
    457       // two simultaneous touches then just ignore them.
    458       if (e.touches.length > 1)
    459         // Note that we could cancel an active touch here.  That would make
    460         // simultaneous touch behave similar to near-simultaneous. However, if
    461         // the user is dragging something, an accidental second touch could be
    462         // quite disruptive if it cancelled their drag.  Better to just ignore
    463         // it.
    464         return;
    465 
    466       // It's still possible there could be an active "touch" if the user is
    467       // simultaneously using a mouse and a touch input.
    468       if (this.activeTouch_ !== undefined)
    469         return;
    470 
    471       var touch = e.targetTouches[0];
    472       this.activeTouch_ = touch.identifier;
    473 
    474       // We've just started touching so shouldn't swallow any upcoming click
    475       if (this.swallowNextClick_)
    476         this.swallowNextClick_ = false;
    477 
    478       // Sign up for end/cancel notifications for this touch.
    479       // Note that we do this on the document so that even if the user drags
    480       // their finger off the element, we'll still know what they're doing.
    481       if (e.type == 'mousedown') {
    482         this.events_.add(document, 'mouseup',
    483             this.mouseToTouchCallback_(this.onEnd_.bind(this)), false);
    484       } else {
    485         this.events_.add(document, 'touchend', this.onEnd_.bind(this), false);
    486         this.events_.add(document, 'touchcancel', this.onEnd_.bind(this),
    487             false);
    488       }
    489 
    490       // This timeout is cleared on touchEnd and onDrag
    491       // If we invoke the function then we have a real long press
    492       window.clearTimeout(this.longPressTimeout_);
    493       this.longPressTimeout_ = window.setTimeout(
    494           this.onLongPress_.bind(this),
    495           TouchHandler.TIME_FOR_LONG_PRESS_);
    496 
    497       // Dispatch the TOUCH_START event
    498       if (!this.dispatchEvent_(TouchHandler.EventType.TOUCH_START, touch))
    499         // Dragging was not enabled, nothing more to do
    500         return;
    501 
    502       // We want dragging notifications
    503       if (e.type == 'mousedown') {
    504         this.events_.add(document, 'mousemove',
    505             this.mouseToTouchCallback_(this.onMove_.bind(this)), false);
    506       } else {
    507         this.events_.add(document, 'touchmove', this.onMove_.bind(this), false);
    508       }
    509 
    510       this.startTouchX_ = this.lastTouchX_ = touch.clientX;
    511       this.startTouchY_ = this.lastTouchY_ = touch.clientY;
    512       this.startTime_ = e.timeStamp;
    513 
    514       this.recentTouchesX_ = [];
    515       this.recentTouchesY_ = [];
    516       this.recentTouchesX_.push(touch.clientX, e.timeStamp);
    517       this.recentTouchesY_.push(touch.clientY, e.timeStamp);
    518 
    519       this.beginTracking_();
    520     },
    521 
    522     /**
    523      * Given a list of Touches, find the one matching our activeTouch
    524      * identifier. Note that Chrome currently always uses 0 as the identifier.
    525      * In that case we'll end up always choosing the first element in the list.
    526      * @param {TouchList} touches The list of Touch objects to search.
    527      * @return {!Touch|undefined} The touch matching our active ID if any.
    528      * @private
    529      */
    530     findActiveTouch_: function(touches) {
    531       assert(this.activeTouch_ !== undefined, 'Expecting an active touch');
    532       // A TouchList isn't actually an array, so we shouldn't use
    533       // Array.prototype.filter/some, etc.
    534       for (var i = 0; i < touches.length; i++) {
    535         if (touches[i].identifier == this.activeTouch_)
    536           return touches[i];
    537       }
    538       return undefined;
    539     },
    540 
    541     /**
    542      * Touch move handler.
    543      * @param {!TouchEvent} e The touchmove event.
    544      * @private
    545      */
    546     onMove_: function(e) {
    547       if (!this.tracking_)
    548         return;
    549 
    550       // Our active touch should always be in the list of touches still active
    551       assert(this.findActiveTouch_(e.touches), 'Missing touchEnd');
    552 
    553       var that = this;
    554       var touch = this.findActiveTouch_(e.changedTouches);
    555       if (!touch)
    556         return;
    557 
    558       var clientX = touch.clientX;
    559       var clientY = touch.clientY;
    560 
    561       var moveX = this.lastTouchX_ - clientX;
    562       var moveY = this.lastTouchY_ - clientY;
    563       this.totalMoveX_ += Math.abs(moveX);
    564       this.totalMoveY_ += Math.abs(moveY);
    565       this.lastTouchX_ = clientX;
    566       this.lastTouchY_ = clientY;
    567 
    568       if (!this.dragging_ && (this.totalMoveY_ >
    569           TouchHandler.MIN_TRACKING_FOR_DRAG_ ||
    570           this.totalMoveX_ >
    571           TouchHandler.MIN_TRACKING_FOR_DRAG_)) {
    572         // If we're waiting for a long press, stop
    573         window.clearTimeout(this.longPressTimeout_);
    574 
    575         // Dispatch the DRAG_START event and record whether dragging should be
    576         // allowed or not.  Note that this relies on the current value of
    577         // startTouchX/Y - handlers may use the initial drag delta to determine
    578         // if dragging should be permitted.
    579         this.dragging_ = this.dispatchEvent_(
    580             TouchHandler.EventType.DRAG_START, touch);
    581 
    582         if (this.dragging_) {
    583           // Update the start position here so that drag deltas have better
    584           // values but don't touch the recent positions so that velocity
    585           // calculations can still use touchstart position in the time and
    586           // distance delta.
    587           this.startTouchX_ = clientX;
    588           this.startTouchY_ = clientY;
    589           this.startTime_ = e.timeStamp;
    590         } else {
    591           this.endTracking_();
    592         }
    593       }
    594 
    595       if (this.dragging_) {
    596         this.dispatchEvent_(TouchHandler.EventType.DRAG_MOVE, touch);
    597 
    598         this.removeTouchesInWrongDirection_(this.recentTouchesX_,
    599             this.lastMoveX_, moveX);
    600         this.removeTouchesInWrongDirection_(this.recentTouchesY_,
    601             this.lastMoveY_, moveY);
    602         this.removeOldTouches_(this.recentTouchesX_, e.timeStamp);
    603         this.removeOldTouches_(this.recentTouchesY_, e.timeStamp);
    604         this.recentTouchesX_.push(clientX, e.timeStamp);
    605         this.recentTouchesY_.push(clientY, e.timeStamp);
    606       }
    607 
    608       this.lastMoveX_ = moveX;
    609       this.lastMoveY_ = moveY;
    610     },
    611 
    612     /**
    613      * Filters the provided recent touches array to remove all touches except
    614      * the last if the move direction has changed.
    615      * @param {!Array.<number>} recentTouches An array of tuples where the first
    616      *     item is the x or y component of the recent touch and the second item
    617      *     is the touch time stamp.
    618      * @param {number|undefined} lastMove The x or y component of the previous
    619      *     move.
    620      * @param {number} recentMove The x or y component of the most recent move.
    621      * @private
    622      */
    623     removeTouchesInWrongDirection_: function(recentTouches, lastMove,
    624         recentMove) {
    625       if (lastMove && recentMove && recentTouches.length > 2 &&
    626           (lastMove > 0 ^ recentMove > 0)) {
    627         recentTouches.splice(0, recentTouches.length - 2);
    628       }
    629     },
    630 
    631     /**
    632      * Filters the provided recent touches array to remove all touches older
    633      * than the max tracking time or the 5th most recent touch.
    634      * @param {!Array.<number>} recentTouches An array of tuples where the first
    635      *     item is the x or y component of the recent touch and the second item
    636      *     is the touch time stamp.
    637      * @param {number} recentTime The time of the most recent event.
    638      * @private
    639      */
    640     removeOldTouches_: function(recentTouches, recentTime) {
    641       while (recentTouches.length && recentTime - recentTouches[1] >
    642           TouchHandler.MAX_TRACKING_TIME_ ||
    643           recentTouches.length >
    644               TouchHandler.MAX_TRACKING_TOUCHES_ * 2) {
    645         recentTouches.splice(0, 2);
    646       }
    647     },
    648 
    649     /**
    650      * Touch end handler.
    651      * @param {!TouchEvent} e The touchend event.
    652      * @private
    653      */
    654     onEnd_: function(e) {
    655       var that = this;
    656       assert(this.activeTouch_ !== undefined, 'Expect to already be touching');
    657 
    658       // If the touch we're tracking isn't changing here, ignore this touch end.
    659       var touch = this.findActiveTouch_(e.changedTouches);
    660       if (!touch) {
    661         // In most cases, our active touch will be in the 'touches' collection,
    662         // but we can't assert that because occasionally two touchend events can
    663         // occur at almost the same time with both having empty 'touches' lists.
    664         // I.e., 'touches' seems like it can be a bit more up-to-date than the
    665         // current event.
    666         return;
    667       }
    668 
    669       // This is touchEnd for the touch we're monitoring
    670       assert(!this.findActiveTouch_(e.touches),
    671              'Touch ended also still active');
    672 
    673       // Indicate that touching has finished
    674       this.stopTouching_();
    675 
    676       if (this.tracking_) {
    677         var clientX = touch.clientX;
    678         var clientY = touch.clientY;
    679 
    680         if (this.dragging_) {
    681           this.endTime_ = e.timeStamp;
    682           this.endTouchX_ = clientX;
    683           this.endTouchY_ = clientY;
    684 
    685           this.removeOldTouches_(this.recentTouchesX_, e.timeStamp);
    686           this.removeOldTouches_(this.recentTouchesY_, e.timeStamp);
    687 
    688           this.dispatchEvent_(TouchHandler.EventType.DRAG_END, touch);
    689 
    690           // Note that in some situations we can get a click event here as well.
    691           // For now this isn't a problem, but we may want to consider having
    692           // some logic that hides clicks that appear to be caused by a touchEnd
    693           // used for dragging.
    694         }
    695 
    696         this.endTracking_();
    697       }
    698 
    699       // Note that we dispatch the touchEnd event last so that events at
    700       // different levels of semantics nest nicely (similar to how DOM
    701       // drag-and-drop events are nested inside of the mouse events that trigger
    702       // them).
    703       this.dispatchEvent_(TouchHandler.EventType.TOUCH_END, touch);
    704     },
    705 
    706     /**
    707      * Get end velocity of the drag. This method is specific to drag behavior,
    708      * so if touch behavior and drag behavior is split then this should go with
    709      * drag behavior. End velocity is defined as deltaXY / deltaTime where
    710      * deltaXY is the difference between endPosition and the oldest recent
    711      * position, and deltaTime is the difference between endTime and the oldest
    712      * recent time stamp.
    713      * @return {Object} The x and y velocity.
    714      */
    715     getEndVelocity: function() {
    716       // Note that we could move velocity to just be an end-event parameter.
    717       var velocityX = this.recentTouchesX_.length ?
    718           (this.endTouchX_ - this.recentTouchesX_[0]) /
    719           (this.endTime_ - this.recentTouchesX_[1]) : 0;
    720       var velocityY = this.recentTouchesY_.length ?
    721           (this.endTouchY_ - this.recentTouchesY_[0]) /
    722           (this.endTime_ - this.recentTouchesY_[1]) : 0;
    723 
    724       velocityX = this.correctVelocity_(velocityX);
    725       velocityY = this.correctVelocity_(velocityY);
    726 
    727       return {
    728         x: velocityX,
    729         y: velocityY
    730       };
    731     },
    732 
    733     /**
    734      * Correct erroneous velocities by capping the velocity if we think it's too
    735      * high, or setting it to a default velocity if know that the event data is
    736      * bad.
    737      * @param {number} velocity The x or y velocity component.
    738      * @return {number} The corrected velocity.
    739      * @private
    740      */
    741     correctVelocity_: function(velocity) {
    742       var absVelocity = Math.abs(velocity);
    743 
    744       // We add to recent touches for each touchstart and touchmove. If we have
    745       // fewer than 3 touches (6 entries), we assume that the thread was blocked
    746       // for the duration of the drag and we received events in quick succession
    747       // with the wrong time stamps.
    748       if (absVelocity > TouchHandler.MAXIMUM_VELOCITY_) {
    749         absVelocity = this.recentTouchesY_.length < 3 ?
    750             TouchHandler.VELOCITY_FOR_INCORRECT_EVENTS_ :
    751                 TouchHandler.MAXIMUM_VELOCITY_;
    752       }
    753       return absVelocity * (velocity < 0 ? -1 : 1);
    754     },
    755 
    756     /**
    757      * Handler when an element has been pressed for a long time
    758      * @private
    759      */
    760     onLongPress_: function() {
    761       // Swallow any click that occurs on this element without an intervening
    762       // touch start event.  This simple click-busting technique should be
    763       // sufficient here since a real click should have a touchstart first.
    764       this.swallowNextClick_ = true;
    765 
    766       // Dispatch to the LONG_PRESS
    767       this.dispatchEventXY_(TouchHandler.EventType.LONG_PRESS, this.element_,
    768           this.startTouchX_, this.startTouchY_);
    769     },
    770 
    771     /**
    772      * Click handler - used to swallow clicks after a long-press
    773      * @param {!Event} e The click event.
    774      * @private
    775      */
    776     onClick_: function(e) {
    777       if (this.swallowNextClick_) {
    778         e.preventDefault();
    779         e.stopPropagation();
    780         this.swallowNextClick_ = false;
    781       }
    782     },
    783 
    784     /**
    785      * Dispatch a TouchHandler event to the element
    786      * @param {string} eventType The event to dispatch.
    787      * @param {Touch} touch The touch triggering this event.
    788      * @return {boolean|undefined} The value of enableDrag after dispatching
    789      *         the event.
    790      * @private
    791      */
    792     dispatchEvent_: function(eventType, touch) {
    793 
    794       // Determine which element was touched.  For mouse events, this is always
    795       // the event/touch target.  But for touch events, the target is always the
    796       // target of the touchstart (and it's unlikely we can change this
    797       // since the common implementation of touch dragging relies on it). Since
    798       // touch is our primary scenario (which we want to emulate with mouse),
    799       // we'll treat both cases the same and not depend on the target.
    800       var touchedElement;
    801       if (eventType == TouchHandler.EventType.TOUCH_START) {
    802         touchedElement = touch.target;
    803       } else {
    804         touchedElement = this.element_.ownerDocument.
    805             elementFromPoint(touch.clientX, touch.clientY);
    806       }
    807 
    808       return this.dispatchEventXY_(eventType, touchedElement, touch.clientX,
    809           touch.clientY);
    810     },
    811 
    812     /**
    813      * Dispatch a TouchHandler event to the element
    814      * @param {string} eventType The event to dispatch.
    815        @param {number} clientX The X location for the event.
    816        @param {number} clientY The Y location for the event.
    817      * @return {boolean|undefined} The value of enableDrag after dispatching
    818      *         the event.
    819      * @private
    820      */
    821     dispatchEventXY_: function(eventType, touchedElement, clientX, clientY) {
    822       var isDrag = (eventType == TouchHandler.EventType.DRAG_START ||
    823           eventType == TouchHandler.EventType.DRAG_MOVE ||
    824           eventType == TouchHandler.EventType.DRAG_END);
    825 
    826       // Drag events don't bubble - we're really just dragging the element,
    827       // not affecting its parent at all.
    828       var bubbles = !isDrag;
    829 
    830       var event = new TouchHandler.Event(eventType, bubbles, clientX, clientY,
    831           touchedElement);
    832 
    833       // Set enableDrag when it can be overridden
    834       if (eventType == TouchHandler.EventType.TOUCH_START)
    835         event.enableDrag = false;
    836       else if (eventType == TouchHandler.EventType.DRAG_START)
    837         event.enableDrag = true;
    838 
    839       if (isDrag) {
    840         event.dragDeltaX = clientX - this.startTouchX_;
    841         event.dragDeltaY = clientY - this.startTouchY_;
    842       }
    843 
    844       this.element_.dispatchEvent(event);
    845       return event.enableDrag;
    846     }
    847   };
    848 
    849   return TouchHandler;
    850 })();
    851