AutorÃa | Ultima modificación | Ver Log |
YUI.add('get', function (Y, NAME) {/*jslint boss:true, expr:true, laxbreak: true *//**Provides dynamic loading of remote JavaScript and CSS resources.@module get@class Get@static**/var Lang = Y.Lang,CUSTOM_ATTRS, // defined lazily in Y.Get.Transaction._createNode()Get, Transaction;Y.Get = Get = {// -- Public Properties ----------------------------------------------------/**Default options for CSS requests. Options specified here will overrideglobal defaults for CSS requests.See the `options` property for all available options.@property cssOptions@type Object@static@since 3.5.0**/cssOptions: {attributes: {rel: 'stylesheet'},doc : Y.config.linkDoc || Y.config.doc,pollInterval: 50},/**Default options for JS requests. Options specified here will override globaldefaults for JS requests.See the `options` property for all available options.@property jsOptions@type Object@static@since 3.5.0**/jsOptions: {autopurge: true,doc : Y.config.scriptDoc || Y.config.doc},/**Default options to use for all requests.Note that while all available options are documented here for ease ofdiscovery, some options (like callback functions) only make sense at thetransaction level.Callback functions specified via the options object or the `options`parameter of the `css()`, `js()`, or `load()` methods will receive thetransaction object as a parameter. See `Y.Get.Transaction` for details onthe properties and methods available on transactions.@static@since 3.5.0@property {Object} options@property {Boolean} [options.async=false] Whether or not to load scriptsasynchronously, meaning they're requested in parallel and executionorder is not guaranteed. Has no effect on CSS, since CSS is alwaysloaded asynchronously.@property {Object} [options.attributes] HTML attribute name/value pairs thatshould be added to inserted nodes. By default, the `charset` attributewill be set to "utf-8" and nodes will be given an auto-generated `id`attribute, but you can override these with your own values if desired.@property {Boolean} [options.autopurge] Whether or not to automaticallypurge inserted nodes after the purge threshold is reached. This is`true` by default for JavaScript, but `false` for CSS since purging aCSS node will also remove any styling applied by the referenced file.@property {Object} [options.context] `this` object to use when callingcallback functions. Defaults to the transaction object.@property {Mixed} [options.data] Arbitrary data object to pass to "on*"callbacks.@property {Document} [options.doc] Document into which nodes should beinserted. By default, the current document is used.@property {HTMLElement|String} [options.insertBefore] HTML element or idstring of an element before which all generated nodes should beinserted. If not specified, Get will automatically determine the bestplace to insert nodes for maximum compatibility.@property {Function} [options.onEnd] Callback to execute after a transactionis complete, regardless of whether it succeeded or failed.@property {Function} [options.onFailure] Callback to execute after atransaction fails, times out, or is aborted.@property {Function} [options.onProgress] Callback to execute after eachindividual request in a transaction either succeeds or fails.@property {Function} [options.onSuccess] Callback to execute after atransaction completes successfully with no errors. Note that in browsersthat don't support the `error` event on CSS `<link>` nodes, a failed CSSrequest may still be reported as a success because in these browsersit can be difficult or impossible to distinguish between success andfailure for CSS resources.@property {Function} [options.onTimeout] Callback to execute after atransaction times out.@property {Number} [options.pollInterval=50] Polling interval (inmilliseconds) for detecting CSS load completion in browsers that don'tsupport the `load` event on `<link>` nodes. This isn't used forJavaScript.@property {Number} [options.purgethreshold=20] Number of nodes to insertbefore triggering an automatic purge when `autopurge` is `true`.@property {Number} [options.timeout] Number of milliseconds to wait beforeaborting a transaction. When a timeout occurs, the `onTimeout` callbackis called, followed by `onFailure` and finally `onEnd`. By default,there is no timeout.@property {String} [options.type] Resource type ("css" or "js"). This optionis set automatically by the `css()` and `js()` functions and will beignored there, but may be useful when using the `load()` function. Ifnot specified, the type will be inferred from the URL, defaulting to"js" if the URL doesn't contain a recognizable file extension.**/options: {attributes: {charset: 'utf-8'},purgethreshold: 20},// -- Protected Properties -------------------------------------------------/**Regex that matches a CSS URL. Used to guess the file type when it's notspecified.@property REGEX_CSS@type RegExp@final@protected@static@since 3.5.0**/REGEX_CSS: /\.css(?:[?;].*)?$/i,/**Regex that matches a JS URL. Used to guess the file type when it's notspecified.@property REGEX_JS@type RegExp@final@protected@static@since 3.5.0**/REGEX_JS : /\.js(?:[?;].*)?$/i,/**Contains information about the current environment, such as what script andlink injection features it supports.This object is created and populated the first time the `_getEnv()` methodis called.@property _env@type Object@protected@static@since 3.5.0**//**Mapping of document _yuid strings to <head> or <base> node references so wedon't have to look the node up each time we want to insert a request node.@property _insertCache@type Object@protected@static@since 3.5.0**/_insertCache: {},/**Information about the currently pending transaction, if any.This is actually an object with two properties: `callback`, containing theoptional callback passed to `css()`, `load()`, or `js()`; and `transaction`,containing the actual transaction instance.@property _pending@type Object@protected@static@since 3.5.0**/_pending: null,/**HTML nodes eligible to be purged next time autopurge is triggered.@property _purgeNodes@type HTMLElement[]@protected@static@since 3.5.0**/_purgeNodes: [],/**Queued transactions and associated callbacks.@property _queue@type Object[]@protected@static@since 3.5.0**/_queue: [],// -- Public Methods -------------------------------------------------------/**Aborts the specified transaction.This will cause the transaction's `onFailure` callback to be called andwill prevent any new script and link nodes from being added to the document,but any resources that have already been requested will continue loading(there's no safe way to prevent this, unfortunately).*Note:* This method is deprecated as of 3.5.0, and will be removed in afuture version of YUI. Use the transaction-level `abort()` method instead.@method abort@param {Get.Transaction} transaction Transaction to abort.@deprecated Use the `abort()` method on the transaction instead.@static**/abort: function (transaction) {var i, id, item, len, pending;if (!transaction.abort) {id = transaction;pending = this._pending;transaction = null;if (pending && pending.transaction.id === id) {transaction = pending.transaction;this._pending = null;} else {for (i = 0, len = this._queue.length; i < len; ++i) {item = this._queue[i].transaction;if (item.id === id) {transaction = item;this._queue.splice(i, 1);break;}}}}transaction && transaction.abort();},/**Loads one or more CSS files.The _urls_ parameter may be provided as a URL string, a request object,or an array of URL strings and/or request objects.A request object is just an object that contains a `url` property and zeroor more options that should apply specifically to that request.Request-specific options take priority over transaction-level options anddefault options.URLs may be relative or absolute, and do not have to have the same originas the current page.The `options` parameter may be omitted completely and a callback passed inits place, if desired.@example// Load a single CSS file and log a message on completion.Y.Get.css('foo.css', function (err) {if (err) {} else {}});// Load multiple CSS files and log a message when all have finished// loading.var urls = ['foo.css', 'http://example.com/bar.css', 'baz/quux.css'];Y.Get.css(urls, function (err) {if (err) {} else {}});// Specify transaction-level options, which will apply to all requests// within the transaction.Y.Get.css(urls, {attributes: {'class': 'my-css'},timeout : 5000});// Specify per-request options, which override transaction-level and// default options.Y.Get.css([{url: 'foo.css', attributes: {id: 'foo'}},{url: 'bar.css', attributes: {id: 'bar', charset: 'iso-8859-1'}}]);@method css@param {String|Object|Array} urls URL string, request object, or arrayof URLs and/or request objects to load.@param {Object} [options] Options for this transaction. See the`Y.Get.options` property for a complete list of available options.@param {Function} [callback] Callback function to be called on completion.This is a general callback and will be called before any more granularcallbacks (`onSuccess`, `onFailure`, etc.) specified in the `options`object.@param {Array|null} callback.err Array of errors that occurred duringthe transaction, or `null` on success.@param {Get.Transaction} callback.transaction Transaction object.@return {Get.Transaction} Transaction object.@static**/css: function (urls, options, callback) {return this._load('css', urls, options, callback);},/**Loads one or more JavaScript resources.The _urls_ parameter may be provided as a URL string, a request object,or an array of URL strings and/or request objects.A request object is just an object that contains a `url` property and zeroor more options that should apply specifically to that request.Request-specific options take priority over transaction-level options anddefault options.URLs may be relative or absolute, and do not have to have the same originas the current page.The `options` parameter may be omitted completely and a callback passed inits place, if desired.Scripts will be executed in the order they're specified unless the `async`option is `true`, in which case they'll be loaded in parallel and executedin whatever order they finish loading.@example// Load a single JS file and log a message on completion.Y.Get.js('foo.js', function (err) {if (err) {} else {}});// Load multiple JS files, execute them in order, and log a message when// all have finished loading.var urls = ['foo.js', 'http://example.com/bar.js', 'baz/quux.js'];Y.Get.js(urls, function (err) {if (err) {} else {}});// Specify transaction-level options, which will apply to all requests// within the transaction.Y.Get.js(urls, {attributes: {'class': 'my-js'},timeout : 5000});// Specify per-request options, which override transaction-level and// default options.Y.Get.js([{url: 'foo.js', attributes: {id: 'foo'}},{url: 'bar.js', attributes: {id: 'bar', charset: 'iso-8859-1'}}]);@method js@param {String|Object|Array} urls URL string, request object, or arrayof URLs and/or request objects to load.@param {Object} [options] Options for this transaction. See the`Y.Get.options` property for a complete list of available options.@param {Function} [callback] Callback function to be called on completion.This is a general callback and will be called before any more granularcallbacks (`onSuccess`, `onFailure`, etc.) specified in the `options`object.@param {Array|null} callback.err Array of errors that occurred duringthe transaction, or `null` on success.@param {Get.Transaction} callback.transaction Transaction object.@return {Get.Transaction} Transaction object.@since 3.5.0@static**/js: function (urls, options, callback) {return this._load('js', urls, options, callback);},/**Loads one or more CSS and/or JavaScript resources in the same transaction.Use this method when you want to load both CSS and JavaScript in a singletransaction and be notified when all requested URLs have finished loading,regardless of type.Behavior and options are the same as for the `css()` and `js()` methods. Ifa resource type isn't specified in per-request options or transaction-leveloptions, Get will guess the file type based on the URL's extension (`.css`or `.js`, with or without a following query string). If the file type can'tbe guessed from the URL, a warning will be logged and Get will assume theURL is a JavaScript resource.@example// Load both CSS and JS files in a single transaction, and log a message// when all files have finished loading.Y.Get.load(['foo.css', 'bar.js', 'baz.css'], function (err) {if (err) {} else {}});@method load@param {String|Object|Array} urls URL string, request object, or arrayof URLs and/or request objects to load.@param {Object} [options] Options for this transaction. See the`Y.Get.options` property for a complete list of available options.@param {Function} [callback] Callback function to be called on completion.This is a general callback and will be called before any more granularcallbacks (`onSuccess`, `onFailure`, etc.) specified in the `options`object.@param {Array|null} err Array of errors that occurred during thetransaction, or `null` on success.@param {Get.Transaction} Transaction object.@return {Get.Transaction} Transaction object.@since 3.5.0@static**/load: function (urls, options, callback) {return this._load(null, urls, options, callback);},// -- Protected Methods ----------------------------------------------------/**Triggers an automatic purge if the purge threshold has been reached.@method _autoPurge@param {Number} threshold Purge threshold to use, in milliseconds.@protected@since 3.5.0@static**/_autoPurge: function (threshold) {if (threshold && this._purgeNodes.length >= threshold) {this._purge(this._purgeNodes);}},/**Populates the `_env` property with information about the currentenvironment.@method _getEnv@return {Object} Environment information.@protected@since 3.5.0@static**/_getEnv: function () {var doc = Y.config.doc,ua = Y.UA;// Note: some of these checks require browser sniffs since it's not// feasible to load test files on every pageview just to perform a// feature test. I'm sorry if this makes you sad.return (this._env = {// True if this is a browser that supports disabling async mode on// dynamically created script nodes. See// https://developer.mozilla.org/En/HTML/Element/Script#Attributes// IE10 doesn't return true for the MDN feature test, so setting it explicitly,// because it is async by default, and allows you to disable async by setting it to falseasync: (doc && doc.createElement('script').async === true) || (ua.ie >= 10),// True if this browser fires an event when a dynamically injected// link node fails to load. This is currently true for Firefox 9+// and WebKit 535.24+cssFail: ua.gecko >= 9 || ua.compareVersions(ua.webkit, 535.24) >= 0,// True if this browser fires an event when a dynamically injected// link node finishes loading. This is currently true for IE, Opera,// Firefox 9+, and WebKit 535.24+. Note that IE versions <9 fire the// DOM 0 "onload" event, but not "load". All versions of IE fire// "onload".// davglass: Seems that Chrome on Android needs this to be false.cssLoad: ((!ua.gecko && !ua.webkit) || ua.gecko >= 9 ||ua.compareVersions(ua.webkit, 535.24) >= 0) && !(ua.chrome && ua.chrome <= 18),// True if this browser preserves script execution order while// loading scripts in parallel as long as the script node's `async`// attribute is set to false to explicitly disable async execution.preservesScriptOrder: !!(ua.gecko || ua.opera || (ua.ie && ua.ie >= 10))});},_getTransaction: function (urls, options) {var requests = [],i, len, req, url;if (!Lang.isArray(urls)) {urls = [urls];}options = Y.merge(this.options, options);// Clone the attributes object so we don't end up modifying it by ref.options.attributes = Y.merge(this.options.attributes,options.attributes);for (i = 0, len = urls.length; i < len; ++i) {url = urls[i];req = {attributes: {}};// If `url` is a string, we create a URL object for it, then mix in// global options and request-specific options. If it's an object// with a "url" property, we assume it's a request object containing// URL-specific options.if (typeof url === 'string') {req.url = url;} else if (url.url) {// URL-specific options override both global defaults and// request-specific options.Y.mix(req, url, false, null, 0, true);url = url.url; // Make url a string so we can use it later.} else {continue;}Y.mix(req, options, false, null, 0, true);// If we didn't get an explicit type for this URL either in the// request options or the URL-specific options, try to determine// one from the file extension.if (!req.type) {if (this.REGEX_CSS.test(url)) {req.type = 'css';} else {if (!this.REGEX_JS.test(url)) {}req.type = 'js';}}// Mix in type-specific default options, but don't overwrite any// options that have already been set.Y.mix(req, req.type === 'js' ? this.jsOptions : this.cssOptions,false, null, 0, true);// Give the node an id attribute if it doesn't already have one.req.attributes.id || (req.attributes.id = Y.guid());// Backcompat for <3.5.0 behavior.if (req.win) {req.doc = req.win.document;} else {req.win = req.doc.defaultView || req.doc.parentWindow;}if (req.charset) {req.attributes.charset = req.charset;}requests.push(req);}return new Transaction(requests, options);},_load: function (type, urls, options, callback) {var transaction;// Allow callback as third param.if (typeof options === 'function') {callback = options;options = {};}options || (options = {});options.type = type;options._onFinish = Get._onTransactionFinish;if (!this._env) {this._getEnv();}transaction = this._getTransaction(urls, options);this._queue.push({callback : callback,transaction: transaction});this._next();return transaction;},_onTransactionFinish : function() {Get._pending = null;Get._next();},_next: function () {var item;if (this._pending) {return;}item = this._queue.shift();if (item) {this._pending = item;item.transaction.execute(item.callback);}},_purge: function (nodes) {var purgeNodes = this._purgeNodes,isTransaction = nodes !== purgeNodes,index, node;while (node = nodes.pop()) { // assignment// Don't purge nodes that haven't finished loading (or errored out),// since this can hang the transaction.if (!node._yuiget_finished) {continue;}node.parentNode && node.parentNode.removeChild(node);// If this is a transaction-level purge and this node also exists in// the Get-level _purgeNodes array, we need to remove it from// _purgeNodes to avoid creating a memory leak. The indexOf lookup// sucks, but until we get WeakMaps, this is the least troublesome// way to do this (we can't just hold onto node ids because they may// not be in the same document).if (isTransaction) {index = Y.Array.indexOf(purgeNodes, node);if (index > -1) {purgeNodes.splice(index, 1);}}}}};/**Alias for `js()`.@method script@static**/Get.script = Get.js;/**Represents a Get transaction, which may contain requests for one or more JS orCSS files.This class should not be instantiated manually. Instances will be created andreturned as needed by Y.Get's `css()`, `js()`, and `load()` methods.@class Get.Transaction@constructor@since 3.5.0**/Get.Transaction = Transaction = function (requests, options) {var self = this;self.id = Transaction._lastId += 1;self.data = options.data;self.errors = [];self.nodes = [];self.options = options;self.requests = requests;self._callbacks = []; // callbacks to call after execution finishesself._queue = [];self._reqsWaiting = 0;// Deprecated pre-3.5.0 properties.self.tId = self.id; // Use `id` instead.self.win = options.win || Y.config.win;};/**Arbitrary data object associated with this transaction.This object comes from the options passed to `Get.css()`, `Get.js()`, or`Get.load()`, and will be `undefined` if no data object was specified.@property {Object} data**//**Array of errors that have occurred during this transaction, if any. Each errorobject has the following properties:`errors.error`: Error message.`errors.request`: Request object related to the error.@since 3.5.0@property {Object[]} errors**//**Numeric id for this transaction, unique among all transactions within the sameYUI sandbox in the current pageview.@property {Number} id@since 3.5.0**//**HTMLElement nodes (native ones, not YUI Node instances) that have been insertedduring the current transaction.@property {HTMLElement[]} nodes**//**Options associated with this transaction.See `Get.options` for the full list of available options.@property {Object} options@since 3.5.0**//**Request objects contained in this transaction. Each request object representsone CSS or JS URL that will be (or has been) requested and loaded into the page.@property {Object} requests@since 3.5.0**//**Id of the most recent transaction.@property _lastId@type Number@protected@static**/Transaction._lastId = 0;Transaction.prototype = {// -- Public Properties ----------------------------------------------------/**Current state of this transaction. One of "new", "executing", or "done".@property _state@type String@protected**/_state: 'new', // "new", "executing", or "done"// -- Public Methods -------------------------------------------------------/**Aborts this transaction.This will cause the transaction's `onFailure` callback to be called andwill prevent any new script and link nodes from being added to the document,but any resources that have already been requested will continue loading(there's no safe way to prevent this, unfortunately).@method abort@param {String} [msg="Aborted."] Optional message to use in the `errors`array describing why the transaction was aborted.**/abort: function (msg) {this._pending = null;this._pendingCSS = null;this._pollTimer = clearTimeout(this._pollTimer);this._queue = [];this._reqsWaiting = 0;this.errors.push({error: msg || 'Aborted'});this._finish();},/**Begins execting the transaction.There's usually no reason to call this manually, since Get will call itautomatically when other pending transactions have finished. If you reallywant to execute your transaction before Get does, you can, but be aware thatthis transaction's scripts may end up executing before the scripts in otherpending transactions.If the transaction is already executing, the specified callback (if any)will be queued and called after execution finishes. If the transaction hasalready finished, the callback will be called immediately (the transactionwill not be executed again).@method execute@param {Function} callback Callback function to execute after all requestsin the transaction are complete, or after the transaction is aborted.**/execute: function (callback) {var self = this,requests = self.requests,state = self._state,i, len, queue, req;if (state === 'done') {callback && callback(self.errors.length ? self.errors : null, self);return;} else {callback && self._callbacks.push(callback);if (state === 'executing') {return;}}self._state = 'executing';self._queue = queue = [];if (self.options.timeout) {self._timeout = setTimeout(function () {self.abort('Timeout');}, self.options.timeout);}self._reqsWaiting = requests.length;for (i = 0, len = requests.length; i < len; ++i) {req = requests[i];if (req.async || req.type === 'css') {// No need to queue CSS or fully async JS.self._insert(req);} else {queue.push(req);}}self._next();},/**Manually purges any `<script>` or `<link>` nodes this transaction hascreated.Be careful when purging a transaction that contains CSS requests, sinceremoving `<link>` nodes will also remove any styles they applied.@method purge**/purge: function () {Get._purge(this.nodes);},// -- Protected Methods ----------------------------------------------------_createNode: function (name, attrs, doc) {var node = doc.createElement(name),attr, testEl;if (!CUSTOM_ATTRS) {// IE6 and IE7 expect property names rather than attribute names for// certain attributes. Rather than sniffing, we do a quick feature// test the first time _createNode() runs to determine whether we// need to provide a workaround.testEl = doc.createElement('div');testEl.setAttribute('class', 'a');CUSTOM_ATTRS = testEl.className === 'a' ? {} : {'for' : 'htmlFor','class': 'className'};}for (attr in attrs) {if (attrs.hasOwnProperty(attr)) {node.setAttribute(CUSTOM_ATTRS[attr] || attr, attrs[attr]);}}return node;},_finish: function () {var errors = this.errors.length ? this.errors : null,options = this.options,thisObj = options.context || this,data, i, len;if (this._state === 'done') {return;}this._state = 'done';for (i = 0, len = this._callbacks.length; i < len; ++i) {this._callbacks[i].call(thisObj, errors, this);}data = this._getEventData();if (errors) {if (options.onTimeout && errors[errors.length - 1].error === 'Timeout') {options.onTimeout.call(thisObj, data);}if (options.onFailure) {options.onFailure.call(thisObj, data);}} else if (options.onSuccess) {options.onSuccess.call(thisObj, data);}if (options.onEnd) {options.onEnd.call(thisObj, data);}if (options._onFinish) {options._onFinish();}},_getEventData: function (req) {if (req) {// This merge is necessary for backcompat. I hate it.return Y.merge(this, {abort : this.abort, // have to copy these because the prototype isn't preservedpurge : this.purge,request: req,url : req.url,win : req.win});} else {return this;}},_getInsertBefore: function (req) {var doc = req.doc,el = req.insertBefore,cache, docStamp;if (el) {return typeof el === 'string' ? doc.getElementById(el) : el;}cache = Get._insertCache;docStamp = Y.stamp(doc);if ((el = cache[docStamp])) { // assignmentreturn el;}// Inserting before a <base> tag apparently works around an IE bug// (according to a comment from pre-3.5.0 Y.Get), but I'm not sure what// bug that is, exactly. Better safe than sorry?if ((el = doc.getElementsByTagName('base')[0])) { // assignmentreturn (cache[docStamp] = el);}// Look for a <head> element.el = doc.head || doc.getElementsByTagName('head')[0];if (el) {// Create a marker node at the end of <head> to use as an insertion// point. Inserting before this node will ensure that all our CSS// gets inserted in the correct order, to maintain style precedence.el.appendChild(doc.createTextNode(''));return (cache[docStamp] = el.lastChild);}// If all else fails, just insert before the first script node on the// page, which is virtually guaranteed to exist.return (cache[docStamp] = doc.getElementsByTagName('script')[0]);},_insert: function (req) {var env = Get._env,insertBefore = this._getInsertBefore(req),isScript = req.type === 'js',node = req.node,self = this,ua = Y.UA,cssTimeout, nodeType;if (!node) {if (isScript) {nodeType = 'script';} else if (!env.cssLoad && ua.gecko) {nodeType = 'style';} else {nodeType = 'link';}node = req.node = this._createNode(nodeType, req.attributes,req.doc);}function onError() {self._progress('Failed to load ' + req.url, req);}function onLoad() {if (cssTimeout) {clearTimeout(cssTimeout);}self._progress(null, req);}// Deal with script asynchronicity.if (isScript) {node.setAttribute('src', req.url);if (req.async) {// Explicitly indicate that we want the browser to execute this// script asynchronously. This is necessary for older browsers// like Firefox <4.node.async = true;} else {if (env.async) {// This browser treats injected scripts as async by default// (standard HTML5 behavior) but asynchronous loading isn't// desired, so tell the browser not to mark this script as// async.node.async = false;}// If this browser doesn't preserve script execution order based// on insertion order, we'll need to avoid inserting other// scripts until this one finishes loading.if (!env.preservesScriptOrder) {this._pending = req;}}} else {if (!env.cssLoad && ua.gecko) {// In Firefox <9, we can import the requested URL into a <style>// node and poll for the existence of node.sheet.cssRules. This// gives us a reliable way to determine CSS load completion that// also works for cross-domain stylesheets.//// Props to Zach Leatherman for calling my attention to this// technique.node.innerHTML = (req.attributes.charset ?'@charset "' + req.attributes.charset + '";' : '') +'@import "' + req.url + '";';} else {node.setAttribute('href', req.url);}}// Inject the node.if (isScript && ua.ie && (ua.ie < 9 || (document.documentMode && document.documentMode < 9))) {// Script on IE < 9, and IE 9+ when in IE 8 or older modes, including quirks mode.node.onreadystatechange = function () {if (/loaded|complete/.test(node.readyState)) {node.onreadystatechange = null;onLoad();}};} else if (!isScript && !env.cssLoad) {// CSS on Firefox <9 or WebKit.this._poll(req);} else {// Script or CSS on everything else. Using DOM 0 events because that// evens the playing field with older IEs.if (ua.ie >= 10) {// We currently need to introduce a timeout for IE10, since it// calls onerror/onload synchronously for 304s - messing up existing// program flow.// Remove this block if the following bug gets fixed by GA/*jshint maxlen: 1500 */// https://connect.microsoft.com/IE/feedback/details/763871/dynamically-loaded-scripts-with-304s-responses-interrupt-the-currently-executing-js-thread-onloadnode.onerror = function() { setTimeout(onError, 0); };node.onload = function() { setTimeout(onLoad, 0); };} else {node.onerror = onError;node.onload = onLoad;}// If this browser doesn't fire an event when CSS fails to load,// fail after a timeout to avoid blocking the transaction queue.if (!env.cssFail && !isScript) {cssTimeout = setTimeout(onError, req.timeout || 3000);}}this.nodes.push(node);insertBefore.parentNode.insertBefore(node, insertBefore);},_next: function () {if (this._pending) {return;}// If there are requests in the queue, insert the next queued request.// Otherwise, if we're waiting on already-inserted requests to finish,// wait longer. If there are no queued requests and we're not waiting// for anything to load, then we're done!if (this._queue.length) {this._insert(this._queue.shift());} else if (!this._reqsWaiting) {this._finish();}},_poll: function (newReq) {var self = this,pendingCSS = self._pendingCSS,isWebKit = Y.UA.webkit,i, hasRules, j, nodeHref, req, sheets;if (newReq) {pendingCSS || (pendingCSS = self._pendingCSS = []);pendingCSS.push(newReq);if (self._pollTimer) {// A poll timeout is already pending, so no need to create a// new one.return;}}self._pollTimer = null;// Note: in both the WebKit and Gecko hacks below, a CSS URL that 404s// will still be treated as a success. There's no good workaround for// this.for (i = 0; i < pendingCSS.length; ++i) {req = pendingCSS[i];if (isWebKit) {// Look for a stylesheet matching the pending URL.sheets = req.doc.styleSheets;j = sheets.length;nodeHref = req.node.href;while (--j >= 0) {if (sheets[j].href === nodeHref) {pendingCSS.splice(i, 1);i -= 1;self._progress(null, req);break;}}} else {// Many thanks to Zach Leatherman for calling my attention to// the @import-based cross-domain technique used here, and to// Oleg Slobodskoi for an earlier same-domain implementation.//// See Zach's blog for more details:// http://www.zachleat.com/web/2010/07/29/load-css-dynamically/try {// We don't really need to store this value since we never// use it again, but if we don't store it, Closure Compiler// assumes the code is useless and removes it.hasRules = !!req.node.sheet.cssRules;// If we get here, the stylesheet has loaded.pendingCSS.splice(i, 1);i -= 1;self._progress(null, req);} catch (ex) {// An exception means the stylesheet is still loading.}}}if (pendingCSS.length) {self._pollTimer = setTimeout(function () {self._poll.call(self);}, self.options.pollInterval);}},_progress: function (err, req) {var options = this.options;if (err) {req.error = err;this.errors.push({error : err,request: req});}req.node._yuiget_finished = req.finished = true;if (options.onProgress) {options.onProgress.call(options.context || this,this._getEventData(req));}if (req.autopurge) {// Pre-3.5.0 Get always excludes the most recent node from an// autopurge. I find this odd, but I'm keeping that behavior for// the sake of backcompat.Get._autoPurge(this.options.purgethreshold);Get._purgeNodes.push(req.node);}if (this._pending === req) {this._pending = null;}this._reqsWaiting -= 1;this._next();}};}, '3.18.1', {"requires": ["yui-base"]});