/** * @author Richard Davey * @copyright 2018 Photon Storm Ltd. * @license {@link https://github.com/photonstorm/phaser/blob/master/license.txt|MIT License} */ var Class = require('../utils/Class'); var CONST = require('./const'); var CustomSet = require('../structs/Set'); var EventEmitter = require('eventemitter3'); var FileTypesManager = require('./FileTypesManager'); var GetFastValue = require('../utils/object/GetFastValue'); var PluginManager = require('../boot/PluginManager'); var XHRSettings = require('./XHRSettings'); /** * @typedef {object} LinkFileObject * * @property {string} type - [description] * @property {Phaser.Loader.File} fileA - [description] * @property {Phaser.Loader.File} fileB - [description] */ /** * @typedef {object} LoaderFileObject * * @property {string} key - [description] * @property {string} type - [description] * @property {string} [url] - [description] * @property {string[]} [urls] - [description] * @property {string} [textureURL] - [description] * @property {string} [atlasURL] - [description] * @property {string} [xmlURL] - [description] * @property {string[]} [textureURLs] - [description] * @property {string[]} [atlasURLs] - [description] * @property {object} [config] - [description] * @property {object} [json] - [description] * @property {XHRSettingsObject} [xhrSettings] - [description] * @property {XHRSettingsObject} [textureXhrSettings] - [description] * @property {XHRSettingsObject} [atlasXhrSettings] - [description] * @property {XHRSettingsObject} [xmlXhrSettings] - [description] * @property {XHRSettingsObject} [audioXhrSettings] - [description] * @property {XHRSettingsObject} [jsonXhrSettings] - [description] */ /** * @classdesc * [description] * * @class LoaderPlugin * @extends Phaser.Events.EventEmitter * @memberOf Phaser.Loader * @constructor * @since 3.0.0 * * @param {Phaser.Scene} scene - [description] */ var LoaderPlugin = new Class({ Extends: EventEmitter, initialize: function LoaderPlugin (scene) { EventEmitter.call(this); var gameConfig = scene.sys.game.config; var sceneConfig = scene.sys.settings.loader; /** * [description] * * @name Phaser.Loader.LoaderPlugin#scene * @type {Phaser.Scene} * @since 3.0.0 */ this.scene = scene; /** * [description] * * @name Phaser.Loader.LoaderPlugin#systems * @type {Phaser.Scenes.Systems} * @since 3.0.0 */ this.systems = scene.sys; /** * A reference to the global Cache Manager. * * @name Phaser.Loader.LoaderPlugin#cacheManager * @type {Phaser.Cache.CacheManager} * @since 3.7.0 */ this.cacheManager = scene.sys.cache; /** * A reference to the global Texture Manager. * * @name Phaser.Loader.LoaderPlugin#textureManager * @type {Phaser.Textures.TextureManager} * @since 3.7.0 */ this.textureManager = scene.sys.textures; // Inject the available filetypes into the Loader FileTypesManager.install(this); /** * An optional prefix that is automatically prepended to the start of every file key. * If prefix was `MENU` and you load an image with the key 'Background' the resulting key would be `MENUBackground`. * * @name Phaser.Loader.LoaderPlugin#prefix * @type {string} * @default '' * @since 3.7.0 */ this.prefix = ''; /** * [description] * * @name Phaser.Loader.LoaderPlugin#path * @type {string} * @default '' * @since 3.0.0 */ this.path = ''; /** * [description] * * @name Phaser.Loader.LoaderPlugin#baseURL * @type {string} * @default '' * @since 3.0.0 */ this.baseURL = ''; this.setBaseURL(GetFastValue(sceneConfig, 'baseURL', gameConfig.loaderBaseURL)); this.setPath(GetFastValue(sceneConfig, 'path', gameConfig.loaderPath)); this.setPrefix(GetFastValue(sceneConfig, 'prefix', gameConfig.loaderPrefix)); /** * [description] * * @name Phaser.Loader.LoaderPlugin#enableParallel * @type {boolean} * @since 3.0.0 */ this.enableParallel = GetFastValue(sceneConfig, 'enableParallel', gameConfig.loaderEnableParallel); /** * [description] * * @name Phaser.Loader.LoaderPlugin#maxParallelDownloads * @type {integer} * @since 3.0.0 */ this.maxParallelDownloads = GetFastValue(sceneConfig, 'maxParallelDownloads', gameConfig.loaderMaxParallelDownloads); /** * xhr specific global settings (can be overridden on a per-file basis) * * @name Phaser.Loader.LoaderPlugin#xhr * @type {XHRSettingsObject} * @since 3.0.0 */ this.xhr = XHRSettings( GetFastValue(sceneConfig, 'responseType', gameConfig.loaderResponseType), GetFastValue(sceneConfig, 'async', gameConfig.loaderAsync), GetFastValue(sceneConfig, 'user', gameConfig.loaderUser), GetFastValue(sceneConfig, 'password', gameConfig.loaderPassword), GetFastValue(sceneConfig, 'timeout', gameConfig.loaderTimeout) ); /** * [description] * * @name Phaser.Loader.LoaderPlugin#crossOrigin * @type {string} * @since 3.0.0 */ this.crossOrigin = GetFastValue(sceneConfig, 'crossOrigin', gameConfig.loaderCrossOrigin); /** * [description] * * @name Phaser.Loader.LoaderPlugin#totalToLoad * @type {integer} * @default 0 * @since 3.0.0 */ this.totalToLoad = 0; /** * [description] * * @name Phaser.Loader.LoaderPlugin#progress * @type {float} * @default 0 * @since 3.0.0 */ this.progress = 0; /** * Files are placed in this Set when they're added to the Loader via `addFile`. * * They are moved to the `inflight` Set when they start loading, and assuming a successful * load, to the `queue` Set for further processing. * * By the end of the load process this Set will be empty. * * @name Phaser.Loader.LoaderPlugin#list * @type {Phaser.Structs.Set.} * @since 3.0.0 */ this.list = new CustomSet(); /** * Files are stored in this Set while they're in the process of being loaded. * * Upon a successful load they are moved to the `queue` Set. * * By the end of the load process this Set will be empty. * * @name Phaser.Loader.LoaderPlugin#inflight * @type {Phaser.Structs.Set.} * @since 3.0.0 */ this.inflight = new CustomSet(); /** * Files are stored in this Set while they're being processed. * * If the process is successful they are moved to their final destination, which could be * a Cache or the Texture Manager. * * At the end of the load process this Set will be empty. * * @name Phaser.Loader.LoaderPlugin#queue * @type {Phaser.Structs.Set.} * @since 3.0.0 */ this.queue = new CustomSet(); /** * [description] * * @name Phaser.Loader.LoaderPlugin#totalFailed * @type {integer} * @default 0 * @since 3.7.0 */ this.totalFailed = 0; /** * [description] * * @name Phaser.Loader.LoaderPlugin#totalComplete * @type {integer} * @default 0 * @since 3.7.0 */ this.totalComplete = 0; /** * [description] * * @name Phaser.Loader.LoaderPlugin#state * @type {integer} * @since 3.0.0 */ this.state = CONST.LOADER_IDLE; scene.sys.events.once('boot', this.boot, this); scene.sys.events.on('start', this.pluginStart, this); }, /** * This method is called automatically, only once, when the Scene is first created. * Do not invoke it directly. * * @method Phaser.Loader.LoaderPlugin#boot * @private * @since 3.5.1 */ boot: function () { this.systems.events.once('destroy', this.destroy, this); }, /** * This method is called automatically by the Scene when it is starting up. * It is responsible for creating local systems, properties and listening for Scene events. * Do not invoke it directly. * * @method Phaser.Loader.LoaderPlugin#pluginStart * @private * @since 3.5.1 */ pluginStart: function () { this.systems.events.once('shutdown', this.shutdown, this); }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#setBaseURL * @since 3.0.0 * * @param {string} url - [description] * * @return {Phaser.Loader.LoaderPlugin} This Loader object. */ setBaseURL: function (url) { if (url !== '' && url.substr(-1) !== '/') { url = url.concat('/'); } this.baseURL = url; return this; }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#setPath * @since 3.0.0 * * @param {string} path - [description] * * @return {Phaser.Loader.LoaderPlugin} This Loader object. */ setPath: function (path) { if (path !== '' && path.substr(-1) !== '/') { path = path.concat('/'); } this.path = path; return this; }, /** * An optional prefix that is automatically prepended to the start of every file key. * If prefix was `MENU` and you load an image with the key 'Background' the resulting key would be `MENUBackground`. * * @method Phaser.Loader.LoaderPlugin#setPrefix * @since 3.7.0 * * @param {string} prefix - The prefix to use. Leave empty to reset. * * @return {Phaser.Loader.LoaderPlugin} This Loader object. */ setPrefix: function (prefix) { if (prefix === undefined) { prefix = ''; } this.prefix = prefix; return this; }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#setCORS * @since 3.0.0 * * @param {string} crossOrigin - [description] * * @return {Phaser.Loader.LoaderPlugin} This Loader object. */ setCORS: function (crossOrigin) { this.crossOrigin = crossOrigin; return this; }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#addFile * @since 3.0.0 * * @param {(Phaser.Loader.File|Phaser.Loader.File[])} file - The file, or array of files, to be added to the load queue. */ addFile: function (file) { if (!Array.isArray(file)) { file = [ file ]; } for (var i = 0; i < file.length; i++) { var item = file[i]; // Does the file already exist in the cache or texture manager? // Or will it conflict with a file already in the queue or inflight? if (!this.keyExists(item)) { this.list.set(item); console.log('addFile', item.key); if (this.isLoading()) { this.totalToLoad++; this.updateProgress(); } } } }, /** * Checks the key and type of the given file to see if it will conflict with anything already * in a Cache, the Texture Manager, or the list or inflight queues. * * @method Phaser.Loader.LoaderPlugin#keyExists * @since 3.7.0 * * @param {Phaser.Loader.File} file - The file to check the key of. * * @return {boolean} `true` if adding this file will cause a cache or queue conflict, otherwise `false`. */ keyExists: function (file) { var keyConflict = file.hasCacheConflict(); if (!keyConflict) { this.list.iterate(function (item) { if (item.type === file.type && item.key === file.key) { keyConflict = true; return false; } }); } if (!keyConflict && this.isLoading()) { this.inflight.iterate(function (item) { if (item.type === file.type && item.key === file.key) { keyConflict = true; return false; } }); this.queue.iterate(function (item) { if (item.type === file.type && item.key === file.key) { keyConflict = true; return false; } }); } return keyConflict; }, /** * Is the Loader actively loading (or processing loaded files) * * @method Phaser.Loader.LoaderPlugin#isLoading * @since 3.0.0 * * @return {boolean} [description] */ isLoading: function () { return (this.state === CONST.LOADER_LOADING || this.state === CONST.LOADER_PROCESSING); }, /** * Is the Loader ready to start a new load? * * @method Phaser.Loader.LoaderPlugin#isReady * @since 3.0.0 * * @return {boolean} [description] */ isReady: function () { return (this.state === CONST.LOADER_IDLE || this.state === CONST.LOADER_COMPLETE); }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#start * @since 3.0.0 */ start: function () { if (!this.isReady()) { return; } this.progress = 0; this.totalFailed = 0; this.totalComplete = 0; this.totalToLoad = this.list.size; console.log('start', this.totalToLoad); this.emit('start', this); if (this.list.size === 0) { this.loadComplete(); } else { this.state = CONST.LOADER_LOADING; this.inflight.clear(); this.queue.clear(); this.updateProgress(); this.processLoadQueue(); } }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#updateProgress * @since 3.0.0 */ updateProgress: function () { this.progress = 1 - ((this.list.size + this.inflight.size) / this.totalToLoad); this.emit('progress', this.progress); }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#processLoadQueue * @since 3.0.0 */ processLoadQueue: function () { this.list.each(function (file) { if (file.state === CONST.FILE_POPULATED || (file.state === CONST.FILE_PENDING && this.inflight.size < this.maxParallelDownloads)) { this.inflight.set(file); this.list.delete(file); // If the file doesn't have its own crossOrigin set, // we'll use the Loaders (which is undefined by default) if (!file.crossOrigin) { file.crossOrigin = this.crossOrigin; } console.log('processLoadQueue', file.key); file.load(); } if (this.inflight.size === this.maxParallelDownloads) { // Tells the Set iterator to abort return false; } }, this); }, /** * Called automatically by the Files XHRLoader function. * * @method Phaser.Loader.LoaderPlugin#nextFile * @since 3.0.0 * * @param {Phaser.Loader.File} file - The File that just finished loading, or errored during load. * @param {boolean} success - `true` if the file loaded successfully, otherwise `false`. */ nextFile: function (file, success) { console.log('nextFile', file.key, success); this.inflight.delete(file); this.updateProgress(); if (success) { this.totalComplete++; this.queue.set(file); this.emit('load', file); console.log('nextFile 1'); file.onProcess(); } else { this.totalFailed++; this.emit('loaderror', file); console.log('nextFile 2'); } if (this.list.size > 0) { console.log('nextFile processLoadQueue'); this.processLoadQueue(); } }, /** * Called automatically by the File when it has finished processing. * * @method Phaser.Loader.LoaderPlugin#fileProcessComplete * @since 3.7.0 * * @param {Phaser.Loader.File} file - [description] */ fileProcessComplete: function (file) { // This file has failed, so move it to the failed Set if (file.state === CONST.FILE_ERRORED) { if (file.linkFile) { file.linkFile.onFileFailed(file); } } else if (file.state === CONST.FILE_COMPLETE) { if (file.linkFile) { // If we got here, then the file processed, so let it add itself to its cache file.linkFile.addToCache(); } else { file.addToCache(); } } // Remove it from the queue as it's now in the cache, or failed this.queue.delete(file); file.destroy(); // Nothing left to do? if (this.list.size === 0 && this.inflight.size === 0 && this.queue.size === 0) { this.loadComplete(); } else { // In case we've added to the list by processing this file this.processLoadQueue(); } }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#loadComplete * @since 3.7.0 */ loadComplete: function () { this.list.clear(); this.inflight.clear(); this.queue.clear(); this.progress = 1; this.state = CONST.LOADER_COMPLETE; this.emit('complete', this, this.totalComplete, this.totalFailed); }, /** * !!! TO BE DELETED !!! * !!! TO BE DELETED !!! * !!! TO BE DELETED !!! * * @method Phaser.Loader.LoaderPlugin#processCallback * @since 3.0.0 */ ___processCallback: function () { if (this.storage.size === 0) { return; } // Process all of the files this.storage.each(function (file) { if (file.linkFile) { file.linkFile.addToCache(); } else { file.addToCache(); } /* switch (file.type) { case 'dataimage': fileA = file.fileA; fileB = file.fileB; if (fileA.linkParent) { textures.addImage(fileA.key, fileA.data, fileB.data); } else { textures.addImage(fileB.key, fileB.data, fileA.data); } break; case 'audioSprite': var files = [ file.fileA, file.fileB ]; files.forEach(function (file) { cache[file.type].add(file.key, file.data); }); break; } */ }); this.emit('processcomplete', this); // Call 'destroy' on each file in storage this.storage.iterateLocal('destroy'); this.storage.clear(); }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#saveJSON * @since 3.0.0 * * @param {*} data - [description] * @param {string} [filename=file.json] - [description] * * @return {Phaser.Loader.LoaderPlugin} This Loader plugin. */ saveJSON: function (data, filename) { return this.save(JSON.stringify(data), filename); }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#save * @since 3.0.0 * * @param {*} data - [description] * @param {string} [filename=file.json] - [description] * @param {string} [filetype=application/json] - [description] * * @return {Phaser.Loader.LoaderPlugin} This Loader plugin. */ save: function (data, filename, filetype) { if (filename === undefined) { filename = 'file.json'; } if (filetype === undefined) { filetype = 'application/json'; } var blob = new Blob([ data ], { type: filetype }); var url = URL.createObjectURL(blob); var a = document.createElement('a'); a.download = filename; a.textContent = 'Download ' + filename; a.href = url; a.click(); return this; }, /** * [description] * * @method Phaser.Loader.LoaderPlugin#reset * @since 3.0.0 */ reset: function () { this.list.clear(); this.inflight.clear(); this.queue.clear(); var gameConfig = this.systems.game.config; var sceneConfig = this.systems.settings.loader; this.setBaseURL(GetFastValue(sceneConfig, 'baseURL', gameConfig.loaderBaseURL)); this.setPath(GetFastValue(sceneConfig, 'path', gameConfig.loaderPath)); this.setPrefix(GetFastValue(sceneConfig, 'prefix', gameConfig.loaderPrefix)); this.state = CONST.LOADER_IDLE; }, /** * !!! TO BE REPLACED BY THE PACK LOADER METHOD !!! * !!! TO BE REPLACED BY THE PACK LOADER METHOD !!! * !!! TO BE REPLACED BY THE PACK LOADER METHOD !!! * * Called by the Scene Manager if you specify a files payload for a pre-Scene Boot. * Takes an array of file objects. * * @method Phaser.Loader.LoaderPlugin#loadArray * @since 3.0.0 * * @param {LoaderFileObject[]} files - An array of files to load. * * @return {boolean} `true` if any files were successfully added to the list, otherwise `false`. */ loadArray: function (files) { if (Array.isArray(files)) { for (var i = 0; i < files.length; i++) { var file = files[i]; // Calls file-type methods like `atlas` or `image` this[file.type](file); } } return (this.list.size > 0); }, /** * The Scene that owns this plugin is shutting down. * We need to kill and reset all internal properties as well as stop listening to Scene events. * * @method Phaser.Loader.LoaderPlugin#shutdown * @private * @since 3.0.0 */ shutdown: function () { this.reset(); this.state = CONST.LOADER_SHUTDOWN; this.systems.events.off('shutdown', this.shutdown, this); }, /** * The Scene that owns this plugin is being destroyed. * We need to shutdown and then kill off all external references. * * @method Phaser.Loader.LoaderPlugin#destroy * @private * @since 3.0.0 */ destroy: function () { this.shutdown(); this.state = CONST.LOADER_DESTROYED; this.systems.events.off('start', this.pluginStart, this); this.list = null; this.inflight = null; this.queue = null; this.scene = null; this.systems = null; this.textureManager = null; this.cacheManager = null; } }); PluginManager.register('Loader', LoaderPlugin, 'load'); module.exports = LoaderPlugin;