phaser/src/loader/File.js

450 lines
12 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
* @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');
2018-01-26 14:23:00 +00:00
var GetFastValue = require('../utils/object/GetFastValue');
var GetURL = require('./GetURL');
var MergeXHRSettings = require('./MergeXHRSettings');
var XHRLoader = require('./XHRLoader');
var XHRSettings = require('./XHRSettings');
2018-03-19 21:57:46 +00:00
/**
* @callback FileProcessCallback
*
* @param {Phaser.Loader.File} file - [description]
*/
2018-02-07 15:27:21 +00:00
/**
* @classdesc
* [description]
*
* @class File
* @memberOf Phaser.Loader
* @constructor
* @since 3.0.0
*
* @param {object} fileConfig - [description]
*/
var File = new Class({
initialize:
function File (fileConfig)
{
2018-01-26 14:23:00 +00:00
/**
* The file type string (image, json, etc) for sorting within the Loader.
2018-03-19 21:57:46 +00:00
*
* @name Phaser.Loader.File#type
* @type {string}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.type = GetFastValue(fileConfig, 'type', false);
2018-01-26 14:23:00 +00:00
/**
* Unique cache key (unique within its file type)
*
* @name Phaser.Loader.File#key
* @type {string}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.key = GetFastValue(fileConfig, 'key', false);
if (!this.type || !this.key)
{
throw new Error('Error calling \'Loader.' + this.type + '\' invalid key provided.');
}
2018-01-26 14:23:00 +00:00
/**
* The URL of the file, not including baseURL.
*
* @name Phaser.Loader.File#url
* @type {string}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.url = GetFastValue(fileConfig, 'url');
if (this.url === undefined)
{
this.url = GetFastValue(fileConfig, 'path', '') + this.key + '.' + GetFastValue(fileConfig, 'extension', '');
}
2018-02-15 17:11:55 +00:00
else if (typeof(this.url) !== 'function')
{
this.url = GetFastValue(fileConfig, 'path', '').concat(this.url);
}
2018-01-26 14:23:00 +00:00
/**
* Set when the Loader calls 'load' on this file.
*
* @name Phaser.Loader.File#src
* @type {string}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.src = '';
2018-01-26 14:23:00 +00:00
/**
* The merged XHRSettings for this file.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#xhrSettings
* @type {Phaser.Loader.XHRSettings}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.xhrSettings = XHRSettings(GetFastValue(fileConfig, 'responseType', undefined));
if (GetFastValue(fileConfig, 'xhrSettings', false))
{
this.xhrSettings = MergeXHRSettings(this.xhrSettings, GetFastValue(fileConfig, 'xhrSettings', {}));
}
2018-01-26 14:23:00 +00:00
/**
* The LoaderPlugin instance that is loading this file.
*
* @name Phaser.Loader.File#loader
* @type {?Phaser.Loader.LoaderPlugin}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.loader = null;
2018-01-26 14:23:00 +00:00
/**
* The XHR Loader instance that is loading this File.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#xhrLoader
* @type {?Phaser.Loader.XHRLoader}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.xhrLoader = null;
2018-01-26 14:23:00 +00:00
/**
* The current state of the file. One of the FILE_CONST values.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#state
* @type {integer}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
2018-03-05 01:40:11 +00:00
this.state = typeof(this.url) === 'function' ? CONST.FILE_POPULATED : CONST.FILE_PENDING;
2018-01-26 14:23:00 +00:00
/**
* The total size of this file.
* Set by onProgress and only if loading via XHR.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#bytesTotal
* @type {number}
2018-01-26 14:23:00 +00:00
* @default 0
* @since 3.0.0
*/
this.bytesTotal = 0;
2018-01-26 14:23:00 +00:00
/**
* Updated as the file loads.
* Only set if loading via XHR.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#bytesLoaded
* @type {number}
2018-01-26 14:23:00 +00:00
* @default -1
* @since 3.0.0
*/
this.bytesLoaded = -1;
2018-01-26 14:23:00 +00:00
/**
* A percentage value between 0 and 1 indicating how much of this file has loaded.
* Only set if loading via XHR.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#percentComplete
* @type {float}
2018-01-26 14:23:00 +00:00
* @default -1
* @since 3.0.0
*/
this.percentComplete = -1;
2018-01-26 14:23:00 +00:00
/**
* For CORs based loading.
* If this is undefined then the File will check BaseLoader.crossOrigin and use that (if set)
*
* @name Phaser.Loader.File#crossOrigin
2018-03-20 15:12:42 +00:00
* @type {(string|undefined)}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.crossOrigin = undefined;
2018-01-26 14:23:00 +00:00
/**
* The processed file data, stored in here after the file has loaded.
*
* @name Phaser.Loader.File#data
2018-03-20 16:15:49 +00:00
* @type {*}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.data = undefined;
2018-01-26 14:23:00 +00:00
/**
* A config object that can be used by file types to store transitional data.
*
* @name Phaser.Loader.File#config
* @type {object}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.config = GetFastValue(fileConfig, 'config', {});
2018-01-26 14:23:00 +00:00
/**
* If this is a multipart file, i.e. an atlas and its json together, then this is a reference
* to the linked file. Set and used internally by the Loader.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#linkFile
* @type {?Phaser.Loader.File}
2018-01-26 14:23:00 +00:00
* @since 3.0.0
*/
this.linkFile = undefined;
2018-01-26 14:23:00 +00:00
/**
* If this is a multipart file, i.e. an atlas and its json together, then this is a reference
* to the type of linked association. Set and used internally by the Loader.
2018-01-26 14:23:00 +00:00
*
* @name Phaser.Loader.File#linkType
* @type {string}
2018-01-26 14:23:00 +00:00
* @default ''
* @since 3.0.0
*/
this.linkType = '';
/**
* If this is a link file, is this the parent or the sibbling?
*
* @name Phaser.Loader.File#linkParent
* @type {boolean}
* @default false
* @since 3.0.0
*/
this.linkParent = false;
},
/**
* If this is a multipart file, i.e. an atlas and its json together, then this is a reference
* to the linked file. Set and used internally by the Loader.
*
* @method Phaser.Loader.File#setLinkFile
* @since 3.0.0
2018-03-19 21:57:46 +00:00
*
* @param {Phaser.Loader.File} fileB - The linked file.
* @param {string} linkType - The type of association.
*/
setLinkFile: function (fileB, linkType)
{
this.linkFile = fileB;
fileB.linkFile = this;
this.linkType = linkType;
fileB.linkType = linkType;
this.linkParent = true;
},
2018-01-26 14:23:00 +00:00
/**
* Resets the XHRLoader instance.
2018-01-26 14:23:00 +00:00
*
* @method Phaser.Loader.File#resetXHR
* @since 3.0.0
*/
resetXHR: function ()
{
if (this.xhrLoader)
{
this.xhrLoader.onload = undefined;
this.xhrLoader.onerror = undefined;
this.xhrLoader.onprogress = undefined;
}
},
2018-01-26 14:23:00 +00:00
/**
* Called by the Loader, starts the actual file downloading.
* During the load the methods onLoad, onProgress, etc are called based on the XHR events.
*
* @method Phaser.Loader.File#load
* @since 3.0.0
*
* @param {Phaser.Loader.LoaderPlugin} loader - The Loader that will load this File.
2018-01-26 14:23:00 +00:00
*/
load: function (loader)
{
this.loader = loader;
if (this.state === CONST.FILE_POPULATED)
{
this.onComplete();
loader.nextFile(this);
}
else
{
this.src = GetURL(this, loader.baseURL);
if (this.src.indexOf('data:') === 0)
{
console.warn('Local data URIs are not supported: ' + this.key);
}
else
{
this.xhrLoader = XHRLoader(this, loader.xhr);
}
}
},
2018-01-26 14:23:00 +00:00
/**
* Called when the file finishes loading, is sent a DOM ProgressEvent.
2018-01-26 14:23:00 +00:00
*
* @method Phaser.Loader.File#onLoad
* @since 3.0.0
*
* @param {ProgressEvent} event - The DOM ProgressEvent that resulted from this load.
2018-01-26 14:23:00 +00:00
*/
onLoad: function (event)
{
this.resetXHR();
if (event.target && event.target.status !== 200)
{
this.loader.nextFile(this, false);
}
else
{
this.loader.nextFile(this, true);
}
},
2018-01-26 14:23:00 +00:00
/**
* Called if the file errors while loading, is sent a DOM ProgressEvent.
2018-01-26 14:23:00 +00:00
*
* @method Phaser.Loader.File#onError
* @since 3.0.0
*
* @param {ProgressEvent} event - The DOM ProgressEvent that resulted from this error.
2018-01-26 14:23:00 +00:00
*/
2018-02-16 19:08:50 +00:00
onError: function ()
{
this.resetXHR();
this.loader.nextFile(this, false);
},
2018-01-26 14:23:00 +00:00
/**
* Called during the file load progress. Is sent a DOM ProgressEvent.
2018-01-26 14:23:00 +00:00
*
* @method Phaser.Loader.File#onProgress
* @since 3.0.0
*
* @param {ProgressEvent} event - The DOM ProgressEvent.
2018-01-26 14:23:00 +00:00
*/
onProgress: function (event)
{
if (event.lengthComputable)
{
this.bytesLoaded = event.loaded;
this.bytesTotal = event.total;
this.percentComplete = Math.min((this.bytesLoaded / this.bytesTotal), 1);
// console.log(this.percentComplete + '% (' + this.bytesLoaded + ' bytes)');
this.loader.emit('fileprogress', this, this.percentComplete);
}
},
2018-01-26 14:23:00 +00:00
/**
* Usually overridden by the FileTypes and is called by Loader.finishedLoading.
2018-01-26 14:23:00 +00:00
* The callback is Loader.processUpdate
*
* @method Phaser.Loader.File#onProcess
* @since 3.0.0
*
2018-03-19 21:57:46 +00:00
* @param {FileProcessCallback} callback - The callback to invoke to process this File.
2018-01-26 14:23:00 +00:00
*/
2016-12-07 00:27:56 +00:00
onProcess: function (callback)
{
2016-12-07 00:27:56 +00:00
this.state = CONST.FILE_PROCESSING;
this.onComplete();
callback(this);
},
2018-01-26 14:23:00 +00:00
/**
* Called with the File has completed loading.
* Checks on the state of its linkfile, if set.
2018-01-26 14:23:00 +00:00
*
* @method Phaser.Loader.File#onComplete
* @since 3.0.0
*/
onComplete: function ()
{
if (this.linkFile)
{
if (this.linkFile.state === CONST.FILE_WAITING_LINKFILE)
{
// The linkfile has finished processing, and is waiting for this file, so let's do them both
this.state = CONST.FILE_COMPLETE;
this.linkFile.state = CONST.FILE_COMPLETE;
}
else
{
// The linkfile still hasn't finished loading and/or processing yet
this.state = CONST.FILE_WAITING_LINKFILE;
}
}
else
{
this.state = CONST.FILE_COMPLETE;
}
}
});
/**
* Static method for creating object URL using URL API and setting it as image 'src' attribute.
* If URL API is not supported (usually on old browsers) it falls back to creating Base64 encoded url using FileReader.
*
* @method Phaser.Loader.File.createObjectURL
* @static
* @param {Image} image - Image object which 'src' attribute should be set to object URL.
* @param {Blob} blob - A Blob object to create an object URL for.
* @param {string} defaultType - Default mime type used if blob type is not available.
*/
File.createObjectURL = function (image, blob, defaultType)
{
if (typeof URL === 'function')
{
image.src = URL.createObjectURL(blob);
}
else
{
var reader = new FileReader();
reader.onload = function ()
{
image.removeAttribute('crossOrigin');
image.src = 'data:' + (blob.type || defaultType) + ';base64,' + reader.result.split(',')[1];
};
reader.onerror = image.onerror;
reader.readAsDataURL(blob);
}
};
/**
* Static method for releasing an existing object URL which was previously created
* by calling {@link File#createObjectURL} method.
*
* @method Phaser.Loader.File.revokeObjectURL
* @static
* @param {Image} image - Image object which 'src' attribute should be revoked.
*/
File.revokeObjectURL = function (image)
{
if (typeof URL === 'function')
{
URL.revokeObjectURL(image.src);
}
};
module.exports = File;