Last active
July 11, 2024 20:29
-
-
Save jpattishall/4180482 to your computer and use it in GitHub Desktop.
setTimeout workaround for iPad and iOS6
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
Workaround for iOS 6 setTimeout bug using requestAnimationFrame to simulate timers during Touch/Gesture-based events | |
Author: Jack Pattishall ([email protected]) | |
This code is free to use anywhere (MIT, etc.) | |
Note: UIWebView does not support requestAnimationFrames. If your timer is failing during a scroll event, | |
take a look at https://gist.github.com/ronkorving/3755461 for a potential workaround. | |
Usage: Pass TRUE as the final argument for setTimeout or setInterval. | |
Ex: | |
setTimeout(func, 1000) // uses native code | |
setTimeout(func, 1000, true) // uses workaround | |
Demos: | |
http://jsfiddle.net/xKh5m/ - uses native setTimeout | |
http://jsfiddle.net/ujxE3/ - uses workaround timers | |
*/ | |
(function(){ | |
// Only apply settimeout workaround for iOS 6.0 - 6.0.2 (iOS 6.1+ addresses issue) | |
if (!navigator.userAgent.match(/OS 6_0/i)) return; | |
// Prevent multiple applications | |
if(window.getTimeouts !== undefined) return; | |
var TIMERID = 'rafTimer', | |
touchTimeouts = {}, | |
touchIntervals = {}, | |
/* Reference to original timers */ | |
_st = window.setTimeout, | |
_si = window.setInterval, | |
_ct = window.clearTimeout, | |
_ci = window.clearInterval, | |
/* Request animation timers */ | |
_clearTouchTimer = function(uid, isInterval){ | |
var interval = isInterval || false, | |
timer = interval ? touchIntervals : touchTimeouts; | |
if(timer[uid]) { | |
timer[uid].callback = undefined; | |
timer[uid].loop = false; | |
return true; | |
} else { | |
return false; | |
} | |
}, | |
_touchTimer = function(callback, wait, isInterval){ | |
var uid, | |
name = callback.name || TIMERID + Math.floor(Math.random() * 1000), | |
delta = new Date().getTime()+ wait, | |
interval = isInterval || false, | |
timer = interval ? touchIntervals : touchTimeouts; | |
uid = name + "" + delta; | |
timer[uid] = {}; | |
timer[uid].loop = true; | |
timer[uid].callback = callback; | |
function _loop() { | |
var now = new Date().getTime(); | |
if (timer[uid].loop !== false) { | |
timer[uid].requestededFrame = webkitRequestAnimationFrame(_loop); | |
timer[uid].loop = now <= delta; | |
} else { | |
if(timer[uid].callback) timer[uid].callback(); | |
if(interval){ | |
delta = new Date().getTime() + wait; | |
timer[uid].loop = now <= delta; | |
timer[uid].requestedFrame = webkitRequestAnimationFrame(_loop); | |
} else { | |
delete timer[uid]; | |
} | |
} | |
}; | |
_loop(); | |
return uid; | |
}, | |
_timer = function(callback, wait, touch, isInterval){ | |
if(touch){ | |
return _touchTimer(callback, wait, isInterval); | |
} else { | |
return isInterval ? _si(callback, wait) : _st(callback, wait); | |
} | |
}, | |
_clear = function(uid, isInterval){ | |
if(uid && uid.indexOf && uid.indexOf(TIMERID) > -1){ | |
return _clearTouchTimer(uid, isInterval); | |
} else { | |
return isInterval ? _ci(uid) : _ct(uid); | |
} | |
}; | |
/* Returns raf-based timers; For debugging purposes */ | |
window.getTimeouts = function(){ | |
return { timeouts: touchTimeouts , intervals : touchIntervals } | |
}; | |
/* Exposed globally */ | |
window.setTimeout = function(callback, wait, touch){ | |
return _timer(callback, wait, touch); | |
}; | |
window.setInterval = function(callback, wait, touch){ | |
return _timer(callback, wait, touch, true); | |
}; | |
window.clearTimeout = function(uid){ | |
return _clear(uid); | |
}; | |
window.clearInterval = function(uid){ | |
return _clear(uid, true); | |
}; | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
happy birthday :)