(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.Loader = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o 0 && arguments[0] !== undefined ? arguments[0] : ''; var concurrency = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 10; _classCallCheck(this, Loader); /** * The base url for all resources loaded by this loader. * * @member {string} */ this.baseUrl = baseUrl; /** * The progress percent of the loader going through the queue. * * @member {number} */ this.progress = 0; /** * Loading state of the loader, true if it is currently loading resources. * * @member {boolean} */ this.loading = false; /** * A querystring to append to every URL added to the loader. * * This should be a valid query string *without* the question-mark (`?`). The loader will * also *not* escape values for you. Make sure to escape your parameters with * [`encodeURIComponent`](https://mdn.io/encodeURIComponent) before assigning this property. * * @example * const loader = new Loader(); * * loader.defaultQueryString = 'user=me&password=secret'; * * // This will request 'image.png?user=me&password=secret' * loader.add('image.png').load(); * * loader.reset(); * * // This will request 'image.png?v=1&user=me&password=secret' * loader.add('iamge.png?v=1').load(); */ this.defaultQueryString = ''; /** * The middleware to run before loading each resource. * * @member {function[]} */ this._beforeMiddleware = []; /** * The middleware to run after loading each resource. * * @member {function[]} */ this._afterMiddleware = []; /** * The tracks the resources we are currently completing parsing for. * * @member {Resource[]} */ this._resourcesParsing = []; /** * The `_loadResource` function bound with this object context. * * @private * @member {function} * @param {Resource} r - The resource to load * @param {Function} d - The dequeue function * @return {undefined} */ this._boundLoadResource = function (r, d) { return _this._loadResource(r, d); }; /** * The resources waiting to be loaded. * * @private * @member {Resource[]} */ this._queue = async.queue(this._boundLoadResource, concurrency); this._queue.pause(); /** * All the resources for this loader keyed by name. * * @member {object} */ this.resources = {}; /** * Dispatched once per loaded or errored resource. * * The callback looks like {@link Loader.OnProgressSignal}. * * @member {Signal} */ this.onProgress = new _miniSignals2.default(); /** * Dispatched once per errored resource. * * The callback looks like {@link Loader.OnErrorSignal}. * * @member {Signal} */ this.onError = new _miniSignals2.default(); /** * Dispatched once per loaded resource. * * The callback looks like {@link Loader.OnLoadSignal}. * * @member {Signal} */ this.onLoad = new _miniSignals2.default(); /** * Dispatched when the loader begins to process the queue. * * The callback looks like {@link Loader.OnStartSignal}. * * @member {Signal} */ this.onStart = new _miniSignals2.default(); /** * Dispatched when the queued resources all load. * * The callback looks like {@link Loader.OnCompleteSignal}. * * @member {Signal} */ this.onComplete = new _miniSignals2.default(); /** * When the progress changes the loader and resource are disaptched. * * @memberof Loader * @callback OnProgressSignal * @param {Loader} loader - The loader the progress is advancing on. * @param {Resource} resource - The resource that has completed or failed to cause the progress to advance. */ /** * When an error occurrs the loader and resource are disaptched. * * @memberof Loader * @callback OnErrorSignal * @param {Loader} loader - The loader the error happened in. * @param {Resource} resource - The resource that caused the error. */ /** * When a load completes the loader and resource are disaptched. * * @memberof Loader * @callback OnLoadSignal * @param {Loader} loader - The loader that laoded the resource. * @param {Resource} resource - The resource that has completed loading. */ /** * When the loader starts loading resources it dispatches this callback. * * @memberof Loader * @callback OnStartSignal * @param {Loader} loader - The loader that has started loading resources. */ /** * When the loader completes loading resources it dispatches this callback. * * @memberof Loader * @callback OnCompleteSignal * @param {Loader} loader - The loader that has finished loading resources. */ } /** * Adds a resource (or multiple resources) to the loader queue. * * This function can take a wide variety of different parameters. The only thing that is always * required the url to load. All the following will work: * * ```js * loader * // normal param syntax * .add('key', 'http://...', function () {}) * .add('http://...', function () {}) * .add('http://...') * * // object syntax * .add({ * name: 'key2', * url: 'http://...' * }, function () {}) * .add({ * url: 'http://...' * }, function () {}) * .add({ * name: 'key3', * url: 'http://...' * onComplete: function () {} * }) * .add({ * url: 'https://...', * onComplete: function () {}, * crossOrigin: true * }) * * // you can also pass an array of objects or urls or both * .add([ * { name: 'key4', url: 'http://...', onComplete: function () {} }, * { url: 'http://...', onComplete: function () {} }, * 'http://...' * ]) * * // and you can use both params and options * .add('key', 'http://...', { crossOrigin: true }, function () {}) * .add('http://...', { crossOrigin: true }, function () {}); * ``` * * @param {string} [name] - The name of the resource to load, if not passed the url is used. * @param {string} [url] - The url for this resource, relative to the baseUrl of this loader. * @param {object} [options] - The options for the load. * @param {boolean} [options.crossOrigin] - Is this request cross-origin? Default is to determine automatically. * @param {Resource.LOAD_TYPE} [options.loadType=Resource.LOAD_TYPE.XHR] - How should this resource be loaded? * @param {Resource.XHR_RESPONSE_TYPE} [options.xhrType=Resource.XHR_RESPONSE_TYPE.DEFAULT] - How should * the data being loaded be interpreted when using XHR? * @param {object} [options.metadata] - Extra configuration for middleware and the Resource object. * @param {HTMLImageElement|HTMLAudioElement|HTMLVideoElement} [options.metadata.loadElement=null] - The * element to use for loading, instead of creating one. * @param {boolean} [options.metadata.skipSource=false] - Skips adding source(s) to the load element. This * is useful if you want to pass in a `loadElement` that you already added load sources to. * @param {function} [cb] - Function to call when this specific resource completes loading. * @return {Loader} Returns itself. */ Loader.prototype.add = function add(name, url, options, cb) { // special case of an array of objects or urls if (Array.isArray(name)) { for (var i = 0; i < name.length; ++i) { this.add(name[i]); } return this; } // if an object is passed instead of params if ((typeof name === 'undefined' ? 'undefined' : _typeof(name)) === 'object') { cb = url || name.callback || name.onComplete; options = name; url = name.url; name = name.name || name.key || name.url; } // case where no name is passed shift all args over by one. if (typeof url !== 'string') { cb = options; options = url; url = name; } // now that we shifted make sure we have a proper url. if (typeof url !== 'string') { throw new Error('No url passed to add resource to loader.'); } // options are optional so people might pass a function and no options if (typeof options === 'function') { cb = options; options = null; } // if loading already you can only add resources that have a parent. if (this.loading && (!options || !options.parentResource)) { throw new Error('Cannot add resources while the loader is running.'); } // check if resource already exists. if (this.resources[name]) { throw new Error('Resource named "' + name + '" already exists.'); } // add base url if this isn't an absolute url url = this._prepareUrl(url); // create the store the resource this.resources[name] = new _Resource2.default(name, url, options); if (typeof cb === 'function') { this.resources[name].onAfterMiddleware.once(cb); } // if actively loading, make sure to adjust progress chunks for that parent and its children if (this.loading) { var parent = options.parentResource; var incompleteChildren = []; for (var _i = 0; _i < parent.children.length; ++_i) { if (!parent.children[_i].isComplete) { incompleteChildren.push(parent.children[_i]); } } var fullChunk = parent.progressChunk * (incompleteChildren.length + 1); // +1 for parent var eachChunk = fullChunk / (incompleteChildren.length + 2); // +2 for parent & new child parent.children.push(this.resources[name]); parent.progressChunk = eachChunk; for (var _i2 = 0; _i2 < incompleteChildren.length; ++_i2) { incompleteChildren[_i2].progressChunk = eachChunk; } this.resources[name].progressChunk = eachChunk; } // add the resource to the queue this._queue.push(this.resources[name]); return this; }; /** * Sets up a middleware function that will run *before* the * resource is loaded. * * @method before * @param {function} fn - The middleware function to register. * @return {Loader} Returns itself. */ Loader.prototype.pre = function pre(fn) { this._beforeMiddleware.push(fn); return this; }; /** * Sets up a middleware function that will run *after* the * resource is loaded. * * @alias use * @method after * @param {function} fn - The middleware function to register. * @return {Loader} Returns itself. */ Loader.prototype.use = function use(fn) { this._afterMiddleware.push(fn); return this; }; /** * Resets the queue of the loader to prepare for a new load. * * @return {Loader} Returns itself. */ Loader.prototype.reset = function reset() { this.progress = 0; this.loading = false; this._queue.kill(); this._queue.pause(); // abort all resource loads for (var k in this.resources) { var res = this.resources[k]; if (res._onLoadBinding) { res._onLoadBinding.detach(); } if (res.isLoading) { res.abort(); } } this.resources = {}; return this; }; /** * Starts loading the queued resources. * * @param {function} [cb] - Optional callback that will be bound to the `complete` event. * @return {Loader} Returns itself. */ Loader.prototype.load = function load(cb) { // register complete callback if they pass one if (typeof cb === 'function') { this.onComplete.once(cb); } // if the queue has already started we are done here if (this.loading) { return this; } if (this._queue.idle()) { this._onStart(); this._onComplete(); } else { // distribute progress chunks var numTasks = this._queue._tasks.length; var chunk = 100 / numTasks; for (var i = 0; i < this._queue._tasks.length; ++i) { this._queue._tasks[i].data.progressChunk = chunk; } // notify we are starting this._onStart(); // start loading this._queue.resume(); } return this; }; /** * The number of resources to load concurrently. * * @member {number} * @default 10 */ /** * Prepares a url for usage based on the configuration of this object * * @private * @param {string} url - The url to prepare. * @return {string} The prepared url. */ Loader.prototype._prepareUrl = function _prepareUrl(url) { var parsedUrl = (0, _parseUri2.default)(url, { strictMode: true }); var result = void 0; // absolute url, just use it as is. if (parsedUrl.protocol || !parsedUrl.path || url.indexOf('//') === 0) { result = url; } // if baseUrl doesn't end in slash and url doesn't start with slash, then add a slash inbetween else if (this.baseUrl.length && this.baseUrl.lastIndexOf('/') !== this.baseUrl.length - 1 && url.charAt(0) !== '/') { result = this.baseUrl + '/' + url; } else { result = this.baseUrl + url; } // if we need to add a default querystring, there is a bit more work if (this.defaultQueryString) { var hash = rgxExtractUrlHash.exec(result)[0]; result = result.substr(0, result.length - hash.length); if (result.indexOf('?') !== -1) { result += '&' + this.defaultQueryString; } else { result += '?' + this.defaultQueryString; } result += hash; } return result; }; /** * Loads a single resource. * * @private * @param {Resource} resource - The resource to load. * @param {function} dequeue - The function to call when we need to dequeue this item. */ Loader.prototype._loadResource = function _loadResource(resource, dequeue) { var _this2 = this; resource._dequeue = dequeue; // run before middleware async.eachSeries(this._beforeMiddleware, function (fn, next) { fn.call(_this2, resource, function () { // if the before middleware marks the resource as complete, // break and don't process any more before middleware next(resource.isComplete ? {} : null); }); }, function () { if (resource.isComplete) { _this2._onLoad(resource); } else { resource._onLoadBinding = resource.onComplete.once(_this2._onLoad, _this2); resource.load(); } }, true); }; /** * Called once loading has started. * * @private */ Loader.prototype._onStart = function _onStart() { this.progress = 0; this.loading = true; this.onStart.dispatch(this); }; /** * Called once each resource has loaded. * * @private */ Loader.prototype._onComplete = function _onComplete() { this.progress = MAX_PROGRESS; this.loading = false; this.onComplete.dispatch(this, this.resources); }; /** * Called each time a resources is loaded. * * @private * @param {Resource} resource - The resource that was loaded */ Loader.prototype._onLoad = function _onLoad(resource) { var _this3 = this; resource._onLoadBinding = null; // remove this resource from the async queue, and add it to our list of resources that are being parsed this._resourcesParsing.push(resource); resource._dequeue(); // run all the after middleware for this resource async.eachSeries(this._afterMiddleware, function (fn, next) { fn.call(_this3, resource, next); }, function () { resource.onAfterMiddleware.dispatch(resource); _this3.progress += resource.progressChunk; _this3.onProgress.dispatch(_this3, resource); if (resource.error) { _this3.onError.dispatch(resource.error, _this3, resource); } else { _this3.onLoad.dispatch(_this3, resource); } _this3._resourcesParsing.splice(_this3._resourcesParsing.indexOf(resource), 1); // do completion check if (_this3._queue.idle() && _this3._resourcesParsing.length === 0) { _this3._onComplete(); } }, true); }; _createClass(Loader, [{ key: 'concurrency', get: function get() { return this._queue.concurrency; } // eslint-disable-next-line require-jsdoc , set: function set(concurrency) { this._queue.concurrency = concurrency; } }]); return Loader; }(); exports.default = Loader; },{"./Resource":2,"./async":3,"mini-signals":6,"parse-uri":7}],2:[function(require,module,exports){ 'use strict'; exports.__esModule = true; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); var _parseUri = require('parse-uri'); var _parseUri2 = _interopRequireDefault(_parseUri); var _miniSignals = require('mini-signals'); var _miniSignals2 = _interopRequireDefault(_miniSignals); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } // tests is CORS is supported in XHR, if not we need to use XDR var useXdr = !!(window.XDomainRequest && !('withCredentials' in new XMLHttpRequest())); var tempAnchor = null; // some status constants var STATUS_NONE = 0; var STATUS_OK = 200; var STATUS_EMPTY = 204; var STATUS_IE_BUG_EMPTY = 1223; var STATUS_TYPE_OK = 2; // noop function _noop() {} /* empty */ /** * Manages the state and loading of a resource and all child resources. * * @class */ var Resource = function () { /** * Sets the load type to be used for a specific extension. * * @static * @param {string} extname - The extension to set the type for, e.g. "png" or "fnt" * @param {Resource.LOAD_TYPE} loadType - The load type to set it to. */ Resource.setExtensionLoadType = function setExtensionLoadType(extname, loadType) { setExtMap(Resource._loadTypeMap, extname, loadType); }; /** * Sets the load type to be used for a specific extension. * * @static * @param {string} extname - The extension to set the type for, e.g. "png" or "fnt" * @param {Resource.XHR_RESPONSE_TYPE} xhrType - The xhr type to set it to. */ Resource.setExtensionXhrType = function setExtensionXhrType(extname, xhrType) { setExtMap(Resource._xhrTypeMap, extname, xhrType); }; /** * @param {string} name - The name of the resource to load. * @param {string|string[]} url - The url for this resource, for audio/video loads you can pass * an array of sources. * @param {object} [options] - The options for the load. * @param {string|boolean} [options.crossOrigin] - Is this request cross-origin? Default is to * determine automatically. * @param {Resource.LOAD_TYPE} [options.loadType=Resource.LOAD_TYPE.XHR] - How should this resource * be loaded? * @param {Resource.XHR_RESPONSE_TYPE} [options.xhrType=Resource.XHR_RESPONSE_TYPE.DEFAULT] - How * should the data being loaded be interpreted when using XHR? * @param {object} [options.metadata] - Extra configuration for middleware and the Resource object. * @param {HTMLImageElement|HTMLAudioElement|HTMLVideoElement} [options.metadata.loadElement=null] - The * element to use for loading, instead of creating one. * @param {boolean} [options.metadata.skipSource=false] - Skips adding source(s) to the load element. This * is useful if you want to pass in a `loadElement` that you already added load sources to. * @param {string|string[]} [options.metadata.mimeType] - The mime type to use for the source element of a video/audio * elment. If the urls are an array, you can pass this as an array as well where each index is the mime type to * use for the corresponding url index. */ function Resource(name, url, options) { _classCallCheck(this, Resource); if (typeof name !== 'string' || typeof url !== 'string') { throw new Error('Both name and url are required for constructing a resource.'); } options = options || {}; /** * The state flags of this resource. * * @member {number} */ this._flags = 0; // set data url flag, needs to be set early for some _determineX checks to work. this._setFlag(Resource.STATUS_FLAGS.DATA_URL, url.indexOf('data:') === 0); /** * The name of this resource. * * @member {string} * @readonly */ this.name = name; /** * The url used to load this resource. * * @member {string} * @readonly */ this.url = url; /** * The extension used to load this resource. * * @member {string} * @readonly */ this.extension = this._getExtension(); /** * The data that was loaded by the resource. * * @member {any} */ this.data = null; /** * Is this request cross-origin? If unset, determined automatically. * * @member {string} */ this.crossOrigin = options.crossOrigin === true ? 'anonymous' : options.crossOrigin; /** * The method of loading to use for this resource. * * @member {Resource.LOAD_TYPE} */ this.loadType = options.loadType || this._determineLoadType(); /** * The type used to load the resource via XHR. If unset, determined automatically. * * @member {string} */ this.xhrType = options.xhrType; /** * Extra info for middleware, and controlling specifics about how the resource loads. * * Note that if you pass in a `loadElement`, the Resource class takes ownership of it. * Meaning it will modify it as it sees fit. * * @member {object} * @property {HTMLImageElement|HTMLAudioElement|HTMLVideoElement} [loadElement=null] - The * element to use for loading, instead of creating one. * @property {boolean} [skipSource=false] - Skips adding source(s) to the load element. This * is useful if you want to pass in a `loadElement` that you already added load sources * to. */ this.metadata = options.metadata || {}; /** * The error that occurred while loading (if any). * * @member {Error} * @readonly */ this.error = null; /** * The XHR object that was used to load this resource. This is only set * when `loadType` is `Resource.LOAD_TYPE.XHR`. * * @member {XMLHttpRequest} * @readonly */ this.xhr = null; /** * The child resources this resource owns. * * @member {Resource[]} * @readonly */ this.children = []; /** * The resource type. * * @member {Resource.TYPE} * @readonly */ this.type = Resource.TYPE.UNKNOWN; /** * The progress chunk owned by this resource. * * @member {number} * @readonly */ this.progressChunk = 0; /** * The `dequeue` method that will be used a storage place for the async queue dequeue method * used privately by the loader. * * @private * @member {function} */ this._dequeue = _noop; /** * Used a storage place for the on load binding used privately by the loader. * * @private * @member {function} */ this._onLoadBinding = null; /** * The `complete` function bound to this resource's context. * * @private * @member {function} */ this._boundComplete = this.complete.bind(this); /** * The `_onError` function bound to this resource's context. * * @private * @member {function} */ this._boundOnError = this._onError.bind(this); /** * The `_onProgress` function bound to this resource's context. * * @private * @member {function} */ this._boundOnProgress = this._onProgress.bind(this); // xhr callbacks this._boundXhrOnError = this._xhrOnError.bind(this); this._boundXhrOnAbort = this._xhrOnAbort.bind(this); this._boundXhrOnLoad = this._xhrOnLoad.bind(this); this._boundXdrOnTimeout = this._xdrOnTimeout.bind(this); /** * Dispatched when the resource beings to load. * * The callback looks like {@link Resource.OnStartSignal}. * * @member {Signal} */ this.onStart = new _miniSignals2.default(); /** * Dispatched each time progress of this resource load updates. * Not all resources types and loader systems can support this event * so sometimes it may not be available. If the resource * is being loaded on a modern browser, using XHR, and the remote server * properly sets Content-Length headers, then this will be available. * * The callback looks like {@link Resource.OnProgressSignal}. * * @member {Signal} */ this.onProgress = new _miniSignals2.default(); /** * Dispatched once this resource has loaded, if there was an error it will * be in the `error` property. * * The callback looks like {@link Resource.OnCompleteSignal}. * * @member {Signal} */ this.onComplete = new _miniSignals2.default(); /** * Dispatched after this resource has had all the *after* middleware run on it. * * The callback looks like {@link Resource.OnCompleteSignal}. * * @member {Signal} */ this.onAfterMiddleware = new _miniSignals2.default(); /** * When the resource starts to load. * * @memberof Resource * @callback OnStartSignal * @param {Resource} resource - The resource that the event happened on. */ /** * When the resource reports loading progress. * * @memberof Resource * @callback OnProgressSignal * @param {Resource} resource - The resource that the event happened on. * @param {number} percentage - The progress of the load in the range [0, 1]. */ /** * When the resource finishes loading. * * @memberof Resource * @callback OnCompleteSignal * @param {Resource} resource - The resource that the event happened on. */ } /** * Stores whether or not this url is a data url. * * @member {boolean} * @readonly */ /** * Marks the resource as complete. * */ Resource.prototype.complete = function complete() { // TODO: Clean this up in a wrapper or something...gross.... if (this.data && this.data.removeEventListener) { this.data.removeEventListener('error', this._boundOnError, false); this.data.removeEventListener('load', this._boundComplete, false); this.data.removeEventListener('progress', this._boundOnProgress, false); this.data.removeEventListener('canplaythrough', this._boundComplete, false); } if (this.xhr) { if (this.xhr.removeEventListener) { this.xhr.removeEventListener('error', this._boundXhrOnError, false); this.xhr.removeEventListener('abort', this._boundXhrOnAbort, false); this.xhr.removeEventListener('progress', this._boundOnProgress, false); this.xhr.removeEventListener('load', this._boundXhrOnLoad, false); } else { this.xhr.onerror = null; this.xhr.ontimeout = null; this.xhr.onprogress = null; this.xhr.onload = null; } } if (this.isComplete) { throw new Error('Complete called again for an already completed resource.'); } this._setFlag(Resource.STATUS_FLAGS.COMPLETE, true); this._setFlag(Resource.STATUS_FLAGS.LOADING, false); this.onComplete.dispatch(this); }; /** * Aborts the loading of this resource, with an optional message. * * @param {string} message - The message to use for the error */ Resource.prototype.abort = function abort(message) { // abort can be called multiple times, ignore subsequent calls. if (this.error) { return; } // store error this.error = new Error(message); // abort the actual loading if (this.xhr) { this.xhr.abort(); } else if (this.xdr) { this.xdr.abort(); } else if (this.data) { // single source if (this.data.src) { this.data.src = Resource.EMPTY_GIF; } // multi-source else { while (this.data.firstChild) { this.data.removeChild(this.data.firstChild); } } } // done now. this.complete(); }; /** * Kicks off loading of this resource. This method is asynchronous. * * @param {function} [cb] - Optional callback to call once the resource is loaded. */ Resource.prototype.load = function load(cb) { var _this = this; if (this.isLoading) { return; } if (this.isComplete) { if (cb) { setTimeout(function () { return cb(_this); }, 1); } return; } else if (cb) { this.onComplete.once(cb); } this._setFlag(Resource.STATUS_FLAGS.LOADING, true); this.onStart.dispatch(this); // if unset, determine the value if (this.crossOrigin === false || typeof this.crossOrigin !== 'string') { this.crossOrigin = this._determineCrossOrigin(this.url); } switch (this.loadType) { case Resource.LOAD_TYPE.IMAGE: this.type = Resource.TYPE.IMAGE; this._loadElement('image'); break; case Resource.LOAD_TYPE.AUDIO: this.type = Resource.TYPE.AUDIO; this._loadSourceElement('audio'); break; case Resource.LOAD_TYPE.VIDEO: this.type = Resource.TYPE.VIDEO; this._loadSourceElement('video'); break; case Resource.LOAD_TYPE.XHR: /* falls through */ default: if (useXdr && this.crossOrigin) { this._loadXdr(); } else { this._loadXhr(); } break; } }; /** * Checks if the flag is set. * * @private * @param {number} flag - The flag to check. * @return {boolean} True if the flag is set. */ Resource.prototype._hasFlag = function _hasFlag(flag) { return !!(this._flags & flag); }; /** * (Un)Sets the flag. * * @private * @param {number} flag - The flag to (un)set. * @param {boolean} value - Whether to set or (un)set the flag. */ Resource.prototype._setFlag = function _setFlag(flag, value) { this._flags = value ? this._flags | flag : this._flags & ~flag; }; /** * Loads this resources using an element that has a single source, * like an HTMLImageElement. * * @private * @param {string} type - The type of element to use. */ Resource.prototype._loadElement = function _loadElement(type) { if (this.metadata.loadElement) { this.data = this.metadata.loadElement; } else if (type === 'image' && typeof window.Image !== 'undefined') { this.data = new Image(); } else { this.data = document.createElement(type); } if (this.crossOrigin) { this.data.crossOrigin = this.crossOrigin; } if (!this.metadata.skipSource) { this.data.src = this.url; } this.data.addEventListener('error', this._boundOnError, false); this.data.addEventListener('load', this._boundComplete, false); this.data.addEventListener('progress', this._boundOnProgress, false); }; /** * Loads this resources using an element that has multiple sources, * like an HTMLAudioElement or HTMLVideoElement. * * @private * @param {string} type - The type of element to use. */ Resource.prototype._loadSourceElement = function _loadSourceElement(type) { if (this.metadata.loadElement) { this.data = this.metadata.loadElement; } else if (type === 'audio' && typeof window.Audio !== 'undefined') { this.data = new Audio(); } else { this.data = document.createElement(type); } if (this.data === null) { this.abort('Unsupported element: ' + type); return; } if (!this.metadata.skipSource) { // support for CocoonJS Canvas+ runtime, lacks document.createElement('source') if (navigator.isCocoonJS) { this.data.src = Array.isArray(this.url) ? this.url[0] : this.url; } else if (Array.isArray(this.url)) { var mimeTypes = this.metadata.mimeType; for (var i = 0; i < this.url.length; ++i) { this.data.appendChild(this._createSource(type, this.url[i], Array.isArray(mimeTypes) ? mimeTypes[i] : mimeTypes)); } } else { var _mimeTypes = this.metadata.mimeType; this.data.appendChild(this._createSource(type, this.url, Array.isArray(_mimeTypes) ? _mimeTypes[0] : _mimeTypes)); } } this.data.addEventListener('error', this._boundOnError, false); this.data.addEventListener('load', this._boundComplete, false); this.data.addEventListener('progress', this._boundOnProgress, false); this.data.addEventListener('canplaythrough', this._boundComplete, false); this.data.load(); }; /** * Loads this resources using an XMLHttpRequest. * * @private */ Resource.prototype._loadXhr = function _loadXhr() { // if unset, determine the value if (typeof this.xhrType !== 'string') { this.xhrType = this._determineXhrType(); } var xhr = this.xhr = new XMLHttpRequest(); // set the request type and url xhr.open('GET', this.url, true); // load json as text and parse it ourselves. We do this because some browsers // *cough* safari *cough* can't deal with it. if (this.xhrType === Resource.XHR_RESPONSE_TYPE.JSON || this.xhrType === Resource.XHR_RESPONSE_TYPE.DOCUMENT) { xhr.responseType = Resource.XHR_RESPONSE_TYPE.TEXT; } else { xhr.responseType = this.xhrType; } xhr.addEventListener('error', this._boundXhrOnError, false); xhr.addEventListener('abort', this._boundXhrOnAbort, false); xhr.addEventListener('progress', this._boundOnProgress, false); xhr.addEventListener('load', this._boundXhrOnLoad, false); xhr.send(); }; /** * Loads this resources using an XDomainRequest. This is here because we need to support IE9 (gross). * * @private */ Resource.prototype._loadXdr = function _loadXdr() { // if unset, determine the value if (typeof this.xhrType !== 'string') { this.xhrType = this._determineXhrType(); } var xdr = this.xhr = new XDomainRequest(); // XDomainRequest has a few quirks. Occasionally it will abort requests // A way to avoid this is to make sure ALL callbacks are set even if not used // More info here: http://stackoverflow.com/questions/15786966/xdomainrequest-aborts-post-on-ie-9 xdr.timeout = 5000; xdr.onerror = this._boundXhrOnError; xdr.ontimeout = this._boundXdrOnTimeout; xdr.onprogress = this._boundOnProgress; xdr.onload = this._boundXhrOnLoad; xdr.open('GET', this.url, true); // Note: The xdr.send() call is wrapped in a timeout to prevent an // issue with the interface where some requests are lost if multiple // XDomainRequests are being sent at the same time. // Some info here: https://github.com/photonstorm/phaser/issues/1248 setTimeout(function () { return xdr.send(); }, 1); }; /** * Creates a source used in loading via an element. * * @private * @param {string} type - The element type (video or audio). * @param {string} url - The source URL to load from. * @param {string} [mime] - The mime type of the video * @return {HTMLSourceElement} The source element. */ Resource.prototype._createSource = function _createSource(type, url, mime) { if (!mime) { mime = type + '/' + this._getExtension(url); } var source = document.createElement('source'); source.src = url; source.type = mime; return source; }; /** * Called if a load errors out. * * @param {Event} event - The error event from the element that emits it. * @private */ Resource.prototype._onError = function _onError(event) { this.abort('Failed to load element using: ' + event.target.nodeName); }; /** * Called if a load progress event fires for xhr/xdr. * * @private * @param {XMLHttpRequestProgressEvent|Event} event - Progress event. */ Resource.prototype._onProgress = function _onProgress(event) { if (event && event.lengthComputable) { this.onProgress.dispatch(this, event.loaded / event.total); } }; /** * Called if an error event fires for xhr/xdr. * * @private * @param {XMLHttpRequestErrorEvent|Event} event - Error event. */ Resource.prototype._xhrOnError = function _xhrOnError() { var xhr = this.xhr; this.abort(reqType(xhr) + ' Request failed. Status: ' + xhr.status + ', text: "' + xhr.statusText + '"'); }; /** * Called if an abort event fires for xhr. * * @private * @param {XMLHttpRequestAbortEvent} event - Abort Event */ Resource.prototype._xhrOnAbort = function _xhrOnAbort() { this.abort(reqType(this.xhr) + ' Request was aborted by the user.'); }; /** * Called if a timeout event fires for xdr. * * @private * @param {Event} event - Timeout event. */ Resource.prototype._xdrOnTimeout = function _xdrOnTimeout() { this.abort(reqType(this.xhr) + ' Request timed out.'); }; /** * Called when data successfully loads from an xhr/xdr request. * * @private * @param {XMLHttpRequestLoadEvent|Event} event - Load event */ Resource.prototype._xhrOnLoad = function _xhrOnLoad() { var xhr = this.xhr; var text = ''; var status = typeof xhr.status === 'undefined' ? STATUS_OK : xhr.status; // XDR has no `.status`, assume 200. // responseText is accessible only if responseType is '' or 'text' and on older browsers if (xhr.responseType === '' || xhr.responseType === 'text' || typeof xhr.responseType === 'undefined') { text = xhr.responseText; } // status can be 0 when using the `file://` protocol so we also check if a response is set. // If it has a response, we assume 200; otherwise a 0 status code with no contents is an aborted request. if (status === STATUS_NONE && (text.length > 0 || xhr.responseType === Resource.XHR_RESPONSE_TYPE.BUFFER)) { status = STATUS_OK; } // handle IE9 bug: http://stackoverflow.com/questions/10046972/msie-returns-status-code-of-1223-for-ajax-request else if (status === STATUS_IE_BUG_EMPTY) { status = STATUS_EMPTY; } var statusType = status / 100 | 0; if (statusType === STATUS_TYPE_OK) { // if text, just return it if (this.xhrType === Resource.XHR_RESPONSE_TYPE.TEXT) { this.data = text; this.type = Resource.TYPE.TEXT; } // if json, parse into json object else if (this.xhrType === Resource.XHR_RESPONSE_TYPE.JSON) { try { this.data = JSON.parse(text); this.type = Resource.TYPE.JSON; } catch (e) { this.abort('Error trying to parse loaded json: ' + e); return; } } // if xml, parse into an xml document or div element else if (this.xhrType === Resource.XHR_RESPONSE_TYPE.DOCUMENT) { try { if (window.DOMParser) { var domparser = new DOMParser(); this.data = domparser.parseFromString(text, 'text/xml'); } else { var div = document.createElement('div'); div.innerHTML = text; this.data = div; } this.type = Resource.TYPE.XML; } catch (e) { this.abort('Error trying to parse loaded xml: ' + e); return; } } // other types just return the response else { this.data = xhr.response || text; } } else { this.abort('[' + xhr.status + '] ' + xhr.statusText + ': ' + xhr.responseURL); return; } this.complete(); }; /** * Sets the `crossOrigin` property for this resource based on if the url * for this resource is cross-origin. If crossOrigin was manually set, this * function does nothing. * * @private * @param {string} url - The url to test. * @param {object} [loc=window.location] - The location object to test against. * @return {string} The crossOrigin value to use (or empty string for none). */ Resource.prototype._determineCrossOrigin = function _determineCrossOrigin(url, loc) { // data: and javascript: urls are considered same-origin if (url.indexOf('data:') === 0) { return ''; } // default is window.location loc = loc || window.location; if (!tempAnchor) { tempAnchor = document.createElement('a'); } // let the browser determine the full href for the url of this resource and then // parse with the node url lib, we can't use the properties of the anchor element // because they don't work in IE9 :( tempAnchor.href = url; url = (0, _parseUri2.default)(tempAnchor.href, { strictMode: true }); var samePort = !url.port && loc.port === '' || url.port === loc.port; var protocol = url.protocol ? url.protocol + ':' : ''; // if cross origin if (url.host !== loc.hostname || !samePort || protocol !== loc.protocol) { return 'anonymous'; } return ''; }; /** * Determines the responseType of an XHR request based on the extension of the * resource being loaded. * * @private * @return {Resource.XHR_RESPONSE_TYPE} The responseType to use. */ Resource.prototype._determineXhrType = function _determineXhrType() { return Resource._xhrTypeMap[this.extension] || Resource.XHR_RESPONSE_TYPE.TEXT; }; /** * Determines the loadType of a resource based on the extension of the * resource being loaded. * * @private * @return {Resource.LOAD_TYPE} The loadType to use. */ Resource.prototype._determineLoadType = function _determineLoadType() { return Resource._loadTypeMap[this.extension] || Resource.LOAD_TYPE.XHR; }; /** * Extracts the extension (sans '.') of the file being loaded by the resource. * * @private * @return {string} The extension. */ Resource.prototype._getExtension = function _getExtension() { var url = this.url; var ext = ''; if (this.isDataUrl) { var slashIndex = url.indexOf('/'); ext = url.substring(slashIndex + 1, url.indexOf(';', slashIndex)); } else { var queryStart = url.indexOf('?'); var hashStart = url.indexOf('#'); var index = Math.min(queryStart > -1 ? queryStart : url.length, hashStart > -1 ? hashStart : url.length); url = url.substring(0, index); ext = url.substring(url.lastIndexOf('.') + 1); } return ext.toLowerCase(); }; /** * Determines the mime type of an XHR request based on the responseType of * resource being loaded. * * @private * @param {Resource.XHR_RESPONSE_TYPE} type - The type to get a mime type for. * @return {string} The mime type to use. */ Resource.prototype._getMimeFromXhrType = function _getMimeFromXhrType(type) { switch (type) { case Resource.XHR_RESPONSE_TYPE.BUFFER: return 'application/octet-binary'; case Resource.XHR_RESPONSE_TYPE.BLOB: return 'application/blob'; case Resource.XHR_RESPONSE_TYPE.DOCUMENT: return 'application/xml'; case Resource.XHR_RESPONSE_TYPE.JSON: return 'application/json'; case Resource.XHR_RESPONSE_TYPE.DEFAULT: case Resource.XHR_RESPONSE_TYPE.TEXT: /* falls through */ default: return 'text/plain'; } }; _createClass(Resource, [{ key: 'isDataUrl', get: function get() { return this._hasFlag(Resource.STATUS_FLAGS.DATA_URL); } /** * Describes if this resource has finished loading. Is true when the resource has completely * loaded. * * @member {boolean} * @readonly */ }, { key: 'isComplete', get: function get() { return this._hasFlag(Resource.STATUS_FLAGS.COMPLETE); } /** * Describes if this resource is currently loading. Is true when the resource starts loading, * and is false again when complete. * * @member {boolean} * @readonly */ }, { key: 'isLoading', get: function get() { return this._hasFlag(Resource.STATUS_FLAGS.LOADING); } }]); return Resource; }(); /** * The types of resources a resource could represent. * * @static * @readonly * @enum {number} */ exports.default = Resource; Resource.STATUS_FLAGS = { NONE: 0, DATA_URL: 1 << 0, COMPLETE: 1 << 1, LOADING: 1 << 2 }; /** * The types of resources a resource could represent. * * @static * @readonly * @enum {number} */ Resource.TYPE = { UNKNOWN: 0, JSON: 1, XML: 2, IMAGE: 3, AUDIO: 4, VIDEO: 5, TEXT: 6 }; /** * The types of loading a resource can use. * * @static * @readonly * @enum {number} */ Resource.LOAD_TYPE = { /** Uses XMLHttpRequest to load the resource. */ XHR: 1, /** Uses an `Image` object to load the resource. */ IMAGE: 2, /** Uses an `Audio` object to load the resource. */ AUDIO: 3, /** Uses a `Video` object to load the resource. */ VIDEO: 4 }; /** * The XHR ready states, used internally. * * @static * @readonly * @enum {string} */ Resource.XHR_RESPONSE_TYPE = { /** string */ DEFAULT: 'text', /** ArrayBuffer */ BUFFER: 'arraybuffer', /** Blob */ BLOB: 'blob', /** Document */ DOCUMENT: 'document', /** Object */ JSON: 'json', /** String */ TEXT: 'text' }; Resource._loadTypeMap = { // images gif: Resource.LOAD_TYPE.IMAGE, png: Resource.LOAD_TYPE.IMAGE, bmp: Resource.LOAD_TYPE.IMAGE, jpg: Resource.LOAD_TYPE.IMAGE, jpeg: Resource.LOAD_TYPE.IMAGE, tif: Resource.LOAD_TYPE.IMAGE, tiff: Resource.LOAD_TYPE.IMAGE, webp: Resource.LOAD_TYPE.IMAGE, tga: Resource.LOAD_TYPE.IMAGE, svg: Resource.LOAD_TYPE.IMAGE, 'svg+xml': Resource.LOAD_TYPE.IMAGE, // for SVG data urls // audio mp3: Resource.LOAD_TYPE.AUDIO, ogg: Resource.LOAD_TYPE.AUDIO, wav: Resource.LOAD_TYPE.AUDIO, // videos mp4: Resource.LOAD_TYPE.VIDEO, webm: Resource.LOAD_TYPE.VIDEO }; Resource._xhrTypeMap = { // xml xhtml: Resource.XHR_RESPONSE_TYPE.DOCUMENT, html: Resource.XHR_RESPONSE_TYPE.DOCUMENT, htm: Resource.XHR_RESPONSE_TYPE.DOCUMENT, xml: Resource.XHR_RESPONSE_TYPE.DOCUMENT, tmx: Resource.XHR_RESPONSE_TYPE.DOCUMENT, svg: Resource.XHR_RESPONSE_TYPE.DOCUMENT, // This was added to handle Tiled Tileset XML, but .tsx is also a TypeScript React Component. // Since it is way less likely for people to be loading TypeScript files instead of Tiled files, // this should probably be fine. tsx: Resource.XHR_RESPONSE_TYPE.DOCUMENT, // images gif: Resource.XHR_RESPONSE_TYPE.BLOB, png: Resource.XHR_RESPONSE_TYPE.BLOB, bmp: Resource.XHR_RESPONSE_TYPE.BLOB, jpg: Resource.XHR_RESPONSE_TYPE.BLOB, jpeg: Resource.XHR_RESPONSE_TYPE.BLOB, tif: Resource.XHR_RESPONSE_TYPE.BLOB, tiff: Resource.XHR_RESPONSE_TYPE.BLOB, webp: Resource.XHR_RESPONSE_TYPE.BLOB, tga: Resource.XHR_RESPONSE_TYPE.BLOB, // json json: Resource.XHR_RESPONSE_TYPE.JSON, // text text: Resource.XHR_RESPONSE_TYPE.TEXT, txt: Resource.XHR_RESPONSE_TYPE.TEXT, // fonts ttf: Resource.XHR_RESPONSE_TYPE.BUFFER, otf: Resource.XHR_RESPONSE_TYPE.BUFFER }; // We can't set the `src` attribute to empty string, so on abort we set it to this 1px transparent gif Resource.EMPTY_GIF = 'data:image/gif;base64,R0lGODlhAQABAIAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw=='; /** * Quick helper to set a value on one of the extension maps. Ensures there is no * dot at the start of the extension. * * @ignore * @param {object} map - The map to set on. * @param {string} extname - The extension (or key) to set. * @param {number} val - The value to set. */ function setExtMap(map, extname, val) { if (extname && extname.indexOf('.') === 0) { extname = extname.substring(1); } if (!extname) { return; } map[extname] = val; } /** * Quick helper to get string xhr type. * * @ignore * @param {XMLHttpRequest|XDomainRequest} xhr - The request to check. * @return {string} The type. */ function reqType(xhr) { return xhr.toString().replace('object ', ''); } },{"mini-signals":6,"parse-uri":7}],3:[function(require,module,exports){ 'use strict'; exports.__esModule = true; exports.eachSeries = eachSeries; exports.queue = queue; /** * Smaller version of the async library constructs. * */ function _noop() {} /* empty */ /** * Iterates an array in series. * * @param {Array.<*>} array - Array to iterate. * @param {function} iterator - Function to call for each element. * @param {function} callback - Function to call when done, or on error. * @param {boolean} [deferNext=false] - Break synchronous each loop by calling next with a setTimeout of 1. */ function eachSeries(array, iterator, callback, deferNext) { var i = 0; var len = array.length; (function next(err) { if (err || i === len) { if (callback) { callback(err); } return; } if (deferNext) { setTimeout(function () { iterator(array[i++], next); }, 1); } else { iterator(array[i++], next); } })(); } /** * Ensures a function is only called once. * * @param {function} fn - The function to wrap. * @return {function} The wrapping function. */ function onlyOnce(fn) { return function onceWrapper() { if (fn === null) { throw new Error('Callback was already called.'); } var callFn = fn; fn = null; callFn.apply(this, arguments); }; } /** * Async queue implementation, * * @param {function} worker - The worker function to call for each task. * @param {number} concurrency - How many workers to run in parrallel. * @return {*} The async queue object. */ function queue(worker, concurrency) { if (concurrency == null) { // eslint-disable-line no-eq-null,eqeqeq concurrency = 1; } else if (concurrency === 0) { throw new Error('Concurrency must not be zero'); } var workers = 0; var q = { _tasks: [], concurrency: concurrency, saturated: _noop, unsaturated: _noop, buffer: concurrency / 4, empty: _noop, drain: _noop, error: _noop, started: false, paused: false, push: function push(data, callback) { _insert(data, false, callback); }, kill: function kill() { workers = 0; q.drain = _noop; q.started = false; q._tasks = []; }, unshift: function unshift(data, callback) { _insert(data, true, callback); }, process: function process() { while (!q.paused && workers < q.concurrency && q._tasks.length) { var task = q._tasks.shift(); if (q._tasks.length === 0) { q.empty(); } workers += 1; if (workers === q.concurrency) { q.saturated(); } worker(task.data, onlyOnce(_next(task))); } }, length: function length() { return q._tasks.length; }, running: function running() { return workers; }, idle: function idle() { return q._tasks.length + workers === 0; }, pause: function pause() { if (q.paused === true) { return; } q.paused = true; }, resume: function resume() { if (q.paused === false) { return; } q.paused = false; // Need to call q.process once per concurrent // worker to preserve full concurrency after pause for (var w = 1; w <= q.concurrency; w++) { q.process(); } } }; function _insert(data, insertAtFront, callback) { if (callback != null && typeof callback !== 'function') { // eslint-disable-line no-eq-null,eqeqeq throw new Error('task callback must be a function'); } q.started = true; if (data == null && q.idle()) { // eslint-disable-line no-eq-null,eqeqeq // call drain immediately if there are no tasks setTimeout(function () { return q.drain(); }, 1); return; } var item = { data: data, callback: typeof callback === 'function' ? callback : _noop }; if (insertAtFront) { q._tasks.unshift(item); } else { q._tasks.push(item); } setTimeout(function () { return q.process(); }, 1); } function _next(task) { return function next() { workers -= 1; task.callback.apply(task, arguments); if (arguments[0] != null) { // eslint-disable-line no-eq-null,eqeqeq q.error(arguments[0], task.data); } if (workers <= q.concurrency - q.buffer) { q.unsaturated(); } if (q.idle()) { q.drain(); } q.process(); }; } return q; } },{}],4:[function(require,module,exports){ 'use strict'; exports.__esModule = true; exports.encodeBinary = encodeBinary; var _keyStr = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='; function encodeBinary(input) { var output = ''; var inx = 0; while (inx < input.length) { // Fill byte buffer array var bytebuffer = [0, 0, 0]; var encodedCharIndexes = [0, 0, 0, 0]; for (var jnx = 0; jnx < bytebuffer.length; ++jnx) { if (inx < input.length) { // throw away high-order byte, as documented at: // https://developer.mozilla.org/En/Using_XMLHttpRequest#Handling_binary_data bytebuffer[jnx] = input.charCodeAt(inx++) & 0xff; } else { bytebuffer[jnx] = 0; } } // Get each encoded character, 6 bits at a time // index 1: first 6 bits encodedCharIndexes[0] = bytebuffer[0] >> 2; // index 2: second 6 bits (2 least significant bits from input byte 1 + 4 most significant bits from byte 2) encodedCharIndexes[1] = (bytebuffer[0] & 0x3) << 4 | bytebuffer[1] >> 4; // index 3: third 6 bits (4 least significant bits from input byte 2 + 2 most significant bits from byte 3) encodedCharIndexes[2] = (bytebuffer[1] & 0x0f) << 2 | bytebuffer[2] >> 6; // index 3: forth 6 bits (6 least significant bits from input byte 3) encodedCharIndexes[3] = bytebuffer[2] & 0x3f; // Determine whether padding happened, and adjust accordingly var paddingBytes = inx - (input.length - 1); switch (paddingBytes) { case 2: // Set last 2 characters to padding char encodedCharIndexes[3] = 64; encodedCharIndexes[2] = 64; break; case 1: // Set last character to padding char encodedCharIndexes[3] = 64; break; default: break; // No padding - proceed } // Now we will grab each appropriate character out of our keystring // based on our index array and append it to the output string for (var _jnx = 0; _jnx < encodedCharIndexes.length; ++_jnx) { output += _keyStr.charAt(encodedCharIndexes[_jnx]); } } return output; } },{}],5:[function(require,module,exports){ 'use strict'; // import Loader from './Loader'; // import Resource from './Resource'; // import * as async from './async'; // import * as b64 from './b64'; /* eslint-disable no-undef */ var Loader = require('./Loader').default; var Resource = require('./Resource').default; var async = require('./async'); var b64 = require('./b64'); Loader.Resource = Resource; Loader.async = async; Loader.base64 = b64; // export manually, and also as default module.exports = Loader; // export default Loader; module.exports.default = Loader; },{"./Loader":1,"./Resource":2,"./async":3,"./b64":4}],6:[function(require,module,exports){ 'use strict'; Object.defineProperty(exports, '__esModule', { value: true }); var _createClass = (function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ('value' in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; })(); function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError('Cannot call a class as a function'); } } var MiniSignalBinding = (function () { function MiniSignalBinding(fn, once, thisArg) { if (once === undefined) once = false; _classCallCheck(this, MiniSignalBinding); this._fn = fn; this._once = once; this._thisArg = thisArg; this._next = this._prev = this._owner = null; } _createClass(MiniSignalBinding, [{ key: 'detach', value: function detach() { if (this._owner === null) return false; this._owner.detach(this); return true; } }]); return MiniSignalBinding; })(); function _addMiniSignalBinding(self, node) { if (!self._head) { self._head = node; self._tail = node; } else { self._tail._next = node; node._prev = self._tail; self._tail = node; } node._owner = self; return node; } var MiniSignal = (function () { function MiniSignal() { _classCallCheck(this, MiniSignal); this._head = this._tail = undefined; } _createClass(MiniSignal, [{ key: 'handlers', value: function handlers() { var exists = arguments.length <= 0 || arguments[0] === undefined ? false : arguments[0]; var node = this._head; if (exists) return !!node; var ee = []; while (node) { ee.push(node); node = node._next; } return ee; } }, { key: 'has', value: function has(node) { if (!(node instanceof MiniSignalBinding)) { throw new Error('MiniSignal#has(): First arg must be a MiniSignalBinding object.'); } return node._owner === this; } }, { key: 'dispatch', value: function dispatch() { var node = this._head; if (!node) return false; while (node) { if (node._once) this.detach(node); node._fn.apply(node._thisArg, arguments); node = node._next; } return true; } }, { key: 'add', value: function add(fn) { var thisArg = arguments.length <= 1 || arguments[1] === undefined ? null : arguments[1]; if (typeof fn !== 'function') { throw new Error('MiniSignal#add(): First arg must be a Function.'); } return _addMiniSignalBinding(this, new MiniSignalBinding(fn, false, thisArg)); } }, { key: 'once', value: function once(fn) { var thisArg = arguments.length <= 1 || arguments[1] === undefined ? null : arguments[1]; if (typeof fn !== 'function') { throw new Error('MiniSignal#once(): First arg must be a Function.'); } return _addMiniSignalBinding(this, new MiniSignalBinding(fn, true, thisArg)); } }, { key: 'detach', value: function detach(node) { if (!(node instanceof MiniSignalBinding)) { throw new Error('MiniSignal#detach(): First arg must be a MiniSignalBinding object.'); } if (node._owner !== this) return this; if (node._prev) node._prev._next = node._next; if (node._next) node._next._prev = node._prev; if (node === this._head) { this._head = node._next; if (node._next === null) { this._tail = null; } } else if (node === this._tail) { this._tail = node._prev; this._tail._next = null; } node._owner = null; return this; } }, { key: 'detachAll', value: function detachAll() { var node = this._head; if (!node) return this; this._head = this._tail = null; while (node) { node._owner = null; node = node._next; } return this; } }]); return MiniSignal; })(); MiniSignal.MiniSignalBinding = MiniSignalBinding; exports['default'] = MiniSignal; module.exports = exports['default']; },{}],7:[function(require,module,exports){ 'use strict' module.exports = function parseURI (str, opts) { opts = opts || {} var o = { key: ['source', 'protocol', 'authority', 'userInfo', 'user', 'password', 'host', 'port', 'relative', 'path', 'directory', 'file', 'query', 'anchor'], q: { name: 'queryKey', parser: /(?:^|&)([^&=]*)=?([^&]*)/g }, parser: { strict: /^(?:([^:\/?#]+):)?(?:\/\/((?:(([^:@]*)(?::([^:@]*))?)?@)?([^:\/?#]*)(?::(\d*))?))?((((?:[^?#\/]*\/)*)([^?#]*))(?:\?([^#]*))?(?:#(.*))?)/, loose: /^(?:(?![^:@]+:[^:@\/]*@)([^:\/?#.]+):)?(?:\/\/)?((?:(([^:@]*)(?::([^:@]*))?)?@)?([^:\/?#]*)(?::(\d*))?)(((\/(?:[^?#](?![^?#\/]*\.[^?#\/.]+(?:[?#]|$)))*\/?)?([^?#\/]*))(?:\?([^#]*))?(?:#(.*))?)/ } } var m = o.parser[opts.strictMode ? 'strict' : 'loose'].exec(str) var uri = {} var i = 14 while (i--) uri[o.key[i]] = m[i] || '' uri[o.q.name] = {} uri[o.key[12]].replace(o.q.parser, function ($0, $1, $2) { if ($1) uri[o.q.name][$1] = $2 }) return uri } },{}]},{},[5])(5) }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","lib/Loader.js","lib/Resource.js","lib/async.js","lib/b64.js","lib/index.js","node_modules/mini-signals/lib/mini-signals.js","node_modules/parse-uri/index.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3oBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACloCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\nexports.__esModule = true;\n\nvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _miniSignals = require('mini-signals');\n\nvar _miniSignals2 = _interopRequireDefault(_miniSignals);\n\nvar _parseUri = require('parse-uri');\n\nvar _parseUri2 = _interopRequireDefault(_parseUri);\n\nvar _async = require('./async');\n\nvar async = _interopRequireWildcard(_async);\n\nvar _Resource = require('./Resource');\n\nvar _Resource2 = _interopRequireDefault(_Resource);\n\nfunction _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key]; } } newObj.default = obj; return newObj; } }\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\n// some constants\nvar MAX_PROGRESS = 100;\nvar rgxExtractUrlHash = /(#[\\w-]+)?$/;\n\n/**\n * Manages the state and loading of multiple resources to load.\n *\n * @class\n */\n\nvar Loader = function () {\n    /**\n     * @param {string} [baseUrl=''] - The base url for all resources loaded by this loader.\n     * @param {number} [concurrency=10] - The number of resources to load concurrently.\n     */\n    function Loader() {\n        var _this = this;\n\n        var baseUrl = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : '';\n        var concurrency = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 10;\n\n        _classCallCheck(this, Loader);\n\n        /**\n         * The base url for all resources loaded by this loader.\n         *\n         * @member {string}\n         */\n        this.baseUrl = baseUrl;\n\n        /**\n         * The progress percent of the loader going through the queue.\n         *\n         * @member {number}\n         */\n        this.progress = 0;\n\n        /**\n         * Loading state of the loader, true if it is currently loading resources.\n         *\n         * @member {boolean}\n         */\n        this.loading = false;\n\n        /**\n         * A querystring to append to every URL added to the loader.\n         *\n         * This should be a valid query string *without* the question-mark (`?`). The loader will\n         * also *not* escape values for you. Make sure to escape your parameters with\n         * [`encodeURIComponent`](https://mdn.io/encodeURIComponent) before assigning this property.\n         *\n         * @example\n         * const loader = new Loader();\n         *\n         * loader.defaultQueryString = 'user=me&password=secret';\n         *\n         * // This will request 'image.png?user=me&password=secret'\n         * loader.add('image.png').load();\n         *\n         * loader.reset();\n         *\n         * // This will request 'image.png?v=1&user=me&password=secret'\n         * loader.add('iamge.png?v=1').load();\n         */\n        this.defaultQueryString = '';\n\n        /**\n         * The middleware to run before loading each resource.\n         *\n         * @member {function[]}\n         */\n        this._beforeMiddleware = [];\n\n        /**\n         * The middleware to run after loading each resource.\n         *\n         * @member {function[]}\n         */\n        this._afterMiddleware = [];\n\n        /**\n         * The tracks the resources we are currently completing parsing for.\n         *\n         * @member {Resource[]}\n         */\n        this._resourcesParsing = [];\n\n        /**\n         * The `_loadResource` function bound with this object context.\n         *\n         * @private\n         * @member {function}\n         * @param {Resource} r - The resource to load\n         * @param {Function} d - The dequeue function\n         * @return {undefined}\n         */\n        this._boundLoadResource = function (r, d) {\n            return _this._loadResource(r, d);\n        };\n\n        /**\n         * The resources waiting to be loaded.\n         *\n         * @private\n         * @member {Resource[]}\n         */\n        this._queue = async.queue(this._boundLoadResource, concurrency);\n\n        this._queue.pause();\n\n        /**\n         * All the resources for this loader keyed by name.\n         *\n         * @member {object<string, Resource>}\n         */\n        this.resources = {};\n\n        /**\n         * Dispatched once per loaded or errored resource.\n         *\n         * The callback looks like {@link Loader.OnProgressSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onProgress = new _miniSignals2.default();\n\n        /**\n         * Dispatched once per errored resource.\n         *\n         * The callback looks like {@link Loader.OnErrorSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onError = new _miniSignals2.default();\n\n        /**\n         * Dispatched once per loaded resource.\n         *\n         * The callback looks like {@link Loader.OnLoadSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onLoad = new _miniSignals2.default();\n\n        /**\n         * Dispatched when the loader begins to process the queue.\n         *\n         * The callback looks like {@link Loader.OnStartSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onStart = new _miniSignals2.default();\n\n        /**\n         * Dispatched when the queued resources all load.\n         *\n         * The callback looks like {@link Loader.OnCompleteSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onComplete = new _miniSignals2.default();\n\n        /**\n         * When the progress changes the loader and resource are disaptched.\n         *\n         * @memberof Loader\n         * @callback OnProgressSignal\n         * @param {Loader} loader - The loader the progress is advancing on.\n         * @param {Resource} resource - The resource that has completed or failed to cause the progress to advance.\n         */\n\n        /**\n         * When an error occurrs the loader and resource are disaptched.\n         *\n         * @memberof Loader\n         * @callback OnErrorSignal\n         * @param {Loader} loader - The loader the error happened in.\n         * @param {Resource} resource - The resource that caused the error.\n         */\n\n        /**\n         * When a load completes the loader and resource are disaptched.\n         *\n         * @memberof Loader\n         * @callback OnLoadSignal\n         * @param {Loader} loader - The loader that laoded the resource.\n         * @param {Resource} resource - The resource that has completed loading.\n         */\n\n        /**\n         * When the loader starts loading resources it dispatches this callback.\n         *\n         * @memberof Loader\n         * @callback OnStartSignal\n         * @param {Loader} loader - The loader that has started loading resources.\n         */\n\n        /**\n         * When the loader completes loading resources it dispatches this callback.\n         *\n         * @memberof Loader\n         * @callback OnCompleteSignal\n         * @param {Loader} loader - The loader that has finished loading resources.\n         */\n    }\n\n    /**\n     * Adds a resource (or multiple resources) to the loader queue.\n     *\n     * This function can take a wide variety of different parameters. The only thing that is always\n     * required the url to load. All the following will work:\n     *\n     * ```js\n     * loader\n     *     // normal param syntax\n     *     .add('key', 'http://...', function () {})\n     *     .add('http://...', function () {})\n     *     .add('http://...')\n     *\n     *     // object syntax\n     *     .add({\n     *         name: 'key2',\n     *         url: 'http://...'\n     *     }, function () {})\n     *     .add({\n     *         url: 'http://...'\n     *     }, function () {})\n     *     .add({\n     *         name: 'key3',\n     *         url: 'http://...'\n     *         onComplete: function () {}\n     *     })\n     *     .add({\n     *         url: 'https://...',\n     *         onComplete: function () {},\n     *         crossOrigin: true\n     *     })\n     *\n     *     // you can also pass an array of objects or urls or both\n     *     .add([\n     *         { name: 'key4', url: 'http://...', onComplete: function () {} },\n     *         { url: 'http://...', onComplete: function () {} },\n     *         'http://...'\n     *     ])\n     *\n     *     // and you can use both params and options\n     *     .add('key', 'http://...', { crossOrigin: true }, function () {})\n     *     .add('http://...', { crossOrigin: true }, function () {});\n     * ```\n     *\n     * @param {string} [name] - The name of the resource to load, if not passed the url is used.\n     * @param {string} [url] - The url for this resource, relative to the baseUrl of this loader.\n     * @param {object} [options] - The options for the load.\n     * @param {boolean} [options.crossOrigin] - Is this request cross-origin? Default is to determine automatically.\n     * @param {Resource.LOAD_TYPE} [options.loadType=Resource.LOAD_TYPE.XHR] - How should this resource be loaded?\n     * @param {Resource.XHR_RESPONSE_TYPE} [options.xhrType=Resource.XHR_RESPONSE_TYPE.DEFAULT] - How should\n     *      the data being loaded be interpreted when using XHR?\n     * @param {object} [options.metadata] - Extra configuration for middleware and the Resource object.\n     * @param {HTMLImageElement|HTMLAudioElement|HTMLVideoElement} [options.metadata.loadElement=null] - The\n     *      element to use for loading, instead of creating one.\n     * @param {boolean} [options.metadata.skipSource=false] - Skips adding source(s) to the load element. This\n     *      is useful if you want to pass in a `loadElement` that you already added load sources to.\n     * @param {function} [cb] - Function to call when this specific resource completes loading.\n     * @return {Loader} Returns itself.\n     */\n\n\n    Loader.prototype.add = function add(name, url, options, cb) {\n        // special case of an array of objects or urls\n        if (Array.isArray(name)) {\n            for (var i = 0; i < name.length; ++i) {\n                this.add(name[i]);\n            }\n\n            return this;\n        }\n\n        // if an object is passed instead of params\n        if ((typeof name === 'undefined' ? 'undefined' : _typeof(name)) === 'object') {\n            cb = url || name.callback || name.onComplete;\n            options = name;\n            url = name.url;\n            name = name.name || name.key || name.url;\n        }\n\n        // case where no name is passed shift all args over by one.\n        if (typeof url !== 'string') {\n            cb = options;\n            options = url;\n            url = name;\n        }\n\n        // now that we shifted make sure we have a proper url.\n        if (typeof url !== 'string') {\n            throw new Error('No url passed to add resource to loader.');\n        }\n\n        // options are optional so people might pass a function and no options\n        if (typeof options === 'function') {\n            cb = options;\n            options = null;\n        }\n\n        // if loading already you can only add resources that have a parent.\n        if (this.loading && (!options || !options.parentResource)) {\n            throw new Error('Cannot add resources while the loader is running.');\n        }\n\n        // check if resource already exists.\n        if (this.resources[name]) {\n            throw new Error('Resource named \"' + name + '\" already exists.');\n        }\n\n        // add base url if this isn't an absolute url\n        url = this._prepareUrl(url);\n\n        // create the store the resource\n        this.resources[name] = new _Resource2.default(name, url, options);\n\n        if (typeof cb === 'function') {\n            this.resources[name].onAfterMiddleware.once(cb);\n        }\n\n        // if actively loading, make sure to adjust progress chunks for that parent and its children\n        if (this.loading) {\n            var parent = options.parentResource;\n            var incompleteChildren = [];\n\n            for (var _i = 0; _i < parent.children.length; ++_i) {\n                if (!parent.children[_i].isComplete) {\n                    incompleteChildren.push(parent.children[_i]);\n                }\n            }\n\n            var fullChunk = parent.progressChunk * (incompleteChildren.length + 1); // +1 for parent\n            var eachChunk = fullChunk / (incompleteChildren.length + 2); // +2 for parent & new child\n\n            parent.children.push(this.resources[name]);\n            parent.progressChunk = eachChunk;\n\n            for (var _i2 = 0; _i2 < incompleteChildren.length; ++_i2) {\n                incompleteChildren[_i2].progressChunk = eachChunk;\n            }\n\n            this.resources[name].progressChunk = eachChunk;\n        }\n\n        // add the resource to the queue\n        this._queue.push(this.resources[name]);\n\n        return this;\n    };\n\n    /**\n     * Sets up a middleware function that will run *before* the\n     * resource is loaded.\n     *\n     * @method before\n     * @param {function} fn - The middleware function to register.\n     * @return {Loader} Returns itself.\n     */\n\n\n    Loader.prototype.pre = function pre(fn) {\n        this._beforeMiddleware.push(fn);\n\n        return this;\n    };\n\n    /**\n     * Sets up a middleware function that will run *after* the\n     * resource is loaded.\n     *\n     * @alias use\n     * @method after\n     * @param {function} fn - The middleware function to register.\n     * @return {Loader} Returns itself.\n     */\n\n\n    Loader.prototype.use = function use(fn) {\n        this._afterMiddleware.push(fn);\n\n        return this;\n    };\n\n    /**\n     * Resets the queue of the loader to prepare for a new load.\n     *\n     * @return {Loader} Returns itself.\n     */\n\n\n    Loader.prototype.reset = function reset() {\n        this.progress = 0;\n        this.loading = false;\n\n        this._queue.kill();\n        this._queue.pause();\n\n        // abort all resource loads\n        for (var k in this.resources) {\n            var res = this.resources[k];\n\n            if (res._onLoadBinding) {\n                res._onLoadBinding.detach();\n            }\n\n            if (res.isLoading) {\n                res.abort();\n            }\n        }\n\n        this.resources = {};\n\n        return this;\n    };\n\n    /**\n     * Starts loading the queued resources.\n     *\n     * @param {function} [cb] - Optional callback that will be bound to the `complete` event.\n     * @return {Loader} Returns itself.\n     */\n\n\n    Loader.prototype.load = function load(cb) {\n        // register complete callback if they pass one\n        if (typeof cb === 'function') {\n            this.onComplete.once(cb);\n        }\n\n        // if the queue has already started we are done here\n        if (this.loading) {\n            return this;\n        }\n\n        if (this._queue.idle()) {\n            this._onStart();\n            this._onComplete();\n        } else {\n            // distribute progress chunks\n            var numTasks = this._queue._tasks.length;\n            var chunk = 100 / numTasks;\n\n            for (var i = 0; i < this._queue._tasks.length; ++i) {\n                this._queue._tasks[i].data.progressChunk = chunk;\n            }\n\n            // notify we are starting\n            this._onStart();\n\n            // start loading\n            this._queue.resume();\n        }\n\n        return this;\n    };\n\n    /**\n     * The number of resources to load concurrently.\n     *\n     * @member {number}\n     * @default 10\n     */\n\n\n    /**\n     * Prepares a url for usage based on the configuration of this object\n     *\n     * @private\n     * @param {string} url - The url to prepare.\n     * @return {string} The prepared url.\n     */\n    Loader.prototype._prepareUrl = function _prepareUrl(url) {\n        var parsedUrl = (0, _parseUri2.default)(url, { strictMode: true });\n        var result = void 0;\n\n        // absolute url, just use it as is.\n        if (parsedUrl.protocol || !parsedUrl.path || url.indexOf('//') === 0) {\n            result = url;\n        }\n        // if baseUrl doesn't end in slash and url doesn't start with slash, then add a slash inbetween\n        else if (this.baseUrl.length && this.baseUrl.lastIndexOf('/') !== this.baseUrl.length - 1 && url.charAt(0) !== '/') {\n                result = this.baseUrl + '/' + url;\n            } else {\n                result = this.baseUrl + url;\n            }\n\n        // if we need to add a default querystring, there is a bit more work\n        if (this.defaultQueryString) {\n            var hash = rgxExtractUrlHash.exec(result)[0];\n\n            result = result.substr(0, result.length - hash.length);\n\n            if (result.indexOf('?') !== -1) {\n                result += '&' + this.defaultQueryString;\n            } else {\n                result += '?' + this.defaultQueryString;\n            }\n\n            result += hash;\n        }\n\n        return result;\n    };\n\n    /**\n     * Loads a single resource.\n     *\n     * @private\n     * @param {Resource} resource - The resource to load.\n     * @param {function} dequeue - The function to call when we need to dequeue this item.\n     */\n\n\n    Loader.prototype._loadResource = function _loadResource(resource, dequeue) {\n        var _this2 = this;\n\n        resource._dequeue = dequeue;\n\n        // run before middleware\n        async.eachSeries(this._beforeMiddleware, function (fn, next) {\n            fn.call(_this2, resource, function () {\n                // if the before middleware marks the resource as complete,\n                // break and don't process any more before middleware\n                next(resource.isComplete ? {} : null);\n            });\n        }, function () {\n            if (resource.isComplete) {\n                _this2._onLoad(resource);\n            } else {\n                resource._onLoadBinding = resource.onComplete.once(_this2._onLoad, _this2);\n                resource.load();\n            }\n        }, true);\n    };\n\n    /**\n     * Called once loading has started.\n     *\n     * @private\n     */\n\n\n    Loader.prototype._onStart = function _onStart() {\n        this.progress = 0;\n        this.loading = true;\n        this.onStart.dispatch(this);\n    };\n\n    /**\n     * Called once each resource has loaded.\n     *\n     * @private\n     */\n\n\n    Loader.prototype._onComplete = function _onComplete() {\n        this.progress = MAX_PROGRESS;\n        this.loading = false;\n        this.onComplete.dispatch(this, this.resources);\n    };\n\n    /**\n     * Called each time a resources is loaded.\n     *\n     * @private\n     * @param {Resource} resource - The resource that was loaded\n     */\n\n\n    Loader.prototype._onLoad = function _onLoad(resource) {\n        var _this3 = this;\n\n        resource._onLoadBinding = null;\n\n        // remove this resource from the async queue, and add it to our list of resources that are being parsed\n        this._resourcesParsing.push(resource);\n        resource._dequeue();\n\n        // run all the after middleware for this resource\n        async.eachSeries(this._afterMiddleware, function (fn, next) {\n            fn.call(_this3, resource, next);\n        }, function () {\n            resource.onAfterMiddleware.dispatch(resource);\n\n            _this3.progress += resource.progressChunk;\n            _this3.onProgress.dispatch(_this3, resource);\n\n            if (resource.error) {\n                _this3.onError.dispatch(resource.error, _this3, resource);\n            } else {\n                _this3.onLoad.dispatch(_this3, resource);\n            }\n\n            _this3._resourcesParsing.splice(_this3._resourcesParsing.indexOf(resource), 1);\n\n            // do completion check\n            if (_this3._queue.idle() && _this3._resourcesParsing.length === 0) {\n                _this3._onComplete();\n            }\n        }, true);\n    };\n\n    _createClass(Loader, [{\n        key: 'concurrency',\n        get: function get() {\n            return this._queue.concurrency;\n        }\n        // eslint-disable-next-line require-jsdoc\n        ,\n        set: function set(concurrency) {\n            this._queue.concurrency = concurrency;\n        }\n    }]);\n\n    return Loader;\n}();\n\nexports.default = Loader;\n//# sourceMappingURL=Loader.js.map","'use strict';\n\nexports.__esModule = true;\n\nvar _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();\n\nvar _parseUri = require('parse-uri');\n\nvar _parseUri2 = _interopRequireDefault(_parseUri);\n\nvar _miniSignals = require('mini-signals');\n\nvar _miniSignals2 = _interopRequireDefault(_miniSignals);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\n// tests is CORS is supported in XHR, if not we need to use XDR\nvar useXdr = !!(window.XDomainRequest && !('withCredentials' in new XMLHttpRequest()));\nvar tempAnchor = null;\n\n// some status constants\nvar STATUS_NONE = 0;\nvar STATUS_OK = 200;\nvar STATUS_EMPTY = 204;\nvar STATUS_IE_BUG_EMPTY = 1223;\nvar STATUS_TYPE_OK = 2;\n\n// noop\nfunction _noop() {} /* empty */\n\n/**\n * Manages the state and loading of a resource and all child resources.\n *\n * @class\n */\n\nvar Resource = function () {\n    /**\n     * Sets the load type to be used for a specific extension.\n     *\n     * @static\n     * @param {string} extname - The extension to set the type for, e.g. \"png\" or \"fnt\"\n     * @param {Resource.LOAD_TYPE} loadType - The load type to set it to.\n     */\n    Resource.setExtensionLoadType = function setExtensionLoadType(extname, loadType) {\n        setExtMap(Resource._loadTypeMap, extname, loadType);\n    };\n\n    /**\n     * Sets the load type to be used for a specific extension.\n     *\n     * @static\n     * @param {string} extname - The extension to set the type for, e.g. \"png\" or \"fnt\"\n     * @param {Resource.XHR_RESPONSE_TYPE} xhrType - The xhr type to set it to.\n     */\n\n\n    Resource.setExtensionXhrType = function setExtensionXhrType(extname, xhrType) {\n        setExtMap(Resource._xhrTypeMap, extname, xhrType);\n    };\n\n    /**\n     * @param {string} name - The name of the resource to load.\n     * @param {string|string[]} url - The url for this resource, for audio/video loads you can pass\n     *      an array of sources.\n     * @param {object} [options] - The options for the load.\n     * @param {string|boolean} [options.crossOrigin] - Is this request cross-origin? Default is to\n     *      determine automatically.\n     * @param {Resource.LOAD_TYPE} [options.loadType=Resource.LOAD_TYPE.XHR] - How should this resource\n     *      be loaded?\n     * @param {Resource.XHR_RESPONSE_TYPE} [options.xhrType=Resource.XHR_RESPONSE_TYPE.DEFAULT] - How\n     *      should the data being loaded be interpreted when using XHR?\n     * @param {object} [options.metadata] - Extra configuration for middleware and the Resource object.\n     * @param {HTMLImageElement|HTMLAudioElement|HTMLVideoElement} [options.metadata.loadElement=null] - The\n     *      element to use for loading, instead of creating one.\n     * @param {boolean} [options.metadata.skipSource=false] - Skips adding source(s) to the load element. This\n     *      is useful if you want to pass in a `loadElement` that you already added load sources to.\n     * @param {string|string[]} [options.metadata.mimeType] - The mime type to use for the source element of a video/audio\n     *      elment. If the urls are an array, you can pass this as an array as well where each index is the mime type to\n     *      use for the corresponding url index.\n     */\n\n\n    function Resource(name, url, options) {\n        _classCallCheck(this, Resource);\n\n        if (typeof name !== 'string' || typeof url !== 'string') {\n            throw new Error('Both name and url are required for constructing a resource.');\n        }\n\n        options = options || {};\n\n        /**\n         * The state flags of this resource.\n         *\n         * @member {number}\n         */\n        this._flags = 0;\n\n        // set data url flag, needs to be set early for some _determineX checks to work.\n        this._setFlag(Resource.STATUS_FLAGS.DATA_URL, url.indexOf('data:') === 0);\n\n        /**\n         * The name of this resource.\n         *\n         * @member {string}\n         * @readonly\n         */\n        this.name = name;\n\n        /**\n         * The url used to load this resource.\n         *\n         * @member {string}\n         * @readonly\n         */\n        this.url = url;\n\n        /**\n         * The extension used to load this resource.\n         *\n         * @member {string}\n         * @readonly\n         */\n        this.extension = this._getExtension();\n\n        /**\n         * The data that was loaded by the resource.\n         *\n         * @member {any}\n         */\n        this.data = null;\n\n        /**\n         * Is this request cross-origin? If unset, determined automatically.\n         *\n         * @member {string}\n         */\n        this.crossOrigin = options.crossOrigin === true ? 'anonymous' : options.crossOrigin;\n\n        /**\n         * The method of loading to use for this resource.\n         *\n         * @member {Resource.LOAD_TYPE}\n         */\n        this.loadType = options.loadType || this._determineLoadType();\n\n        /**\n         * The type used to load the resource via XHR. If unset, determined automatically.\n         *\n         * @member {string}\n         */\n        this.xhrType = options.xhrType;\n\n        /**\n         * Extra info for middleware, and controlling specifics about how the resource loads.\n         *\n         * Note that if you pass in a `loadElement`, the Resource class takes ownership of it.\n         * Meaning it will modify it as it sees fit.\n         *\n         * @member {object}\n         * @property {HTMLImageElement|HTMLAudioElement|HTMLVideoElement} [loadElement=null] - The\n         *  element to use for loading, instead of creating one.\n         * @property {boolean} [skipSource=false] - Skips adding source(s) to the load element. This\n         *  is useful if you want to pass in a `loadElement` that you already added load sources\n         *  to.\n         */\n        this.metadata = options.metadata || {};\n\n        /**\n         * The error that occurred while loading (if any).\n         *\n         * @member {Error}\n         * @readonly\n         */\n        this.error = null;\n\n        /**\n         * The XHR object that was used to load this resource. This is only set\n         * when `loadType` is `Resource.LOAD_TYPE.XHR`.\n         *\n         * @member {XMLHttpRequest}\n         * @readonly\n         */\n        this.xhr = null;\n\n        /**\n         * The child resources this resource owns.\n         *\n         * @member {Resource[]}\n         * @readonly\n         */\n        this.children = [];\n\n        /**\n         * The resource type.\n         *\n         * @member {Resource.TYPE}\n         * @readonly\n         */\n        this.type = Resource.TYPE.UNKNOWN;\n\n        /**\n         * The progress chunk owned by this resource.\n         *\n         * @member {number}\n         * @readonly\n         */\n        this.progressChunk = 0;\n\n        /**\n         * The `dequeue` method that will be used a storage place for the async queue dequeue method\n         * used privately by the loader.\n         *\n         * @private\n         * @member {function}\n         */\n        this._dequeue = _noop;\n\n        /**\n         * Used a storage place for the on load binding used privately by the loader.\n         *\n         * @private\n         * @member {function}\n         */\n        this._onLoadBinding = null;\n\n        /**\n         * The `complete` function bound to this resource's context.\n         *\n         * @private\n         * @member {function}\n         */\n        this._boundComplete = this.complete.bind(this);\n\n        /**\n         * The `_onError` function bound to this resource's context.\n         *\n         * @private\n         * @member {function}\n         */\n        this._boundOnError = this._onError.bind(this);\n\n        /**\n         * The `_onProgress` function bound to this resource's context.\n         *\n         * @private\n         * @member {function}\n         */\n        this._boundOnProgress = this._onProgress.bind(this);\n\n        // xhr callbacks\n        this._boundXhrOnError = this._xhrOnError.bind(this);\n        this._boundXhrOnAbort = this._xhrOnAbort.bind(this);\n        this._boundXhrOnLoad = this._xhrOnLoad.bind(this);\n        this._boundXdrOnTimeout = this._xdrOnTimeout.bind(this);\n\n        /**\n         * Dispatched when the resource beings to load.\n         *\n         * The callback looks like {@link Resource.OnStartSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onStart = new _miniSignals2.default();\n\n        /**\n         * Dispatched each time progress of this resource load updates.\n         * Not all resources types and loader systems can support this event\n         * so sometimes it may not be available. If the resource\n         * is being loaded on a modern browser, using XHR, and the remote server\n         * properly sets Content-Length headers, then this will be available.\n         *\n         * The callback looks like {@link Resource.OnProgressSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onProgress = new _miniSignals2.default();\n\n        /**\n         * Dispatched once this resource has loaded, if there was an error it will\n         * be in the `error` property.\n         *\n         * The callback looks like {@link Resource.OnCompleteSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onComplete = new _miniSignals2.default();\n\n        /**\n         * Dispatched after this resource has had all the *after* middleware run on it.\n         *\n         * The callback looks like {@link Resource.OnCompleteSignal}.\n         *\n         * @member {Signal}\n         */\n        this.onAfterMiddleware = new _miniSignals2.default();\n\n        /**\n         * When the resource starts to load.\n         *\n         * @memberof Resource\n         * @callback OnStartSignal\n         * @param {Resource} resource - The resource that the event happened on.\n         */\n\n        /**\n         * When the resource reports loading progress.\n         *\n         * @memberof Resource\n         * @callback OnProgressSignal\n         * @param {Resource} resource - The resource that the event happened on.\n         * @param {number} percentage - The progress of the load in the range [0, 1].\n         */\n\n        /**\n         * When the resource finishes loading.\n         *\n         * @memberof Resource\n         * @callback OnCompleteSignal\n         * @param {Resource} resource - The resource that the event happened on.\n         */\n    }\n\n    /**\n     * Stores whether or not this url is a data url.\n     *\n     * @member {boolean}\n     * @readonly\n     */\n\n\n    /**\n     * Marks the resource as complete.\n     *\n     */\n    Resource.prototype.complete = function complete() {\n        // TODO: Clean this up in a wrapper or something...gross....\n        if (this.data && this.data.removeEventListener) {\n            this.data.removeEventListener('error', this._boundOnError, false);\n            this.data.removeEventListener('load', this._boundComplete, false);\n            this.data.removeEventListener('progress', this._boundOnProgress, false);\n            this.data.removeEventListener('canplaythrough', this._boundComplete, false);\n        }\n\n        if (this.xhr) {\n            if (this.xhr.removeEventListener) {\n                this.xhr.removeEventListener('error', this._boundXhrOnError, false);\n                this.xhr.removeEventListener('abort', this._boundXhrOnAbort, false);\n                this.xhr.removeEventListener('progress', this._boundOnProgress, false);\n                this.xhr.removeEventListener('load', this._boundXhrOnLoad, false);\n            } else {\n                this.xhr.onerror = null;\n                this.xhr.ontimeout = null;\n                this.xhr.onprogress = null;\n                this.xhr.onload = null;\n            }\n        }\n\n        if (this.isComplete) {\n            throw new Error('Complete called again for an already completed resource.');\n        }\n\n        this._setFlag(Resource.STATUS_FLAGS.COMPLETE, true);\n        this._setFlag(Resource.STATUS_FLAGS.LOADING, false);\n\n        this.onComplete.dispatch(this);\n    };\n\n    /**\n     * Aborts the loading of this resource, with an optional message.\n     *\n     * @param {string} message - The message to use for the error\n     */\n\n\n    Resource.prototype.abort = function abort(message) {\n        // abort can be called multiple times, ignore subsequent calls.\n        if (this.error) {\n            return;\n        }\n\n        // store error\n        this.error = new Error(message);\n\n        // abort the actual loading\n        if (this.xhr) {\n            this.xhr.abort();\n        } else if (this.xdr) {\n            this.xdr.abort();\n        } else if (this.data) {\n            // single source\n            if (this.data.src) {\n                this.data.src = Resource.EMPTY_GIF;\n            }\n            // multi-source\n            else {\n                    while (this.data.firstChild) {\n                        this.data.removeChild(this.data.firstChild);\n                    }\n                }\n        }\n\n        // done now.\n        this.complete();\n    };\n\n    /**\n     * Kicks off loading of this resource. This method is asynchronous.\n     *\n     * @param {function} [cb] - Optional callback to call once the resource is loaded.\n     */\n\n\n    Resource.prototype.load = function load(cb) {\n        var _this = this;\n\n        if (this.isLoading) {\n            return;\n        }\n\n        if (this.isComplete) {\n            if (cb) {\n                setTimeout(function () {\n                    return cb(_this);\n                }, 1);\n            }\n\n            return;\n        } else if (cb) {\n            this.onComplete.once(cb);\n        }\n\n        this._setFlag(Resource.STATUS_FLAGS.LOADING, true);\n\n        this.onStart.dispatch(this);\n\n        // if unset, determine the value\n        if (this.crossOrigin === false || typeof this.crossOrigin !== 'string') {\n            this.crossOrigin = this._determineCrossOrigin(this.url);\n        }\n\n        switch (this.loadType) {\n            case Resource.LOAD_TYPE.IMAGE:\n                this.type = Resource.TYPE.IMAGE;\n                this._loadElement('image');\n                break;\n\n            case Resource.LOAD_TYPE.AUDIO:\n                this.type = Resource.TYPE.AUDIO;\n                this._loadSourceElement('audio');\n                break;\n\n            case Resource.LOAD_TYPE.VIDEO:\n                this.type = Resource.TYPE.VIDEO;\n                this._loadSourceElement('video');\n                break;\n\n            case Resource.LOAD_TYPE.XHR:\n            /* falls through */\n            default:\n                if (useXdr && this.crossOrigin) {\n                    this._loadXdr();\n                } else {\n                    this._loadXhr();\n                }\n                break;\n        }\n    };\n\n    /**\n     * Checks if the flag is set.\n     *\n     * @private\n     * @param {number} flag - The flag to check.\n     * @return {boolean} True if the flag is set.\n     */\n\n\n    Resource.prototype._hasFlag = function _hasFlag(flag) {\n        return !!(this._flags & flag);\n    };\n\n    /**\n     * (Un)Sets the flag.\n     *\n     * @private\n     * @param {number} flag - The flag to (un)set.\n     * @param {boolean} value - Whether to set or (un)set the flag.\n     */\n\n\n    Resource.prototype._setFlag = function _setFlag(flag, value) {\n        this._flags = value ? this._flags | flag : this._flags & ~flag;\n    };\n\n    /**\n     * Loads this resources using an element that has a single source,\n     * like an HTMLImageElement.\n     *\n     * @private\n     * @param {string} type - The type of element to use.\n     */\n\n\n    Resource.prototype._loadElement = function _loadElement(type) {\n        if (this.metadata.loadElement) {\n            this.data = this.metadata.loadElement;\n        } else if (type === 'image' && typeof window.Image !== 'undefined') {\n            this.data = new Image();\n        } else {\n            this.data = document.createElement(type);\n        }\n\n        if (this.crossOrigin) {\n            this.data.crossOrigin = this.crossOrigin;\n        }\n\n        if (!this.metadata.skipSource) {\n            this.data.src = this.url;\n        }\n\n        this.data.addEventListener('error', this._boundOnError, false);\n        this.data.addEventListener('load', this._boundComplete, false);\n        this.data.addEventListener('progress', this._boundOnProgress, false);\n    };\n\n    /**\n     * Loads this resources using an element that has multiple sources,\n     * like an HTMLAudioElement or HTMLVideoElement.\n     *\n     * @private\n     * @param {string} type - The type of element to use.\n     */\n\n\n    Resource.prototype._loadSourceElement = function _loadSourceElement(type) {\n        if (this.metadata.loadElement) {\n            this.data = this.metadata.loadElement;\n        } else if (type === 'audio' && typeof window.Audio !== 'undefined') {\n            this.data = new Audio();\n        } else {\n            this.data = document.createElement(type);\n        }\n\n        if (this.data === null) {\n            this.abort('Unsupported element: ' + type);\n\n            return;\n        }\n\n        if (!this.metadata.skipSource) {\n            // support for CocoonJS Canvas+ runtime, lacks document.createElement('source')\n            if (navigator.isCocoonJS) {\n                this.data.src = Array.isArray(this.url) ? this.url[0] : this.url;\n            } else if (Array.isArray(this.url)) {\n                var mimeTypes = this.metadata.mimeType;\n\n                for (var i = 0; i < this.url.length; ++i) {\n                    this.data.appendChild(this._createSource(type, this.url[i], Array.isArray(mimeTypes) ? mimeTypes[i] : mimeTypes));\n                }\n            } else {\n                var _mimeTypes = this.metadata.mimeType;\n\n                this.data.appendChild(this._createSource(type, this.url, Array.isArray(_mimeTypes) ? _mimeTypes[0] : _mimeTypes));\n            }\n        }\n\n        this.data.addEventListener('error', this._boundOnError, false);\n        this.data.addEventListener('load', this._boundComplete, false);\n        this.data.addEventListener('progress', this._boundOnProgress, false);\n        this.data.addEventListener('canplaythrough', this._boundComplete, false);\n\n        this.data.load();\n    };\n\n    /**\n     * Loads this resources using an XMLHttpRequest.\n     *\n     * @private\n     */\n\n\n    Resource.prototype._loadXhr = function _loadXhr() {\n        // if unset, determine the value\n        if (typeof this.xhrType !== 'string') {\n            this.xhrType = this._determineXhrType();\n        }\n\n        var xhr = this.xhr = new XMLHttpRequest();\n\n        // set the request type and url\n        xhr.open('GET', this.url, true);\n\n        // load json as text and parse it ourselves. We do this because some browsers\n        // *cough* safari *cough* can't deal with it.\n        if (this.xhrType === Resource.XHR_RESPONSE_TYPE.JSON || this.xhrType === Resource.XHR_RESPONSE_TYPE.DOCUMENT) {\n            xhr.responseType = Resource.XHR_RESPONSE_TYPE.TEXT;\n        } else {\n            xhr.responseType = this.xhrType;\n        }\n\n        xhr.addEventListener('error', this._boundXhrOnError, false);\n        xhr.addEventListener('abort', this._boundXhrOnAbort, false);\n        xhr.addEventListener('progress', this._boundOnProgress, false);\n        xhr.addEventListener('load', this._boundXhrOnLoad, false);\n\n        xhr.send();\n    };\n\n    /**\n     * Loads this resources using an XDomainRequest. This is here because we need to support IE9 (gross).\n     *\n     * @private\n     */\n\n\n    Resource.prototype._loadXdr = function _loadXdr() {\n        // if unset, determine the value\n        if (typeof this.xhrType !== 'string') {\n            this.xhrType = this._determineXhrType();\n        }\n\n        var xdr = this.xhr = new XDomainRequest();\n\n        // XDomainRequest has a few quirks. Occasionally it will abort requests\n        // A way to avoid this is to make sure ALL callbacks are set even if not used\n        // More info here: http://stackoverflow.com/questions/15786966/xdomainrequest-aborts-post-on-ie-9\n        xdr.timeout = 5000;\n\n        xdr.onerror = this._boundXhrOnError;\n        xdr.ontimeout = this._boundXdrOnTimeout;\n        xdr.onprogress = this._boundOnProgress;\n        xdr.onload = this._boundXhrOnLoad;\n\n        xdr.open('GET', this.url, true);\n\n        // Note: The xdr.send() call is wrapped in a timeout to prevent an\n        // issue with the interface where some requests are lost if multiple\n        // XDomainRequests are being sent at the same time.\n        // Some info here: https://github.com/photonstorm/phaser/issues/1248\n        setTimeout(function () {\n            return xdr.send();\n        }, 1);\n    };\n\n    /**\n     * Creates a source used in loading via an element.\n     *\n     * @private\n     * @param {string} type - The element type (video or audio).\n     * @param {string} url - The source URL to load from.\n     * @param {string} [mime] - The mime type of the video\n     * @return {HTMLSourceElement} The source element.\n     */\n\n\n    Resource.prototype._createSource = function _createSource(type, url, mime) {\n        if (!mime) {\n            mime = type + '/' + this._getExtension(url);\n        }\n\n        var source = document.createElement('source');\n\n        source.src = url;\n        source.type = mime;\n\n        return source;\n    };\n\n    /**\n     * Called if a load errors out.\n     *\n     * @param {Event} event - The error event from the element that emits it.\n     * @private\n     */\n\n\n    Resource.prototype._onError = function _onError(event) {\n        this.abort('Failed to load element using: ' + event.target.nodeName);\n    };\n\n    /**\n     * Called if a load progress event fires for xhr/xdr.\n     *\n     * @private\n     * @param {XMLHttpRequestProgressEvent|Event} event - Progress event.\n     */\n\n\n    Resource.prototype._onProgress = function _onProgress(event) {\n        if (event && event.lengthComputable) {\n            this.onProgress.dispatch(this, event.loaded / event.total);\n        }\n    };\n\n    /**\n     * Called if an error event fires for xhr/xdr.\n     *\n     * @private\n     * @param {XMLHttpRequestErrorEvent|Event} event - Error event.\n     */\n\n\n    Resource.prototype._xhrOnError = function _xhrOnError() {\n        var xhr = this.xhr;\n\n        this.abort(reqType(xhr) + ' Request failed. Status: ' + xhr.status + ', text: \"' + xhr.statusText + '\"');\n    };\n\n    /**\n     * Called if an abort event fires for xhr.\n     *\n     * @private\n     * @param {XMLHttpRequestAbortEvent} event - Abort Event\n     */\n\n\n    Resource.prototype._xhrOnAbort = function _xhrOnAbort() {\n        this.abort(reqType(this.xhr) + ' Request was aborted by the user.');\n    };\n\n    /**\n     * Called if a timeout event fires for xdr.\n     *\n     * @private\n     * @param {Event} event - Timeout event.\n     */\n\n\n    Resource.prototype._xdrOnTimeout = function _xdrOnTimeout() {\n        this.abort(reqType(this.xhr) + ' Request timed out.');\n    };\n\n    /**\n     * Called when data successfully loads from an xhr/xdr request.\n     *\n     * @private\n     * @param {XMLHttpRequestLoadEvent|Event} event - Load event\n     */\n\n\n    Resource.prototype._xhrOnLoad = function _xhrOnLoad() {\n        var xhr = this.xhr;\n        var text = '';\n        var status = typeof xhr.status === 'undefined' ? STATUS_OK : xhr.status; // XDR has no `.status`, assume 200.\n\n        // responseText is accessible only if responseType is '' or 'text' and on older browsers\n        if (xhr.responseType === '' || xhr.responseType === 'text' || typeof xhr.responseType === 'undefined') {\n            text = xhr.responseText;\n        }\n\n        // status can be 0 when using the `file://` protocol so we also check if a response is set.\n        // If it has a response, we assume 200; otherwise a 0 status code with no contents is an aborted request.\n        if (status === STATUS_NONE && (text.length > 0 || xhr.responseType === Resource.XHR_RESPONSE_TYPE.BUFFER)) {\n            status = STATUS_OK;\n        }\n        // handle IE9 bug: http://stackoverflow.com/questions/10046972/msie-returns-status-code-of-1223-for-ajax-request\n        else if (status === STATUS_IE_BUG_EMPTY) {\n                status = STATUS_EMPTY;\n            }\n\n        var statusType = status / 100 | 0;\n\n        if (statusType === STATUS_TYPE_OK) {\n            // if text, just return it\n            if (this.xhrType === Resource.XHR_RESPONSE_TYPE.TEXT) {\n                this.data = text;\n                this.type = Resource.TYPE.TEXT;\n            }\n            // if json, parse into json object\n            else if (this.xhrType === Resource.XHR_RESPONSE_TYPE.JSON) {\n                    try {\n                        this.data = JSON.parse(text);\n                        this.type = Resource.TYPE.JSON;\n                    } catch (e) {\n                        this.abort('Error trying to parse loaded json: ' + e);\n\n                        return;\n                    }\n                }\n                // if xml, parse into an xml document or div element\n                else if (this.xhrType === Resource.XHR_RESPONSE_TYPE.DOCUMENT) {\n                        try {\n                            if (window.DOMParser) {\n                                var domparser = new DOMParser();\n\n                                this.data = domparser.parseFromString(text, 'text/xml');\n                            } else {\n                                var div = document.createElement('div');\n\n                                div.innerHTML = text;\n\n                                this.data = div;\n                            }\n\n                            this.type = Resource.TYPE.XML;\n                        } catch (e) {\n                            this.abort('Error trying to parse loaded xml: ' + e);\n\n                            return;\n                        }\n                    }\n                    // other types just return the response\n                    else {\n                            this.data = xhr.response || text;\n                        }\n        } else {\n            this.abort('[' + xhr.status + '] ' + xhr.statusText + ': ' + xhr.responseURL);\n\n            return;\n        }\n\n        this.complete();\n    };\n\n    /**\n     * Sets the `crossOrigin` property for this resource based on if the url\n     * for this resource is cross-origin. If crossOrigin was manually set, this\n     * function does nothing.\n     *\n     * @private\n     * @param {string} url - The url to test.\n     * @param {object} [loc=window.location] - The location object to test against.\n     * @return {string} The crossOrigin value to use (or empty string for none).\n     */\n\n\n    Resource.prototype._determineCrossOrigin = function _determineCrossOrigin(url, loc) {\n        // data: and javascript: urls are considered same-origin\n        if (url.indexOf('data:') === 0) {\n            return '';\n        }\n\n        // default is window.location\n        loc = loc || window.location;\n\n        if (!tempAnchor) {\n            tempAnchor = document.createElement('a');\n        }\n\n        // let the browser determine the full href for the url of this resource and then\n        // parse with the node url lib, we can't use the properties of the anchor element\n        // because they don't work in IE9 :(\n        tempAnchor.href = url;\n        url = (0, _parseUri2.default)(tempAnchor.href, { strictMode: true });\n\n        var samePort = !url.port && loc.port === '' || url.port === loc.port;\n        var protocol = url.protocol ? url.protocol + ':' : '';\n\n        // if cross origin\n        if (url.host !== loc.hostname || !samePort || protocol !== loc.protocol) {\n            return 'anonymous';\n        }\n\n        return '';\n    };\n\n    /**\n     * Determines the responseType of an XHR request based on the extension of the\n     * resource being loaded.\n     *\n     * @private\n     * @return {Resource.XHR_RESPONSE_TYPE} The responseType to use.\n     */\n\n\n    Resource.prototype._determineXhrType = function _determineXhrType() {\n        return Resource._xhrTypeMap[this.extension] || Resource.XHR_RESPONSE_TYPE.TEXT;\n    };\n\n    /**\n     * Determines the loadType of a resource based on the extension of the\n     * resource being loaded.\n     *\n     * @private\n     * @return {Resource.LOAD_TYPE} The loadType to use.\n     */\n\n\n    Resource.prototype._determineLoadType = function _determineLoadType() {\n        return Resource._loadTypeMap[this.extension] || Resource.LOAD_TYPE.XHR;\n    };\n\n    /**\n     * Extracts the extension (sans '.') of the file being loaded by the resource.\n     *\n     * @private\n     * @return {string} The extension.\n     */\n\n\n    Resource.prototype._getExtension = function _getExtension() {\n        var url = this.url;\n        var ext = '';\n\n        if (this.isDataUrl) {\n            var slashIndex = url.indexOf('/');\n\n            ext = url.substring(slashIndex + 1, url.indexOf(';', slashIndex));\n        } else {\n            var queryStart = url.indexOf('?');\n            var hashStart = url.indexOf('#');\n            var index = Math.min(queryStart > -1 ? queryStart : url.length, hashStart > -1 ? hashStart : url.length);\n\n            url = url.substring(0, index);\n            ext = url.substring(url.lastIndexOf('.') + 1);\n        }\n\n        return ext.toLowerCase();\n    };\n\n    /**\n     * Determines the mime type of an XHR request based on the responseType of\n     * resource being loaded.\n     *\n     * @private\n     * @param {Resource.XHR_RESPONSE_TYPE} type - The type to get a mime type for.\n     * @return {string} The mime type to use.\n     */\n\n\n    Resource.prototype._getMimeFromXhrType = function _getMimeFromXhrType(type) {\n        switch (type) {\n            case Resource.XHR_RESPONSE_TYPE.BUFFER:\n                return 'application/octet-binary';\n\n            case Resource.XHR_RESPONSE_TYPE.BLOB:\n                return 'application/blob';\n\n            case Resource.XHR_RESPONSE_TYPE.DOCUMENT:\n                return 'application/xml';\n\n            case Resource.XHR_RESPONSE_TYPE.JSON:\n                return 'application/json';\n\n            case Resource.XHR_RESPONSE_TYPE.DEFAULT:\n            case Resource.XHR_RESPONSE_TYPE.TEXT:\n            /* falls through */\n            default:\n                return 'text/plain';\n\n        }\n    };\n\n    _createClass(Resource, [{\n        key: 'isDataUrl',\n        get: function get() {\n            return this._hasFlag(Resource.STATUS_FLAGS.DATA_URL);\n        }\n\n        /**\n         * Describes if this resource has finished loading. Is true when the resource has completely\n         * loaded.\n         *\n         * @member {boolean}\n         * @readonly\n         */\n\n    }, {\n        key: 'isComplete',\n        get: function get() {\n            return this._hasFlag(Resource.STATUS_FLAGS.COMPLETE);\n        }\n\n        /**\n         * Describes if this resource is currently loading. Is true when the resource starts loading,\n         * and is false again when complete.\n         *\n         * @member {boolean}\n         * @readonly\n         */\n\n    }, {\n        key: 'isLoading',\n        get: function get() {\n            return this._hasFlag(Resource.STATUS_FLAGS.LOADING);\n        }\n    }]);\n\n    return Resource;\n}();\n\n/**\n * The types of resources a resource could represent.\n *\n * @static\n * @readonly\n * @enum {number}\n */\n\n\nexports.default = Resource;\nResource.STATUS_FLAGS = {\n    NONE: 0,\n    DATA_URL: 1 << 0,\n    COMPLETE: 1 << 1,\n    LOADING: 1 << 2\n};\n\n/**\n * The types of resources a resource could represent.\n *\n * @static\n * @readonly\n * @enum {number}\n */\nResource.TYPE = {\n    UNKNOWN: 0,\n    JSON: 1,\n    XML: 2,\n    IMAGE: 3,\n    AUDIO: 4,\n    VIDEO: 5,\n    TEXT: 6\n};\n\n/**\n * The types of loading a resource can use.\n *\n * @static\n * @readonly\n * @enum {number}\n */\nResource.LOAD_TYPE = {\n    /** Uses XMLHttpRequest to load the resource. */\n    XHR: 1,\n    /** Uses an `Image` object to load the resource. */\n    IMAGE: 2,\n    /** Uses an `Audio` object to load the resource. */\n    AUDIO: 3,\n    /** Uses a `Video` object to load the resource. */\n    VIDEO: 4\n};\n\n/**\n * The XHR ready states, used internally.\n *\n * @static\n * @readonly\n * @enum {string}\n */\nResource.XHR_RESPONSE_TYPE = {\n    /** string */\n    DEFAULT: 'text',\n    /** ArrayBuffer */\n    BUFFER: 'arraybuffer',\n    /** Blob */\n    BLOB: 'blob',\n    /** Document */\n    DOCUMENT: 'document',\n    /** Object */\n    JSON: 'json',\n    /** String */\n    TEXT: 'text'\n};\n\nResource._loadTypeMap = {\n    // images\n    gif: Resource.LOAD_TYPE.IMAGE,\n    png: Resource.LOAD_TYPE.IMAGE,\n    bmp: Resource.LOAD_TYPE.IMAGE,\n    jpg: Resource.LOAD_TYPE.IMAGE,\n    jpeg: Resource.LOAD_TYPE.IMAGE,\n    tif: Resource.LOAD_TYPE.IMAGE,\n    tiff: Resource.LOAD_TYPE.IMAGE,\n    webp: Resource.LOAD_TYPE.IMAGE,\n    tga: Resource.LOAD_TYPE.IMAGE,\n    svg: Resource.LOAD_TYPE.IMAGE,\n    'svg+xml': Resource.LOAD_TYPE.IMAGE, // for SVG data urls\n\n    // audio\n    mp3: Resource.LOAD_TYPE.AUDIO,\n    ogg: Resource.LOAD_TYPE.AUDIO,\n    wav: Resource.LOAD_TYPE.AUDIO,\n\n    // videos\n    mp4: Resource.LOAD_TYPE.VIDEO,\n    webm: Resource.LOAD_TYPE.VIDEO\n};\n\nResource._xhrTypeMap = {\n    // xml\n    xhtml: Resource.XHR_RESPONSE_TYPE.DOCUMENT,\n    html: Resource.XHR_RESPONSE_TYPE.DOCUMENT,\n    htm: Resource.XHR_RESPONSE_TYPE.DOCUMENT,\n    xml: Resource.XHR_RESPONSE_TYPE.DOCUMENT,\n    tmx: Resource.XHR_RESPONSE_TYPE.DOCUMENT,\n    svg: Resource.XHR_RESPONSE_TYPE.DOCUMENT,\n\n    // This was added to handle Tiled Tileset XML, but .tsx is also a TypeScript React Component.\n    // Since it is way less likely for people to be loading TypeScript files instead of Tiled files,\n    // this should probably be fine.\n    tsx: Resource.XHR_RESPONSE_TYPE.DOCUMENT,\n\n    // images\n    gif: Resource.XHR_RESPONSE_TYPE.BLOB,\n    png: Resource.XHR_RESPONSE_TYPE.BLOB,\n    bmp: Resource.XHR_RESPONSE_TYPE.BLOB,\n    jpg: Resource.XHR_RESPONSE_TYPE.BLOB,\n    jpeg: Resource.XHR_RESPONSE_TYPE.BLOB,\n    tif: Resource.XHR_RESPONSE_TYPE.BLOB,\n    tiff: Resource.XHR_RESPONSE_TYPE.BLOB,\n    webp: Resource.XHR_RESPONSE_TYPE.BLOB,\n    tga: Resource.XHR_RESPONSE_TYPE.BLOB,\n\n    // json\n    json: Resource.XHR_RESPONSE_TYPE.JSON,\n\n    // text\n    text: Resource.XHR_RESPONSE_TYPE.TEXT,\n    txt: Resource.XHR_RESPONSE_TYPE.TEXT,\n\n    // fonts\n    ttf: Resource.XHR_RESPONSE_TYPE.BUFFER,\n    otf: Resource.XHR_RESPONSE_TYPE.BUFFER\n};\n\n// We can't set the `src` attribute to empty string, so on abort we set it to this 1px transparent gif\nResource.EMPTY_GIF = 'data:image/gif;base64,R0lGODlhAQABAIAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw==';\n\n/**\n * Quick helper to set a value on one of the extension maps. Ensures there is no\n * dot at the start of the extension.\n *\n * @ignore\n * @param {object} map - The map to set on.\n * @param {string} extname - The extension (or key) to set.\n * @param {number} val - The value to set.\n */\nfunction setExtMap(map, extname, val) {\n    if (extname && extname.indexOf('.') === 0) {\n        extname = extname.substring(1);\n    }\n\n    if (!extname) {\n        return;\n    }\n\n    map[extname] = val;\n}\n\n/**\n * Quick helper to get string xhr type.\n *\n * @ignore\n * @param {XMLHttpRequest|XDomainRequest} xhr - The request to check.\n * @return {string} The type.\n */\nfunction reqType(xhr) {\n    return xhr.toString().replace('object ', '');\n}\n//# sourceMappingURL=Resource.js.map","'use strict';\n\nexports.__esModule = true;\nexports.eachSeries = eachSeries;\nexports.queue = queue;\n/**\n * Smaller version of the async library constructs.\n *\n */\nfunction _noop() {} /* empty */\n\n/**\n * Iterates an array in series.\n *\n * @param {Array.<*>} array - Array to iterate.\n * @param {function} iterator - Function to call for each element.\n * @param {function} callback - Function to call when done, or on error.\n * @param {boolean} [deferNext=false] - Break synchronous each loop by calling next with a setTimeout of 1.\n */\nfunction eachSeries(array, iterator, callback, deferNext) {\n    var i = 0;\n    var len = array.length;\n\n    (function next(err) {\n        if (err || i === len) {\n            if (callback) {\n                callback(err);\n            }\n\n            return;\n        }\n\n        if (deferNext) {\n            setTimeout(function () {\n                iterator(array[i++], next);\n            }, 1);\n        } else {\n            iterator(array[i++], next);\n        }\n    })();\n}\n\n/**\n * Ensures a function is only called once.\n *\n * @param {function} fn - The function to wrap.\n * @return {function} The wrapping function.\n */\nfunction onlyOnce(fn) {\n    return function onceWrapper() {\n        if (fn === null) {\n            throw new Error('Callback was already called.');\n        }\n\n        var callFn = fn;\n\n        fn = null;\n        callFn.apply(this, arguments);\n    };\n}\n\n/**\n * Async queue implementation,\n *\n * @param {function} worker - The worker function to call for each task.\n * @param {number} concurrency - How many workers to run in parrallel.\n * @return {*} The async queue object.\n */\nfunction queue(worker, concurrency) {\n    if (concurrency == null) {\n        // eslint-disable-line no-eq-null,eqeqeq\n        concurrency = 1;\n    } else if (concurrency === 0) {\n        throw new Error('Concurrency must not be zero');\n    }\n\n    var workers = 0;\n    var q = {\n        _tasks: [],\n        concurrency: concurrency,\n        saturated: _noop,\n        unsaturated: _noop,\n        buffer: concurrency / 4,\n        empty: _noop,\n        drain: _noop,\n        error: _noop,\n        started: false,\n        paused: false,\n        push: function push(data, callback) {\n            _insert(data, false, callback);\n        },\n        kill: function kill() {\n            workers = 0;\n            q.drain = _noop;\n            q.started = false;\n            q._tasks = [];\n        },\n        unshift: function unshift(data, callback) {\n            _insert(data, true, callback);\n        },\n        process: function process() {\n            while (!q.paused && workers < q.concurrency && q._tasks.length) {\n                var task = q._tasks.shift();\n\n                if (q._tasks.length === 0) {\n                    q.empty();\n                }\n\n                workers += 1;\n\n                if (workers === q.concurrency) {\n                    q.saturated();\n                }\n\n                worker(task.data, onlyOnce(_next(task)));\n            }\n        },\n        length: function length() {\n            return q._tasks.length;\n        },\n        running: function running() {\n            return workers;\n        },\n        idle: function idle() {\n            return q._tasks.length + workers === 0;\n        },\n        pause: function pause() {\n            if (q.paused === true) {\n                return;\n            }\n\n            q.paused = true;\n        },\n        resume: function resume() {\n            if (q.paused === false) {\n                return;\n            }\n\n            q.paused = false;\n\n            // Need to call q.process once per concurrent\n            // worker to preserve full concurrency after pause\n            for (var w = 1; w <= q.concurrency; w++) {\n                q.process();\n            }\n        }\n    };\n\n    function _insert(data, insertAtFront, callback) {\n        if (callback != null && typeof callback !== 'function') {\n            // eslint-disable-line no-eq-null,eqeqeq\n            throw new Error('task callback must be a function');\n        }\n\n        q.started = true;\n\n        if (data == null && q.idle()) {\n            // eslint-disable-line no-eq-null,eqeqeq\n            // call drain immediately if there are no tasks\n            setTimeout(function () {\n                return q.drain();\n            }, 1);\n\n            return;\n        }\n\n        var item = {\n            data: data,\n            callback: typeof callback === 'function' ? callback : _noop\n        };\n\n        if (insertAtFront) {\n            q._tasks.unshift(item);\n        } else {\n            q._tasks.push(item);\n        }\n\n        setTimeout(function () {\n            return q.process();\n        }, 1);\n    }\n\n    function _next(task) {\n        return function next() {\n            workers -= 1;\n\n            task.callback.apply(task, arguments);\n\n            if (arguments[0] != null) {\n                // eslint-disable-line no-eq-null,eqeqeq\n                q.error(arguments[0], task.data);\n            }\n\n            if (workers <= q.concurrency - q.buffer) {\n                q.unsaturated();\n            }\n\n            if (q.idle()) {\n                q.drain();\n            }\n\n            q.process();\n        };\n    }\n\n    return q;\n}\n//# sourceMappingURL=async.js.map","'use strict';\n\nexports.__esModule = true;\nexports.encodeBinary = encodeBinary;\nvar _keyStr = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';\n\nfunction encodeBinary(input) {\n    var output = '';\n    var inx = 0;\n\n    while (inx < input.length) {\n        // Fill byte buffer array\n        var bytebuffer = [0, 0, 0];\n        var encodedCharIndexes = [0, 0, 0, 0];\n\n        for (var jnx = 0; jnx < bytebuffer.length; ++jnx) {\n            if (inx < input.length) {\n                // throw away high-order byte, as documented at:\n                // https://developer.mozilla.org/En/Using_XMLHttpRequest#Handling_binary_data\n                bytebuffer[jnx] = input.charCodeAt(inx++) & 0xff;\n            } else {\n                bytebuffer[jnx] = 0;\n            }\n        }\n\n        // Get each encoded character, 6 bits at a time\n        // index 1: first 6 bits\n        encodedCharIndexes[0] = bytebuffer[0] >> 2;\n\n        // index 2: second 6 bits (2 least significant bits from input byte 1 + 4 most significant bits from byte 2)\n        encodedCharIndexes[1] = (bytebuffer[0] & 0x3) << 4 | bytebuffer[1] >> 4;\n\n        // index 3: third 6 bits (4 least significant bits from input byte 2 + 2 most significant bits from byte 3)\n        encodedCharIndexes[2] = (bytebuffer[1] & 0x0f) << 2 | bytebuffer[2] >> 6;\n\n        // index 3: forth 6 bits (6 least significant bits from input byte 3)\n        encodedCharIndexes[3] = bytebuffer[2] & 0x3f;\n\n        // Determine whether padding happened, and adjust accordingly\n        var paddingBytes = inx - (input.length - 1);\n\n        switch (paddingBytes) {\n            case 2:\n                // Set last 2 characters to padding char\n                encodedCharIndexes[3] = 64;\n                encodedCharIndexes[2] = 64;\n                break;\n\n            case 1:\n                // Set last character to padding char\n                encodedCharIndexes[3] = 64;\n                break;\n\n            default:\n                break; // No padding - proceed\n        }\n\n        // Now we will grab each appropriate character out of our keystring\n        // based on our index array and append it to the output string\n        for (var _jnx = 0; _jnx < encodedCharIndexes.length; ++_jnx) {\n            output += _keyStr.charAt(encodedCharIndexes[_jnx]);\n        }\n    }\n\n    return output;\n}\n//# sourceMappingURL=b64.js.map","'use strict';\n\n// import Loader from './Loader';\n// import Resource from './Resource';\n// import * as async from './async';\n// import * as b64 from './b64';\n\n/* eslint-disable no-undef */\n\nvar Loader = require('./Loader').default;\nvar Resource = require('./Resource').default;\nvar async = require('./async');\nvar b64 = require('./b64');\n\nLoader.Resource = Resource;\nLoader.async = async;\nLoader.base64 = b64;\n\n// export manually, and also as default\nmodule.exports = Loader;\n// export default Loader;\nmodule.exports.default = Loader;\n//# sourceMappingURL=index.js.map","'use strict';\n\nObject.defineProperty(exports, '__esModule', {\n  value: true\n});\n\nvar _createClass = (function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ('value' in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; })();\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError('Cannot call a class as a function'); } }\n\nvar MiniSignalBinding = (function () {\n  function MiniSignalBinding(fn, once, thisArg) {\n    if (once === undefined) once = false;\n\n    _classCallCheck(this, MiniSignalBinding);\n\n    this._fn = fn;\n    this._once = once;\n    this._thisArg = thisArg;\n    this._next = this._prev = this._owner = null;\n  }\n\n  _createClass(MiniSignalBinding, [{\n    key: 'detach',\n    value: function detach() {\n      if (this._owner === null) return false;\n      this._owner.detach(this);\n      return true;\n    }\n  }]);\n\n  return MiniSignalBinding;\n})();\n\nfunction _addMiniSignalBinding(self, node) {\n  if (!self._head) {\n    self._head = node;\n    self._tail = node;\n  } else {\n    self._tail._next = node;\n    node._prev = self._tail;\n    self._tail = node;\n  }\n\n  node._owner = self;\n\n  return node;\n}\n\nvar MiniSignal = (function () {\n  function MiniSignal() {\n    _classCallCheck(this, MiniSignal);\n\n    this._head = this._tail = undefined;\n  }\n\n  _createClass(MiniSignal, [{\n    key: 'handlers',\n    value: function handlers() {\n      var exists = arguments.length <= 0 || arguments[0] === undefined ? false : arguments[0];\n\n      var node = this._head;\n\n      if (exists) return !!node;\n\n      var ee = [];\n\n      while (node) {\n        ee.push(node);\n        node = node._next;\n      }\n\n      return ee;\n    }\n  }, {\n    key: 'has',\n    value: function has(node) {\n      if (!(node instanceof MiniSignalBinding)) {\n        throw new Error('MiniSignal#has(): First arg must be a MiniSignalBinding object.');\n      }\n\n      return node._owner === this;\n    }\n  }, {\n    key: 'dispatch',\n    value: function dispatch() {\n      var node = this._head;\n\n      if (!node) return false;\n\n      while (node) {\n        if (node._once) this.detach(node);\n        node._fn.apply(node._thisArg, arguments);\n        node = node._next;\n      }\n\n      return true;\n    }\n  }, {\n    key: 'add',\n    value: function add(fn) {\n      var thisArg = arguments.length <= 1 || arguments[1] === undefined ? null : arguments[1];\n\n      if (typeof fn !== 'function') {\n        throw new Error('MiniSignal#add(): First arg must be a Function.');\n      }\n      return _addMiniSignalBinding(this, new MiniSignalBinding(fn, false, thisArg));\n    }\n  }, {\n    key: 'once',\n    value: function once(fn) {\n      var thisArg = arguments.length <= 1 || arguments[1] === undefined ? null : arguments[1];\n\n      if (typeof fn !== 'function') {\n        throw new Error('MiniSignal#once(): First arg must be a Function.');\n      }\n      return _addMiniSignalBinding(this, new MiniSignalBinding(fn, true, thisArg));\n    }\n  }, {\n    key: 'detach',\n    value: function detach(node) {\n      if (!(node instanceof MiniSignalBinding)) {\n        throw new Error('MiniSignal#detach(): First arg must be a MiniSignalBinding object.');\n      }\n      if (node._owner !== this) return this;\n\n      if (node._prev) node._prev._next = node._next;\n      if (node._next) node._next._prev = node._prev;\n\n      if (node === this._head) {\n        this._head = node._next;\n        if (node._next === null) {\n          this._tail = null;\n        }\n      } else if (node === this._tail) {\n        this._tail = node._prev;\n        this._tail._next = null;\n      }\n\n      node._owner = null;\n      return this;\n    }\n  }, {\n    key: 'detachAll',\n    value: function detachAll() {\n      var node = this._head;\n      if (!node) return this;\n\n      this._head = this._tail = null;\n\n      while (node) {\n        node._owner = null;\n        node = node._next;\n      }\n      return this;\n    }\n  }]);\n\n  return MiniSignal;\n})();\n\nMiniSignal.MiniSignalBinding = MiniSignalBinding;\n\nexports['default'] = MiniSignal;\nmodule.exports = exports['default'];\n","'use strict'\n\nmodule.exports = function parseURI (str, opts) {\n  opts = opts || {}\n\n  var o = {\n    key: ['source', 'protocol', 'authority', 'userInfo', 'user', 'password', 'host', 'port', 'relative', 'path', 'directory', 'file', 'query', 'anchor'],\n    q: {\n      name: 'queryKey',\n      parser: /(?:^|&)([^&=]*)=?([^&]*)/g\n    },\n    parser: {\n      strict: /^(?:([^:\\/?#]+):)?(?:\\/\\/((?:(([^:@]*)(?::([^:@]*))?)?@)?([^:\\/?#]*)(?::(\\d*))?))?((((?:[^?#\\/]*\\/)*)([^?#]*))(?:\\?([^#]*))?(?:#(.*))?)/,\n      loose: /^(?:(?![^:@]+:[^:@\\/]*@)([^:\\/?#.]+):)?(?:\\/\\/)?((?:(([^:@]*)(?::([^:@]*))?)?@)?([^:\\/?#]*)(?::(\\d*))?)(((\\/(?:[^?#](?![^?#\\/]*\\.[^?#\\/.]+(?:[?#]|$)))*\\/?)?([^?#\\/]*))(?:\\?([^#]*))?(?:#(.*))?)/\n    }\n  }\n\n  var m = o.parser[opts.strictMode ? 'strict' : 'loose'].exec(str)\n  var uri = {}\n  var i = 14\n\n  while (i--) uri[o.key[i]] = m[i] || ''\n\n  uri[o.q.name] = {}\n  uri[o.key[12]].replace(o.q.parser, function ($0, $1, $2) {\n    if ($1) uri[o.q.name][$1] = $2\n  })\n\n  return uri\n}\n"]}