|  |  |  |  | (function (global, undefined) { | 
					
						
							|  |  |  |  |     "use strict"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (global.setImmediate) { | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     var nextHandle = 1; // Spec says greater than zero
 | 
					
						
							|  |  |  |  |     var tasksByHandle = {}; | 
					
						
							|  |  |  |  |     var currentlyRunningATask = false; | 
					
						
							|  |  |  |  |     var doc = global.document; | 
					
						
							|  |  |  |  |     var registerImmediate; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function setImmediate(callback) { | 
					
						
							|  |  |  |  |       // Callback can either be a function or a string
 | 
					
						
							|  |  |  |  |       if (typeof callback !== "function") { | 
					
						
							|  |  |  |  |         callback = new Function("" + callback); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |       // Copy function arguments
 | 
					
						
							|  |  |  |  |       var args = new Array(arguments.length - 1); | 
					
						
							|  |  |  |  |       for (var i = 0; i < args.length; i++) { | 
					
						
							|  |  |  |  |           args[i] = arguments[i + 1]; | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |       // Store and register the task
 | 
					
						
							|  |  |  |  |       var task = { callback: callback, args: args }; | 
					
						
							|  |  |  |  |       tasksByHandle[nextHandle] = task; | 
					
						
							|  |  |  |  |       registerImmediate(nextHandle); | 
					
						
							|  |  |  |  |       return nextHandle++; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function clearImmediate(handle) { | 
					
						
							|  |  |  |  |         delete tasksByHandle[handle]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function run(task) { | 
					
						
							|  |  |  |  |         var callback = task.callback; | 
					
						
							|  |  |  |  |         var args = task.args; | 
					
						
							|  |  |  |  |         switch (args.length) { | 
					
						
							|  |  |  |  |         case 0: | 
					
						
							|  |  |  |  |             callback(); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case 1: | 
					
						
							|  |  |  |  |             callback(args[0]); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case 2: | 
					
						
							|  |  |  |  |             callback(args[0], args[1]); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         case 3: | 
					
						
							|  |  |  |  |             callback(args[0], args[1], args[2]); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             callback.apply(undefined, args); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function runIfPresent(handle) { | 
					
						
							|  |  |  |  |         // From the spec: "Wait until any invocations of this algorithm started before this one have completed."
 | 
					
						
							|  |  |  |  |         // So if we're currently running a task, we'll need to delay this invocation.
 | 
					
						
							|  |  |  |  |         if (currentlyRunningATask) { | 
					
						
							|  |  |  |  |             // Delay by doing a setTimeout. setImmediate was tried instead, but in Firefox 7 it generated a
 | 
					
						
							|  |  |  |  |             // "too much recursion" error.
 | 
					
						
							|  |  |  |  |             setTimeout(runIfPresent, 0, handle); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             var task = tasksByHandle[handle]; | 
					
						
							|  |  |  |  |             if (task) { | 
					
						
							|  |  |  |  |                 currentlyRunningATask = true; | 
					
						
							|  |  |  |  |                 try { | 
					
						
							|  |  |  |  |                     run(task); | 
					
						
							|  |  |  |  |                 } finally { | 
					
						
							|  |  |  |  |                     clearImmediate(handle); | 
					
						
							|  |  |  |  |                     currentlyRunningATask = false; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function installNextTickImplementation() { | 
					
						
							|  |  |  |  |         registerImmediate = function(handle) { | 
					
						
							|  |  |  |  |             process.nextTick(function () { runIfPresent(handle); }); | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function canUsePostMessage() { | 
					
						
							|  |  |  |  |         // The test against `importScripts` prevents this implementation from being installed inside a web worker,
 | 
					
						
							|  |  |  |  |         // where `global.postMessage` means something completely different and can't be used for this purpose.
 | 
					
						
							|  |  |  |  |         if (global.postMessage && !global.importScripts) { | 
					
						
							|  |  |  |  |             var postMessageIsAsynchronous = true; | 
					
						
							|  |  |  |  |             var oldOnMessage = global.onmessage; | 
					
						
							|  |  |  |  |             global.onmessage = function() { | 
					
						
							|  |  |  |  |                 postMessageIsAsynchronous = false; | 
					
						
							|  |  |  |  |             }; | 
					
						
							|  |  |  |  |             global.postMessage("", "*"); | 
					
						
							|  |  |  |  |             global.onmessage = oldOnMessage; | 
					
						
							|  |  |  |  |             return postMessageIsAsynchronous; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function installPostMessageImplementation() { | 
					
						
							|  |  |  |  |         // Installs an event handler on `global` for the `message` event: see
 | 
					
						
							|  |  |  |  |         // * https://developer.mozilla.org/en/DOM/window.postMessage
 | 
					
						
							|  |  |  |  |         // * http://www.whatwg.org/specs/web-apps/current-work/multipage/comms.html#crossDocumentMessages
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         var messagePrefix = "setImmediate$" + Math.random() + "$"; | 
					
						
							|  |  |  |  |         var onGlobalMessage = function(event) { | 
					
						
							|  |  |  |  |             if (event.source === global && | 
					
						
							|  |  |  |  |                 typeof event.data === "string" && | 
					
						
							|  |  |  |  |                 event.data.indexOf(messagePrefix) === 0) { | 
					
						
							|  |  |  |  |                 runIfPresent(+event.data.slice(messagePrefix.length)); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (global.addEventListener) { | 
					
						
							|  |  |  |  |             global.addEventListener("message", onGlobalMessage, false); | 
					
						
							|  |  |  |  |         } else { | 
					
						
							|  |  |  |  |             global.attachEvent("onmessage", onGlobalMessage); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         registerImmediate = function(handle) { | 
					
						
							|  |  |  |  |             global.postMessage(messagePrefix + handle, "*"); | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function installMessageChannelImplementation() { | 
					
						
							|  |  |  |  |         var channel = new MessageChannel(); | 
					
						
							|  |  |  |  |         channel.port1.onmessage = function(event) { | 
					
						
							|  |  |  |  |             var handle = event.data; | 
					
						
							|  |  |  |  |             runIfPresent(handle); | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         registerImmediate = function(handle) { | 
					
						
							|  |  |  |  |             channel.port2.postMessage(handle); | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function installReadyStateChangeImplementation() { | 
					
						
							|  |  |  |  |         var html = doc.documentElement; | 
					
						
							|  |  |  |  |         registerImmediate = function(handle) { | 
					
						
							|  |  |  |  |             // Create a <script> element; its readystatechange event will be fired asynchronously once it is inserted
 | 
					
						
							|  |  |  |  |             // into the document. Do so, thus queuing up the task. Remember to clean up once it's been called.
 | 
					
						
							|  |  |  |  |             var script = doc.createElement("script"); | 
					
						
							|  |  |  |  |             script.onreadystatechange = function () { | 
					
						
							|  |  |  |  |                 runIfPresent(handle); | 
					
						
							|  |  |  |  |                 script.onreadystatechange = null; | 
					
						
							|  |  |  |  |                 html.removeChild(script); | 
					
						
							|  |  |  |  |                 script = null; | 
					
						
							|  |  |  |  |             }; | 
					
						
							|  |  |  |  |             html.appendChild(script); | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     function installSetTimeoutImplementation() { | 
					
						
							|  |  |  |  |         registerImmediate = function(handle) { | 
					
						
							|  |  |  |  |             setTimeout(runIfPresent, 0, handle); | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // If supported, we should attach to the prototype of global, since that is where setTimeout et al. live.
 | 
					
						
							|  |  |  |  |     var attachTo = Object.getPrototypeOf && Object.getPrototypeOf(global); | 
					
						
							|  |  |  |  |     attachTo = attachTo && attachTo.setTimeout ? attachTo : global; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // Don't get fooled by e.g. browserify environments.
 | 
					
						
							|  |  |  |  |     if ({}.toString.call(global.process) === "[object process]") { | 
					
						
							|  |  |  |  |         // For Node.js before 0.9
 | 
					
						
							|  |  |  |  |         installNextTickImplementation(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     } else if (canUsePostMessage()) { | 
					
						
							|  |  |  |  |         // For non-IE10 modern browsers
 | 
					
						
							|  |  |  |  |         installPostMessageImplementation(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     } else if (global.MessageChannel) { | 
					
						
							|  |  |  |  |         // For web workers, where supported
 | 
					
						
							|  |  |  |  |         installMessageChannelImplementation(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     } else if (doc && "onreadystatechange" in doc.createElement("script")) { | 
					
						
							|  |  |  |  |         // For IE 6–8
 | 
					
						
							|  |  |  |  |         installReadyStateChangeImplementation(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |         // For older browsers
 | 
					
						
							|  |  |  |  |         installSetTimeoutImplementation(); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     attachTo.setImmediate = setImmediate; | 
					
						
							|  |  |  |  |     attachTo.clearImmediate = clearImmediate; | 
					
						
							|  |  |  |  | }(typeof self === "undefined" ? typeof global === "undefined" ? this : global : self)); |