You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
223 lines
9.4 KiB
223 lines
9.4 KiB
"use strict"; |
|
|
|
// Use the fastest means possible to execute a task in its own turn, with |
|
// priority over other events including IO, animation, reflow, and redraw |
|
// events in browsers. |
|
// |
|
// An exception thrown by a task will permanently interrupt the processing of |
|
// subsequent tasks. The higher level `asap` function ensures that if an |
|
// exception is thrown by a task, that the task queue will continue flushing as |
|
// soon as possible, but if you use `rawAsap` directly, you are responsible to |
|
// either ensure that no exceptions are thrown from your task, or to manually |
|
// call `rawAsap.requestFlush` if an exception is thrown. |
|
module.exports = rawAsap; |
|
function rawAsap(task) { |
|
if (!queue.length) { |
|
requestFlush(); |
|
flushing = true; |
|
} |
|
// Equivalent to push, but avoids a function call. |
|
queue[queue.length] = task; |
|
} |
|
|
|
var queue = []; |
|
// Once a flush has been requested, no further calls to `requestFlush` are |
|
// necessary until the next `flush` completes. |
|
var flushing = false; |
|
// `requestFlush` is an implementation-specific method that attempts to kick |
|
// off a `flush` event as quickly as possible. `flush` will attempt to exhaust |
|
// the event queue before yielding to the browser's own event loop. |
|
var requestFlush; |
|
// The position of the next task to execute in the task queue. This is |
|
// preserved between calls to `flush` so that it can be resumed if |
|
// a task throws an exception. |
|
var index = 0; |
|
// If a task schedules additional tasks recursively, the task queue can grow |
|
// unbounded. To prevent memory exhaustion, the task queue will periodically |
|
// truncate already-completed tasks. |
|
var capacity = 1024; |
|
|
|
// The flush function processes all tasks that have been scheduled with |
|
// `rawAsap` unless and until one of those tasks throws an exception. |
|
// If a task throws an exception, `flush` ensures that its state will remain |
|
// consistent and will resume where it left off when called again. |
|
// However, `flush` does not make any arrangements to be called again if an |
|
// exception is thrown. |
|
function flush() { |
|
while (index < queue.length) { |
|
var currentIndex = index; |
|
// Advance the index before calling the task. This ensures that we will |
|
// begin flushing on the next task the task throws an error. |
|
index = index + 1; |
|
queue[currentIndex].call(); |
|
// Prevent leaking memory for long chains of recursive calls to `asap`. |
|
// If we call `asap` within tasks scheduled by `asap`, the queue will |
|
// grow, but to avoid an O(n) walk for every task we execute, we don't |
|
// shift tasks off the queue after they have been executed. |
|
// Instead, we periodically shift 1024 tasks off the queue. |
|
if (index > capacity) { |
|
// Manually shift all values starting at the index back to the |
|
// beginning of the queue. |
|
for (var scan = 0, newLength = queue.length - index; scan < newLength; scan++) { |
|
queue[scan] = queue[scan + index]; |
|
} |
|
queue.length -= index; |
|
index = 0; |
|
} |
|
} |
|
queue.length = 0; |
|
index = 0; |
|
flushing = false; |
|
} |
|
|
|
// `requestFlush` is implemented using a strategy based on data collected from |
|
// every available SauceLabs Selenium web driver worker at time of writing. |
|
// https://docs.google.com/spreadsheets/d/1mG-5UYGup5qxGdEMWkhP6BWCz053NUb2E1QoUTU16uA/edit#gid=783724593 |
|
|
|
// Safari 6 and 6.1 for desktop, iPad, and iPhone are the only browsers that |
|
// have WebKitMutationObserver but not un-prefixed MutationObserver. |
|
// Must use `global` or `self` instead of `window` to work in both frames and web |
|
// workers. `global` is a provision of Browserify, Mr, Mrs, or Mop. |
|
|
|
/* globals self */ |
|
var scope = typeof global !== "undefined" ? global : self; |
|
var BrowserMutationObserver = scope.MutationObserver || scope.WebKitMutationObserver; |
|
|
|
// MutationObservers are desirable because they have high priority and work |
|
// reliably everywhere they are implemented. |
|
// They are implemented in all modern browsers. |
|
// |
|
// - Android 4-4.3 |
|
// - Chrome 26-34 |
|
// - Firefox 14-29 |
|
// - Internet Explorer 11 |
|
// - iPad Safari 6-7.1 |
|
// - iPhone Safari 7-7.1 |
|
// - Safari 6-7 |
|
if (typeof BrowserMutationObserver === "function") { |
|
requestFlush = makeRequestCallFromMutationObserver(flush); |
|
|
|
// MessageChannels are desirable because they give direct access to the HTML |
|
// task queue, are implemented in Internet Explorer 10, Safari 5.0-1, and Opera |
|
// 11-12, and in web workers in many engines. |
|
// Although message channels yield to any queued rendering and IO tasks, they |
|
// would be better than imposing the 4ms delay of timers. |
|
// However, they do not work reliably in Internet Explorer or Safari. |
|
|
|
// Internet Explorer 10 is the only browser that has setImmediate but does |
|
// not have MutationObservers. |
|
// Although setImmediate yields to the browser's renderer, it would be |
|
// preferrable to falling back to setTimeout since it does not have |
|
// the minimum 4ms penalty. |
|
// Unfortunately there appears to be a bug in Internet Explorer 10 Mobile (and |
|
// Desktop to a lesser extent) that renders both setImmediate and |
|
// MessageChannel useless for the purposes of ASAP. |
|
// https://github.com/kriskowal/q/issues/396 |
|
|
|
// Timers are implemented universally. |
|
// We fall back to timers in workers in most engines, and in foreground |
|
// contexts in the following browsers. |
|
// However, note that even this simple case requires nuances to operate in a |
|
// broad spectrum of browsers. |
|
// |
|
// - Firefox 3-13 |
|
// - Internet Explorer 6-9 |
|
// - iPad Safari 4.3 |
|
// - Lynx 2.8.7 |
|
} else { |
|
requestFlush = makeRequestCallFromTimer(flush); |
|
} |
|
|
|
// `requestFlush` requests that the high priority event queue be flushed as |
|
// soon as possible. |
|
// This is useful to prevent an error thrown in a task from stalling the event |
|
// queue if the exception handled by Node.js’s |
|
// `process.on("uncaughtException")` or by a domain. |
|
rawAsap.requestFlush = requestFlush; |
|
|
|
// To request a high priority event, we induce a mutation observer by toggling |
|
// the text of a text node between "1" and "-1". |
|
function makeRequestCallFromMutationObserver(callback) { |
|
var toggle = 1; |
|
var observer = new BrowserMutationObserver(callback); |
|
var node = document.createTextNode(""); |
|
observer.observe(node, {characterData: true}); |
|
return function requestCall() { |
|
toggle = -toggle; |
|
node.data = toggle; |
|
}; |
|
} |
|
|
|
// The message channel technique was discovered by Malte Ubl and was the |
|
// original foundation for this library. |
|
// http://www.nonblocking.io/2011/06/windownexttick.html |
|
|
|
// Safari 6.0.5 (at least) intermittently fails to create message ports on a |
|
// page's first load. Thankfully, this version of Safari supports |
|
// MutationObservers, so we don't need to fall back in that case. |
|
|
|
// function makeRequestCallFromMessageChannel(callback) { |
|
// var channel = new MessageChannel(); |
|
// channel.port1.onmessage = callback; |
|
// return function requestCall() { |
|
// channel.port2.postMessage(0); |
|
// }; |
|
// } |
|
|
|
// For reasons explained above, we are also unable to use `setImmediate` |
|
// under any circumstances. |
|
// Even if we were, there is another bug in Internet Explorer 10. |
|
// It is not sufficient to assign `setImmediate` to `requestFlush` because |
|
// `setImmediate` must be called *by name* and therefore must be wrapped in a |
|
// closure. |
|
// Never forget. |
|
|
|
// function makeRequestCallFromSetImmediate(callback) { |
|
// return function requestCall() { |
|
// setImmediate(callback); |
|
// }; |
|
// } |
|
|
|
// Safari 6.0 has a problem where timers will get lost while the user is |
|
// scrolling. This problem does not impact ASAP because Safari 6.0 supports |
|
// mutation observers, so that implementation is used instead. |
|
// However, if we ever elect to use timers in Safari, the prevalent work-around |
|
// is to add a scroll event listener that calls for a flush. |
|
|
|
// `setTimeout` does not call the passed callback if the delay is less than |
|
// approximately 7 in web workers in Firefox 8 through 18, and sometimes not |
|
// even then. |
|
|
|
function makeRequestCallFromTimer(callback) { |
|
return function requestCall() { |
|
// We dispatch a timeout with a specified delay of 0 for engines that |
|
// can reliably accommodate that request. This will usually be snapped |
|
// to a 4 milisecond delay, but once we're flushing, there's no delay |
|
// between events. |
|
var timeoutHandle = setTimeout(handleTimer, 0); |
|
// However, since this timer gets frequently dropped in Firefox |
|
// workers, we enlist an interval handle that will try to fire |
|
// an event 20 times per second until it succeeds. |
|
var intervalHandle = setInterval(handleTimer, 50); |
|
|
|
function handleTimer() { |
|
// Whichever timer succeeds will cancel both timers and |
|
// execute the callback. |
|
clearTimeout(timeoutHandle); |
|
clearInterval(intervalHandle); |
|
callback(); |
|
} |
|
}; |
|
} |
|
|
|
// This is for `asap.js` only. |
|
// Its name will be periodically randomized to break any code that depends on |
|
// its existence. |
|
rawAsap.makeRequestCallFromTimer = makeRequestCallFromTimer; |
|
|
|
// ASAP was originally a nextTick shim included in Q. This was factored out |
|
// into this ASAP package. It was later adapted to RSVP which made further |
|
// amendments. These decisions, particularly to marginalize MessageChannel and |
|
// to capture the MutationObserver implementation in a closure, were integrated |
|
// back into ASAP proper. |
|
// https://github.com/tildeio/rsvp.js/blob/cddf7232546a9cf858524b75cde6f9edf72620a7/lib/rsvp/asap.js
|
|
|