2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
var CONST = require('./const');
|
2016-11-30 17:16:45 +00:00
|
|
|
var Set = require('../structs/Set');
|
|
|
|
var XHRSettings = require('./XHRSettings');
|
2016-12-06 15:15:42 +00:00
|
|
|
var Event = require('./events/');
|
|
|
|
var EventDispatcher = require('../events/EventDispatcher');
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
var BaseLoader = function ()
|
|
|
|
{
|
2016-12-05 15:29:53 +00:00
|
|
|
// To finish the loader ...
|
|
|
|
//
|
|
|
|
// 3) Progress update
|
|
|
|
// 4) JSON loader
|
|
|
|
// 5) XML Loader
|
|
|
|
// 6) Multi File support (atlas + data)
|
|
|
|
// 7) Atlas Loader
|
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
this.events = new EventDispatcher();
|
2016-12-05 15:29:53 +00:00
|
|
|
|
2016-11-30 17:16:45 +00:00
|
|
|
// Move to a 'setURL' method?
|
|
|
|
this.baseURL = '';
|
|
|
|
this.path = '';
|
|
|
|
|
|
|
|
// Read from Game / State Config
|
|
|
|
this.enableParallel = true;
|
2016-12-08 16:21:16 +00:00
|
|
|
this.maxParallelDownloads = 4;
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
// xhr specific global settings (can be overridden on a per-file basis)
|
|
|
|
this.xhr = XHRSettings();
|
|
|
|
|
|
|
|
this.crossOrigin = undefined;
|
|
|
|
|
|
|
|
this.list = new Set();
|
|
|
|
this.inflight = new Set();
|
|
|
|
this.failed = new Set();
|
|
|
|
this.queue = new Set();
|
|
|
|
this.storage = new Set();
|
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
this._state = CONST.LOADER_IDLE;
|
2016-11-30 17:16:45 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
BaseLoader.prototype.contructor = BaseLoader;
|
|
|
|
|
|
|
|
BaseLoader.prototype = {
|
|
|
|
|
|
|
|
addFile: function (file)
|
|
|
|
{
|
|
|
|
if (!this.isReady())
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Multipart file?
|
2016-12-07 04:43:02 +00:00
|
|
|
if (file.linkFile)
|
2016-11-30 17:16:45 +00:00
|
|
|
{
|
2016-12-07 04:43:02 +00:00
|
|
|
var fileA = file;
|
|
|
|
var fileB = file.linkFile;
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-07 04:43:02 +00:00
|
|
|
fileA.path = this.path;
|
|
|
|
fileB.path = this.path;
|
|
|
|
|
|
|
|
this.list.add(fileA);
|
|
|
|
this.list.add(fileB);
|
2016-11-30 17:16:45 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
file.path = this.path;
|
|
|
|
|
|
|
|
this.list.add(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
// Is the Loader actively loading (or processing loaded files)
|
|
|
|
isLoading: function ()
|
|
|
|
{
|
2016-12-06 15:15:42 +00:00
|
|
|
return (this._state === CONST.LOADER_LOADING || this._state === CONST.LOADER_PROCESSING);
|
2016-11-30 17:16:45 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
// Is the Loader ready to start a new load?
|
|
|
|
isReady: function ()
|
|
|
|
{
|
2016-12-06 15:15:42 +00:00
|
|
|
return (this._state === CONST.LOADER_IDLE || this._state === CONST.LOADER_COMPLETE || this._state === CONST.LOADER_FAILED);
|
2016-11-30 17:16:45 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
start: function ()
|
|
|
|
{
|
2016-12-08 16:21:16 +00:00
|
|
|
console.log('BaseLoader start. Files to load:', this.list.size);
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
if (!this.isReady())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
this.events.dispatch(new Event.LOADER_START_EVENT(this));
|
2016-12-05 17:19:12 +00:00
|
|
|
|
2016-11-30 17:16:45 +00:00
|
|
|
if (this.list.size === 0)
|
|
|
|
{
|
|
|
|
this.finishedLoading();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-12-06 15:15:42 +00:00
|
|
|
this._state = CONST.LOADER_LOADING;
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
this.failed.clear();
|
|
|
|
this.inflight.clear();
|
|
|
|
this.queue.clear();
|
|
|
|
|
2016-12-08 16:21:16 +00:00
|
|
|
this.queue.debug = true;
|
|
|
|
|
2016-11-30 17:16:45 +00:00
|
|
|
this.updateProgress();
|
|
|
|
|
|
|
|
this.processLoadQueue();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
updateProgress: function ()
|
|
|
|
{
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
processLoadQueue: function ()
|
|
|
|
{
|
2016-12-08 16:21:16 +00:00
|
|
|
// console.log('======== BaseLoader processLoadQueue');
|
|
|
|
// console.log('List size', this.list.size);
|
|
|
|
// console.log(this.inflight.size, 'items still in flight. Can load another', (this.maxParallelDownloads - this.inflight.size));
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
var _this = this;
|
|
|
|
|
|
|
|
this.list.each(function (file)
|
|
|
|
{
|
2016-12-06 15:15:42 +00:00
|
|
|
if (file.state === CONST.FILE_PENDING && _this.inflight.size < _this.maxParallelDownloads)
|
2016-11-30 17:16:45 +00:00
|
|
|
{
|
|
|
|
_this.inflight.add(file);
|
|
|
|
|
|
|
|
_this.list.delete(file);
|
|
|
|
|
|
|
|
_this.loadFile(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_this.inflight.size === _this.maxParallelDownloads)
|
|
|
|
{
|
|
|
|
// Tells the Set iterator to abort
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
// private
|
|
|
|
loadFile: function (file)
|
|
|
|
{
|
2016-12-06 15:25:24 +00:00
|
|
|
// console.log('LOADING', file.key);
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2016-12-01 12:50:58 +00:00
|
|
|
file.load(this.nextFile.bind(this), this.baseURL);
|
2016-11-30 17:16:45 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
nextFile: function (previousFile, success)
|
|
|
|
{
|
2016-12-06 15:25:24 +00:00
|
|
|
// console.log('LOADED:', previousFile.src, success);
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
// Move the file that just loaded from the inflight list to the queue or failed Set
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
this.queue.add(previousFile);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.failed.add(previousFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.inflight.delete(previousFile);
|
|
|
|
|
|
|
|
if (this.list.size > 0)
|
|
|
|
{
|
2016-12-06 15:25:24 +00:00
|
|
|
// console.log('nextFile - still something in the list');
|
2016-11-30 17:16:45 +00:00
|
|
|
this.processLoadQueue();
|
|
|
|
}
|
|
|
|
else if (this.inflight.size === 0)
|
|
|
|
{
|
2016-12-06 15:25:24 +00:00
|
|
|
// console.log('nextFile calling finishedLoading');
|
2016-11-30 17:16:45 +00:00
|
|
|
this.finishedLoading();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
finishedLoading: function ()
|
|
|
|
{
|
2016-12-08 16:21:16 +00:00
|
|
|
// console.log('---> BaseLoader.finishedLoading PROCESSING', this.queue.size, 'files');
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
this._state = CONST.LOADER_PROCESSING;
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-08 16:21:16 +00:00
|
|
|
this.storage.clear();
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-07 00:27:56 +00:00
|
|
|
var _this = this;
|
2016-11-30 17:16:45 +00:00
|
|
|
|
|
|
|
this.queue.each(function (file)
|
|
|
|
{
|
2016-12-08 16:21:16 +00:00
|
|
|
// console.log('%c Calling process on ' + file.key, 'color: #000000; background: #ffff00;');
|
|
|
|
|
2016-12-07 00:27:56 +00:00
|
|
|
file.onProcess(_this.processUpdate.bind(_this));
|
|
|
|
});
|
|
|
|
},
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-07 04:43:02 +00:00
|
|
|
// Called automatically by the File when it has finished processing
|
2016-12-07 00:27:56 +00:00
|
|
|
processUpdate: function (file)
|
|
|
|
{
|
2016-12-08 16:21:16 +00:00
|
|
|
// console.log('-> processUpdate', file.key, file.state);
|
|
|
|
|
2016-12-07 04:43:02 +00:00
|
|
|
// This file has failed to load, so move it to the failed Set
|
2016-12-07 01:13:17 +00:00
|
|
|
if (file.state === CONST.FILE_ERRORED)
|
|
|
|
{
|
|
|
|
this.failed.add(file);
|
2016-12-07 04:43:02 +00:00
|
|
|
|
|
|
|
if (file.linkFile)
|
|
|
|
{
|
|
|
|
this.queue.delete(file.linkFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.removeFromQueue(file);
|
2016-12-07 01:13:17 +00:00
|
|
|
}
|
2016-12-07 04:43:02 +00:00
|
|
|
|
|
|
|
// If we got here, then the file loaded
|
|
|
|
|
|
|
|
// Special handling for multi-part files
|
|
|
|
|
|
|
|
if (file.linkFile)
|
2016-12-07 01:13:17 +00:00
|
|
|
{
|
2016-12-07 04:43:02 +00:00
|
|
|
if (file.state === CONST.FILE_COMPLETE && file.linkFile.state === CONST.FILE_COMPLETE)
|
|
|
|
{
|
|
|
|
// Partner has loaded, so add them both to Storage
|
2016-12-07 10:50:10 +00:00
|
|
|
|
|
|
|
this.storage.add({ type: file.linkType, fileA: file, fileB: file.linkFile });
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-07 04:43:02 +00:00
|
|
|
this.queue.delete(file.linkFile);
|
2016-12-07 01:13:17 +00:00
|
|
|
|
2016-12-07 04:43:02 +00:00
|
|
|
this.removeFromQueue(file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2016-12-07 00:27:56 +00:00
|
|
|
{
|
2016-12-07 04:43:02 +00:00
|
|
|
this.storage.add(file);
|
2016-12-08 16:21:16 +00:00
|
|
|
|
2016-12-07 04:43:02 +00:00
|
|
|
this.removeFromQueue(file);
|
2016-12-07 00:27:56 +00:00
|
|
|
}
|
|
|
|
},
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-08 16:21:16 +00:00
|
|
|
removeFromQueue: function (file)
|
|
|
|
{
|
|
|
|
this.queue.delete(file);
|
|
|
|
|
|
|
|
if (this.queue.size === 0 && this._state === CONST.LOADER_PROCESSING)
|
|
|
|
{
|
|
|
|
// We've processed all the files we loaded
|
|
|
|
this.processComplete();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-12-07 00:27:56 +00:00
|
|
|
processComplete: function ()
|
|
|
|
{
|
2016-12-08 16:21:16 +00:00
|
|
|
console.log('Loader Complete. Loaded:', this.storage.size, 'Failed:', this.failed.size);
|
|
|
|
|
2016-11-30 17:16:45 +00:00
|
|
|
this.list.clear();
|
|
|
|
this.inflight.clear();
|
|
|
|
this.queue.clear();
|
|
|
|
|
2016-12-07 01:13:17 +00:00
|
|
|
if (this.processCallback)
|
|
|
|
{
|
|
|
|
this.processCallback();
|
|
|
|
}
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
this._state = CONST.LOADER_COMPLETE;
|
2016-11-30 17:16:45 +00:00
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
this.events.dispatch(new Event.LOADER_COMPLETE_EVENT(this));
|
2016-11-30 17:16:45 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
reset: function ()
|
|
|
|
{
|
|
|
|
this.list.clear();
|
|
|
|
this.inflight.clear();
|
|
|
|
this.failed.clear();
|
|
|
|
this.queue.clear();
|
|
|
|
this.storage.clear();
|
|
|
|
|
|
|
|
this.tag = '';
|
|
|
|
this.path = '';
|
|
|
|
this.baseURL = '';
|
|
|
|
|
2016-12-06 15:15:42 +00:00
|
|
|
this._state = CONST.LOADER_IDLE;
|
2016-11-30 17:16:45 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
destroy: function ()
|
|
|
|
{
|
|
|
|
this.reset();
|
2016-12-06 15:15:42 +00:00
|
|
|
this._state = CONST.LOADER_DESTROYED;
|
2016-11-30 17:16:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = BaseLoader;
|