phaser/v3/src/state/GlobalStateManager.js

620 lines
15 KiB
JavaScript
Raw Normal View History

2016-11-29 13:01:16 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
* @copyright 2016 Photon Storm Ltd.
* @license {@link https://github.com/photonstorm/phaser/blob/master/license.txt|MIT License}
*/
var CONST = require('../const');
var NOOP = require('../utils/NOOP');
var State = require('./State');
var Systems = require('./Systems');
var GetValue = require('../utils/object/GetValue');
var EventDispatcher = require('../events/EventDispatcher');
var Rectangle = require('../geom/rectangle/Rectangle');
var CanvasPool = require('../dom/CanvasPool');
var CanvasInterpolation = require('../dom/CanvasInterpolation');
var GetContext = require('../canvas/GetContext');
2016-11-29 13:01:16 +00:00
/**
* The State Manager is responsible for loading, setting up and switching game states.
*
* @class Phaser.GlobalStateManager
2016-11-29 13:01:16 +00:00
* @constructor
* @param {Phaser.Game} game - A reference to the currently running game.
*/
var GlobalStateManager = function (game, stateConfig)
2016-11-29 13:01:16 +00:00
{
this.game = game;
// Everything kept in here
this.keys = {};
this.states = [];
// Only active states are kept in here
this.active = [];
this._pending = [];
if (stateConfig)
{
if (Array.isArray(stateConfig))
{
for (var i = 0; i < stateConfig.length; i++)
{
// The i === 0 part just starts the first State given
this._pending.push({
index: i,
key: 'default',
state: stateConfig[i],
2017-02-16 17:18:50 +00:00
autoStart: (i === 0),
data: {}
});
2016-11-29 13:01:16 +00:00
}
}
else
{
this._pending.push({
index: 0,
key: 'default',
state: stateConfig,
2017-02-16 17:18:50 +00:00
autoStart: true,
data: {}
});
2016-11-29 13:01:16 +00:00
}
}
};
GlobalStateManager.prototype.constructor = GlobalStateManager;
2016-11-29 13:01:16 +00:00
GlobalStateManager.prototype = {
2016-11-29 13:01:16 +00:00
/**
* The Boot handler is called by Phaser.Game when it first starts up.
* The renderer is available by now.
*
* @method Phaser.GlobalStateManager#boot
2016-11-29 13:01:16 +00:00
* @private
*/
boot: function ()
{
for (var i = 0; i < this._pending.length; i++)
{
var entry = this._pending[i];
this.add(entry.key, entry.state, entry.autoStart);
}
// Clear the pending list
this._pending = [];
},
2017-02-16 17:18:50 +00:00
// private
2016-11-29 13:01:16 +00:00
getKey: function (key, stateConfig)
{
if (!key) { key = 'default'; }
if (typeof stateConfig === 'function')
{
return key;
}
else if (stateConfig instanceof State)
2016-11-29 13:01:16 +00:00
{
key = stateConfig.sys.settings.key;
2016-11-29 13:01:16 +00:00
}
else if (typeof stateConfig === 'object' && stateConfig.hasOwnProperty('key'))
{
key = stateConfig.key;
}
// By this point it's either 'default' or extracted from the State
if (this.keys.hasOwnProperty(key))
{
throw new Error('Cannot add a State with duplicate key: ' + key);
}
else
{
return key;
}
},
/**
* Adds a new State into the GlobalStateManager. You must give each State a unique key by which you'll identify it.
2016-11-29 13:01:16 +00:00
* The State can be either a Phaser.State object (or an object that extends it), a plain JavaScript object or a function.
* If a function is given a new state object will be created by calling it.
*
* @method Phaser.GlobalStateManager#add
2016-11-29 13:01:16 +00:00
* @param {string} key - A unique key you use to reference this state, i.e. "MainMenu", "Level1".
* @param {Phaser.State|object|function} state - The state you want to switch to.
* @param {boolean} [autoStart=false] - If true the State will be started immediately after adding it.
*/
add: function (key, stateConfig, autoStart)
{
if (autoStart === undefined) { autoStart = false; }
// if not booted, then put state into a holding pattern
if (!this.game.isBooted)
{
this._pending.push({
index: this._pending.length,
key: key,
state: stateConfig,
autoStart: autoStart
});
// console.log('GlobalStateManager not yet booted, adding to list', this._pending.length);
2016-11-29 13:01:16 +00:00
return;
}
key = this.getKey(key, stateConfig);
// console.log('GlobalStateManager.add', key, stateConfig, autoStart);
2016-11-29 13:01:16 +00:00
var newState;
if (stateConfig instanceof State)
2016-11-29 13:01:16 +00:00
{
// console.log('GlobalStateManager.add from instance:', key);
2016-11-29 13:01:16 +00:00
newState = this.createStateFromInstance(key, stateConfig);
}
else if (typeof stateConfig === 'object')
{
// console.log('GlobalStateManager.add from object:', key);
2016-11-29 13:01:16 +00:00
stateConfig.key = key;
newState = this.createStateFromObject(key, stateConfig);
}
else if (typeof stateConfig === 'function')
{
// console.log('GlobalStateManager.add from function:', key);
2016-11-29 13:01:16 +00:00
newState = this.createStateFromFunction(key, stateConfig);
}
// Replace key incase the state changed it
key = newState.sys.settings.key;
2016-11-29 13:01:16 +00:00
this.keys[key] = newState;
this.states.push(newState);
if (autoStart || newState.sys.settings.active)
2016-11-29 13:01:16 +00:00
{
if (this.game.isBooted)
{
this.start(key);
}
else
{
this._start.push(key);
}
}
return newState;
},
createStateFromInstance: function (key, newState)
{
newState.sys.settings.key = key;
2016-11-29 13:01:16 +00:00
newState.sys.init(this.game);
2016-11-29 13:01:16 +00:00
this.createStateDisplay(newState);
2016-11-29 13:01:16 +00:00
return newState;
},
createStateFromObject: function (key, stateConfig)
{
var newState = new State(stateConfig);
2016-11-29 13:01:16 +00:00
newState.sys.init(this.game);
2016-11-29 13:01:16 +00:00
this.createStateDisplay(newState);
2016-11-29 13:01:16 +00:00
return this.setupCallbacks(newState, stateConfig);
2016-11-29 13:01:16 +00:00
},
createStateFromFunction: function (key, state)
{
var newState = new state();
if (newState instanceof State)
2016-11-29 13:01:16 +00:00
{
key = newState.sys.settings.key;
if (this.keys.hasOwnProperty(key))
{
throw new Error('Cannot add a State with duplicate key: ' + key);
}
2016-11-29 13:01:16 +00:00
return this.createStateFromInstance(key, newState);
}
else
{
newState.sys = new Systems(newState);
2016-11-29 13:01:16 +00:00
newState.sys.init(this.game);
2016-11-29 13:01:16 +00:00
this.createStateDisplay(newState);
2016-11-29 13:01:16 +00:00
// Default required functions
if (!newState.init)
{
newState.init = NOOP;
}
if (!newState.preload)
{
newState.preload = NOOP;
}
if (!newState.create)
{
newState.create = NOOP;
}
if (!newState.shutdown)
{
newState.shutdown = NOOP;
}
if (!newState.update)
{
newState.update = NOOP;
}
if (!newState.render)
{
newState.render = NOOP;
}
return newState;
}
},
2016-11-29 13:01:16 +00:00
setupCallbacks: function (state, stateConfig)
{
if (stateConfig === undefined) { stateConfig = state; }
2016-11-29 13:01:16 +00:00
// Extract callbacks or set NOOP
2016-11-29 13:01:16 +00:00
state.init = GetValue(stateConfig, 'init', NOOP);
state.preload = GetValue(stateConfig, 'preload', NOOP);
state.create = GetValue(stateConfig, 'create', NOOP);
state.shutdown = GetValue(stateConfig, 'shutdown', NOOP);
2016-11-29 13:01:16 +00:00
// Game Loop level callbacks
state.update = GetValue(stateConfig, 'update', NOOP);
state.render = GetValue(stateConfig, 'render', NOOP);
2016-11-29 13:01:16 +00:00
return state;
2016-11-29 13:01:16 +00:00
},
createStateDisplay: function (state)
2016-11-29 13:01:16 +00:00
{
// console.log('createStateDisplay', state.sys.settings.key);
var settings = state.sys.settings;
// var x = settings.x;
// var y = settings.y;
var width = settings.width;
var height = settings.height;
var config = this.game.config;
2016-11-29 13:01:16 +00:00
if (config.renderType === CONST.CANVAS)
2016-11-29 13:01:16 +00:00
{
if (settings.renderToTexture)
{
2017-02-07 22:00:55 +00:00
// console.log('renderToTexture', width, height);
state.sys.canvas = CanvasPool.create(state, width, height);
state.sys.context = GetContext(state.sys.canvas);
2016-11-29 13:01:16 +00:00
// Pixel Art mode?
if (config.pixelArt)
{
CanvasInterpolation.setCrisp(state.sys.canvas);
}
}
else
{
2017-02-07 22:00:55 +00:00
// console.log('using game canvas');
state.sys.mask = new Rectangle(0, 0, width, height);
state.sys.canvas = this.game.canvas;
state.sys.context = this.game.context;
}
2016-11-29 13:01:16 +00:00
}
else if (config.renderType === CONST.WEBGL)
{
// state.sys.fbo = this.game.renderer.createFBO(state, x, y, width, height);
}
2016-11-29 13:01:16 +00:00
},
getState: function (key)
{
return this.keys[key];
},
getStateIndex: function (state)
{
return this.states.indexOf(state);
},
getActiveState: function (key)
{
var state = this.getState(key);
for (var i = 0; i < this.active.length; i++)
{
if (this.active[i].state === state)
{
return this.active[i];
}
}
},
2016-11-29 13:01:16 +00:00
getActiveStateIndex: function (state)
{
2017-02-07 20:47:41 +00:00
var index = -1;
2016-11-29 13:01:16 +00:00
for (var i = 0; i < this.active.length; i++)
{
if (this.active[i].state === state)
{
2017-02-07 20:47:41 +00:00
index = this.active[i].index;
2016-11-29 13:01:16 +00:00
}
}
2017-02-07 20:47:41 +00:00
return index;
2016-11-29 13:01:16 +00:00
},
isActive: function (key)
{
var state = this.getState(key);
return (state && state.sys.settings.active && this.active.indexOf(state) !== -1);
2016-11-29 13:01:16 +00:00
},
2017-02-16 17:18:50 +00:00
start: function (key, data)
2016-11-29 13:01:16 +00:00
{
2017-02-16 17:18:50 +00:00
if (data === undefined) { data = {}; }
console.log('start:', key);
2017-02-17 02:07:56 +00:00
// console.dir(data);
2016-11-29 13:01:16 +00:00
// if not booted, then put state into a holding pattern
if (!this.game.isBooted)
{
// console.log('GlobalStateManager not yet booted, setting autoStart on pending list');
2016-11-29 13:01:16 +00:00
for (var i = 0; i < this._pending.length; i++)
{
var entry = this._pending[i];
if (entry.key === key)
{
entry.autoStart = true;
2017-02-16 17:18:50 +00:00
entry.data = data;
2016-11-29 13:01:16 +00:00
}
}
return;
}
var state = this.getState(key);
// console.log(state);
2016-11-29 13:01:16 +00:00
if (state)
{
// Already started? Nothing more to do here ...
if (this.isActive(key))
{
return;
}
state.sys.settings.active = true;
2016-11-29 13:01:16 +00:00
state.sys.settings.data = data;
2017-02-17 02:07:56 +00:00
var loader = state.sys.load;
2016-11-29 13:01:16 +00:00
// Files payload?
if (loader && Array.isArray(state.sys.settings.files))
2016-11-29 13:01:16 +00:00
{
loader.reset();
2016-11-29 13:01:16 +00:00
if (loader.loadArray(state.sys.settings.files))
2016-11-29 13:01:16 +00:00
{
2017-02-17 02:07:56 +00:00
loader.events.once('LOADER_COMPLETE_EVENT', this.payloadComplete.bind(this));
loader.start();
2016-11-29 13:01:16 +00:00
}
else
{
2017-02-17 02:07:56 +00:00
this.bootState(state);
2016-11-29 13:01:16 +00:00
}
}
else
{
2017-02-17 02:07:56 +00:00
this.bootState(state);
}
}
},
2017-02-17 02:07:56 +00:00
payloadComplete: function (event)
{
var state = event.loader.state;
// console.log('payloadComplete', state.sys.settings.key);
2017-02-17 02:07:56 +00:00
this.bootState(state);
},
2017-02-17 02:07:56 +00:00
bootState: function (state)
{
2017-02-17 02:07:56 +00:00
// console.log('bootState', state.sys.settings.key);
if (state.init)
{
2017-02-17 02:07:56 +00:00
state.init.call(state, state.sys.settings.data);
}
var loader = state.sys.load;
loader.reset();
if (state.preload)
{
2017-02-07 20:47:41 +00:00
state.preload(this.game);
// Is the loader empty?
if (loader.list.size === 0)
{
2017-02-17 02:07:56 +00:00
this.create(state);
2016-11-29 13:01:16 +00:00
}
else
{
// Start the loader going as we have something in the queue
2017-02-17 02:07:56 +00:00
loader.events.once('LOADER_COMPLETE_EVENT', this.loadComplete.bind(this));
loader.start();
}
}
else
{
// No preload? Then there was nothing to load either
2017-02-17 02:07:56 +00:00
this.create(state);
2016-11-29 13:01:16 +00:00
}
},
2017-02-17 02:07:56 +00:00
loadComplete: function (event)
2016-11-29 13:01:16 +00:00
{
var state = event.loader.state;
// console.log('loadComplete', state.sys.settings.key);
2017-02-17 02:07:56 +00:00
this.create(state);
2016-11-29 13:01:16 +00:00
},
2017-02-17 02:07:56 +00:00
create: function (state)
2016-11-29 13:01:16 +00:00
{
2017-02-17 02:07:56 +00:00
// console.log('create', state.sys.settings.key);
// console.log(state);
2016-11-29 13:01:16 +00:00
// Insert at the correct index, or it just all goes wrong :)
var i = this.getStateIndex(state);
// console.log('create.index', state.sys.settings.key, i);
2016-11-29 13:01:16 +00:00
this.active.push({ index: i, state: state });
// Sort the 'active' array based on the index property
2017-02-07 19:53:04 +00:00
this.active.sort(this.sortStates);
2016-11-29 13:01:16 +00:00
state.sys.updates.running = true;
if (state.create)
{
state.create.call(state, state.sys.settings.data);
}
2016-11-29 13:01:16 +00:00
},
pause: function (key)
{
var entry = this.getActiveState(key);
if (entry)
{
entry.state.sys.pause();
}
},
resume: function (key)
{
var entry = this.getActiveState(key);
if (entry)
{
entry.state.sys.resume();
}
},
sleep: function (key)
{
var entry = this.getActiveState(key);
if (entry)
{
entry.state.sys.sleep();
}
},
wake: function (key)
{
var entry = this.getActiveState(key);
if (entry)
{
entry.state.sys.wake();
}
},
swap: function (from, to)
{
this.sleep(from);
this.start(to);
},
stop: function (key)
{
var entry = this.getActiveState(key);
2016-11-29 13:01:16 +00:00
if (entry)
2016-11-29 13:01:16 +00:00
{
entry.state.sys.shutdown();
2016-11-29 13:01:16 +00:00
// Remove from the active list
var index = this.active.indexOf(entry);
2016-11-29 13:01:16 +00:00
if (index !== -1)
{
this.active.splice(index, 1);
2016-11-29 13:01:16 +00:00
this.active.sort(this.sortStates);
}
2016-11-29 13:01:16 +00:00
}
},
sortStates: function (stateA, stateB)
{
2017-02-07 19:53:04 +00:00
// console.log('sortStates', stateA.state.sys.settings.key, stateA.index, stateB.state.sys.settings.key, stateB.index);
2016-11-29 13:01:16 +00:00
// Sort descending
if (stateA.index < stateB.index)
{
return -1;
}
else if (stateA.index > stateB.index)
{
return 1;
}
else
{
return 0;
}
}
2016-11-29 13:01:16 +00:00
};
module.exports = GlobalStateManager;