@ -1,7 +1,7 @@
/ * *
/ * *
* @ preserve FastClick : polyfill to remove click delays on browsers with touch UIs .
* @ preserve FastClick : polyfill to remove click delays on browsers with touch UIs .
*
*
* @ version 0.6. 11
* @ version 1.0. 0
* @ codingstandard ftlabs - jsv2
* @ codingstandard ftlabs - jsv2
* @ copyright The Financial Times Limited [ All Rights Reserved ]
* @ copyright The Financial Times Limited [ All Rights Reserved ]
* @ license MIT License ( see LICENSE . txt )
* @ license MIT License ( see LICENSE . txt )
@ -19,7 +19,7 @@
* /
* /
function FastClick ( layer ) {
function FastClick ( layer ) {
'use strict' ;
'use strict' ;
var oldOnClick , self = this ;
var oldOnClick ;
/ * *
/ * *
@ -85,44 +85,27 @@ function FastClick(layer) {
* /
* /
this . layer = layer ;
this . layer = layer ;
if ( ! layer || ! layer . nodeType ) {
throw new TypeError ( 'Layer must be a document node' ) ;
}
/** @type function() */
this . onClick = function ( ) { return FastClick . prototype . onClick . apply ( self , arguments ) ; } ;
/** @type function() */
this . onMouse = function ( ) { return FastClick . prototype . onMouse . apply ( self , arguments ) ; } ;
/** @type function() */
this . onTouchStart = function ( ) { return FastClick . prototype . onTouchStart . apply ( self , arguments ) ; } ;
/** @type function() */
this . onTouchMove = function ( ) { return FastClick . prototype . onTouchMove . apply ( self , arguments ) ; } ;
/** @type function() */
this . onTouchEnd = function ( ) { return FastClick . prototype . onTouchEnd . apply ( self , arguments ) ; } ;
/** @type function() */
this . onTouchCancel = function ( ) { return FastClick . prototype . onTouchCancel . apply ( self , arguments ) ; } ;
if ( FastClick . notNeeded ( layer ) ) {
if ( FastClick . notNeeded ( layer ) ) {
return ;
return ;
}
}
// Some old versions of Android don't have Function.prototype.bind
function bind ( method , context ) {
return function ( ) { return method . apply ( context , arguments ) ; } ;
}
// Set up event handlers as required
// Set up event handlers as required
if ( this . deviceIsAndroid ) {
if ( deviceIsAndroid ) {
layer . addEventListener ( 'mouseover' , this . onMouse , true ) ;
layer . addEventListener ( 'mouseover' , bind ( this . onMouse , this ) , true ) ;
layer . addEventListener ( 'mousedown' , this . onMouse , true ) ;
layer . addEventListener ( 'mousedown' , bind ( this . onMouse , this ) , true ) ;
layer . addEventListener ( 'mouseup' , this . onMouse , true ) ;
layer . addEventListener ( 'mouseup' , bind ( this . onMouse , this ) , true ) ;
}
}
layer . addEventListener ( 'click' , this . onClick , true ) ;
layer . addEventListener ( 'click' , bind ( this . onClick , this ) , true ) ;
layer . addEventListener ( 'touchstart' , this . onTouchStart , false ) ;
layer . addEventListener ( 'touchstart' , bind ( this . onTouchStart , this ) , false ) ;
layer . addEventListener ( 'touchmove' , this . onTouchMove , false ) ;
layer . addEventListener ( 'touchmove' , bind ( this . onTouchMove , this ) , false ) ;
layer . addEventListener ( 'touchend' , this . onTouchEnd , false ) ;
layer . addEventListener ( 'touchend' , bind ( this . onTouchEnd , this ) , false ) ;
layer . addEventListener ( 'touchcancel' , this . onTouchCancel , false ) ;
layer . addEventListener ( 'touchcancel' , bind ( this . onTouchCancel , this ) , false ) ;
// Hack is required for browsers that don't support Event#stopImmediatePropagation (e.g. Android 2)
// Hack is required for browsers that don't support Event#stopImmediatePropagation (e.g. Android 2)
// which is how FastClick normally stops click events bubbling to callbacks registered on the FastClick
// which is how FastClick normally stops click events bubbling to callbacks registered on the FastClick
@ -172,7 +155,7 @@ function FastClick(layer) {
*
*
* @ type boolean
* @ type boolean
* /
* /
FastClick . prototype . deviceIsAndroid = navigator . userAgent . indexOf ( 'Android' ) > 0 ;
var deviceIsAndroid = navigator . userAgent . indexOf ( 'Android' ) > 0 ;
/ * *
/ * *
@ -180,7 +163,7 @@ FastClick.prototype.deviceIsAndroid = navigator.userAgent.indexOf('Android') > 0
*
*
* @ type boolean
* @ type boolean
* /
* /
FastClick . prototype . deviceIsIOS = /iP(ad|hone|od)/ . test ( navigator . userAgent ) ;
var deviceIsIOS = /iP(ad|hone|od)/ . test ( navigator . userAgent ) ;
/ * *
/ * *
@ -188,7 +171,7 @@ FastClick.prototype.deviceIsIOS = /iP(ad|hone|od)/.test(navigator.userAgent);
*
*
* @ type boolean
* @ type boolean
* /
* /
FastClick . prototype . deviceIsIOS4 = FastClick . prototype . deviceIsIOS && ( /OS 4_\d(_\d)?/ ) . test ( navigator . userAgent ) ;
var deviceIsIOS4 = deviceIsIOS && ( /OS 4_\d(_\d)?/ ) . test ( navigator . userAgent ) ;
/ * *
/ * *
@ -196,7 +179,7 @@ FastClick.prototype.deviceIsIOS4 = FastClick.prototype.deviceIsIOS && (/OS 4_\d(
*
*
* @ type boolean
* @ type boolean
* /
* /
FastClick . prototype . deviceIsIOSWithBadTarget = FastClick . prototype . deviceIsIOS && ( /OS ([6-9]|\d{2})_\d/ ) . test ( navigator . userAgent ) ;
var deviceIsIOSWithBadTarget = deviceIsIOS && ( /OS ([6-9]|\d{2})_\d/ ) . test ( navigator . userAgent ) ;
/ * *
/ * *
@ -221,7 +204,7 @@ FastClick.prototype.needsClick = function(target) {
case 'input' :
case 'input' :
// File inputs need real clicks on iOS 6 due to a browser bug (issue #68)
// File inputs need real clicks on iOS 6 due to a browser bug (issue #68)
if ( ( this . deviceIsIOS && target . type === 'file' ) || target . disabled ) {
if ( ( deviceIsIOS && target . type === 'file' ) || target . disabled ) {
return true ;
return true ;
}
}
@ -247,7 +230,7 @@ FastClick.prototype.needsFocus = function(target) {
case 'textarea' :
case 'textarea' :
return true ;
return true ;
case 'select' :
case 'select' :
return ! this . deviceIsAndroid ;
return ! deviceIsAndroid ;
case 'input' :
case 'input' :
switch ( target . type ) {
switch ( target . type ) {
case 'button' :
case 'button' :
@ -295,7 +278,7 @@ FastClick.prototype.determineEventType = function(targetElement) {
'use strict' ;
'use strict' ;
//Issue #159: Android Chrome Select Box does not open with a synthetic click event
//Issue #159: Android Chrome Select Box does not open with a synthetic click event
if ( this . deviceIsAndroid && targetElement . tagName . toLowerCase ( ) === 'select' ) {
if ( deviceIsAndroid && targetElement . tagName . toLowerCase ( ) === 'select' ) {
return 'mousedown' ;
return 'mousedown' ;
}
}
@ -311,7 +294,7 @@ FastClick.prototype.focus = function(targetElement) {
var length ;
var length ;
// Issue #160: on iOS 7, some input elements (e.g. date datetime) throw a vague TypeError on setSelectionRange. These elements don't have an integer value for the selectionStart and selectionEnd properties, but unfortunately that can't be used for detection because accessing the properties also throws a TypeError. Just check the type instead. Filed as Apple bug #15122724.
// Issue #160: on iOS 7, some input elements (e.g. date datetime) throw a vague TypeError on setSelectionRange. These elements don't have an integer value for the selectionStart and selectionEnd properties, but unfortunately that can't be used for detection because accessing the properties also throws a TypeError. Just check the type instead. Filed as Apple bug #15122724.
if ( this . deviceIsIOS && targetElement . setSelectionRange && targetElement . type . indexOf ( 'date' ) !== 0 && targetElement . type !== 'time' ) {
if ( deviceIsIOS && targetElement . setSelectionRange && targetElement . type . indexOf ( 'date' ) !== 0 && targetElement . type !== 'time' ) {
length = targetElement . value . length ;
length = targetElement . value . length ;
targetElement . setSelectionRange ( length , length ) ;
targetElement . setSelectionRange ( length , length ) ;
} else {
} else {
@ -387,7 +370,7 @@ FastClick.prototype.onTouchStart = function(event) {
targetElement = this . getTargetElementFromEventTarget ( event . target ) ;
targetElement = this . getTargetElementFromEventTarget ( event . target ) ;
touch = event . targetTouches [ 0 ] ;
touch = event . targetTouches [ 0 ] ;
if ( this . deviceIsIOS ) {
if ( deviceIsIOS ) {
// Only trusted events will deselect text on iOS (issue #49)
// Only trusted events will deselect text on iOS (issue #49)
selection = window . getSelection ( ) ;
selection = window . getSelection ( ) ;
@ -395,7 +378,7 @@ FastClick.prototype.onTouchStart = function(event) {
return true ;
return true ;
}
}
if ( ! this . deviceIsIOS4 ) {
if ( ! deviceIsIOS4 ) {
// Weird things happen on iOS when an alert or confirm dialog is opened from a click event callback (issue #23):
// Weird things happen on iOS when an alert or confirm dialog is opened from a click event callback (issue #23):
// when the user next taps anywhere else on the page, new touchstart and touchend events are dispatched
// when the user next taps anywhere else on the page, new touchstart and touchend events are dispatched
@ -533,7 +516,7 @@ FastClick.prototype.onTouchEnd = function(event) {
// is performing a transition or scroll, and has to be re-detected manually. Note that
// is performing a transition or scroll, and has to be re-detected manually. Note that
// for this to function correctly, it must be called *after* the event target is checked!
// for this to function correctly, it must be called *after* the event target is checked!
// See issue #57; also filed as rdar://13048589 .
// See issue #57; also filed as rdar://13048589 .
if ( this . deviceIsIOSWithBadTarget ) {
if ( deviceIsIOSWithBadTarget ) {
touch = event . changedTouches [ 0 ] ;
touch = event . changedTouches [ 0 ] ;
// In certain cases arguments of elementFromPoint can be negative, so prevent setting targetElement to null
// In certain cases arguments of elementFromPoint can be negative, so prevent setting targetElement to null
@ -546,7 +529,7 @@ FastClick.prototype.onTouchEnd = function(event) {
forElement = this . findControl ( targetElement ) ;
forElement = this . findControl ( targetElement ) ;
if ( forElement ) {
if ( forElement ) {
this . focus ( targetElement ) ;
this . focus ( targetElement ) ;
if ( this . deviceIsAndroid ) {
if ( deviceIsAndroid ) {
return false ;
return false ;
}
}
@ -556,15 +539,16 @@ FastClick.prototype.onTouchEnd = function(event) {
// Case 1: If the touch started a while ago (best guess is 100ms based on tests for issue #36) then focus will be triggered anyway. Return early and unset the target element reference so that the subsequent click will be allowed through.
// Case 1: If the touch started a while ago (best guess is 100ms based on tests for issue #36) then focus will be triggered anyway. Return early and unset the target element reference so that the subsequent click will be allowed through.
// Case 2: Without this exception for input elements tapped when the document is contained in an iframe, then any inputted text won't be visible even though the value attribute is updated as the user types (issue #37).
// Case 2: Without this exception for input elements tapped when the document is contained in an iframe, then any inputted text won't be visible even though the value attribute is updated as the user types (issue #37).
if ( ( event . timeStamp - trackingClickStart ) > 100 || ( this . deviceIsIOS && window . top !== window && targetTagName === 'input' ) ) {
if ( ( event . timeStamp - trackingClickStart ) > 100 || ( deviceIsIOS && window . top !== window && targetTagName === 'input' ) ) {
this . targetElement = null ;
this . targetElement = null ;
return false ;
return false ;
}
}
this . focus ( targetElement ) ;
this . focus ( targetElement ) ;
this . sendClick ( targetElement , event ) ;
// Select elements need the event to go through on iOS 4, otherwise the selector menu won't open.
// Select elements need the event to go through on iOS 4, otherwise the selector menu won't open.
if ( ! this . deviceIsIOS4 || targetTagName !== 'select' ) {
if ( ! deviceIsIOS4 || targetTagName !== 'select' ) {
this . targetElement = null ;
this . targetElement = null ;
event . preventDefault ( ) ;
event . preventDefault ( ) ;
}
}
@ -572,7 +556,7 @@ FastClick.prototype.onTouchEnd = function(event) {
return false ;
return false ;
}
}
if ( this . deviceIsIOS && ! this . deviceIsIOS4 ) {
if ( deviceIsIOS && ! deviceIsIOS4 ) {
// Don't send a synthetic click event if the target element is contained within a parent layer that was scrolled
// Don't send a synthetic click event if the target element is contained within a parent layer that was scrolled
// and this tap is being used to stop the scrolling (usually initiated by a fling - issue #42).
// and this tap is being used to stop the scrolling (usually initiated by a fling - issue #42).
@ -699,7 +683,7 @@ FastClick.prototype.destroy = function() {
'use strict' ;
'use strict' ;
var layer = this . layer ;
var layer = this . layer ;
if ( this . deviceIsAndroid ) {
if ( deviceIsAndroid ) {
layer . removeEventListener ( 'mouseover' , this . onMouse , true ) ;
layer . removeEventListener ( 'mouseover' , this . onMouse , true ) ;
layer . removeEventListener ( 'mousedown' , this . onMouse , true ) ;
layer . removeEventListener ( 'mousedown' , this . onMouse , true ) ;
layer . removeEventListener ( 'mouseup' , this . onMouse , true ) ;
layer . removeEventListener ( 'mouseup' , this . onMouse , true ) ;
@ -733,7 +717,7 @@ FastClick.notNeeded = function(layer) {
if ( chromeVersion ) {
if ( chromeVersion ) {
if ( FastClick . prototype . deviceIsAndroid ) {
if ( deviceIsAndroid ) {
metaViewport = document . querySelector ( 'meta[name=viewport]' ) ;
metaViewport = document . querySelector ( 'meta[name=viewport]' ) ;
if ( metaViewport ) {
if ( metaViewport ) {