phaser/v3/src/physics/matter-js/World.js

233 lines
6.8 KiB
JavaScript

// Phaser.Physics.Matter.World
var Bodies = require('./lib/factory/Bodies');
var Class = require('../../utils/Class');
var Engine = require('./lib/core/Engine');
var EventDispatcher = require('../../events/EventDispatcher');
var GetFastValue = require('../../utils/object/GetFastValue');
var GetValue = require('../../utils/object/GetValue');
var MatterBody = require('./lib/body/Body');
var MatterEvents = require('./lib/core/Events');
var MatterWorld = require('./lib/body/World');
var PhysicsEvent = require('./events/');
var World = new Class({
initialize:
function World (scene, config)
{
this.scene = scene;
this.engine = Engine.create(config);
this.localWorld = this.engine.world;
this.events = new EventDispatcher();
var gravity = GetValue(config, 'gravity', null);
if (gravity)
{
this.setGravity(gravity.x, gravity.y, gravity.scale);
}
/**
* @property {object} walls - An object containing the 4 wall bodies that bound the physics world.
*/
this.walls = { left: null, right: null, top: null, bottom: null };
if (GetFastValue(config, 'setBounds', false))
{
var boundsConfig = config['setBounds'];
if (typeof boundsConfig === 'boolean')
{
this.setBounds();
}
else
{
var x = GetFastValue(boundsConfig, 'x', 0);
var y = GetFastValue(boundsConfig, 'y', 0);
var width = GetFastValue(boundsConfig, 'width', scene.sys.game.config.width);
var height = GetFastValue(boundsConfig, 'height', scene.sys.game.config.height);
var thickness = GetFastValue(boundsConfig, 'thickness', 64);
var left = GetFastValue(boundsConfig, 'left', true);
var right = GetFastValue(boundsConfig, 'right', true);
var top = GetFastValue(boundsConfig, 'top', true);
var bottom = GetFastValue(boundsConfig, 'bottom', true);
this.setBounds(x, y, width, height, thickness, left, right, top, bottom);
}
}
this.setEventsProxy();
},
setEventsProxy: function ()
{
var localEvents = this.events;
MatterEvents.on(this.engine, 'beforeUpdate', function (event) {
localEvents.dispatch(new PhysicsEvent.BEFORE_UPDATE(event));
});
MatterEvents.on(this.engine, 'afterUpdate', function (event) {
localEvents.dispatch(new PhysicsEvent.AFTER_UPDATE(event));
});
MatterEvents.on(this.engine, 'collisionStart', function (event) {
localEvents.dispatch(new PhysicsEvent.COLLISION_START(event.pairs));
});
MatterEvents.on(this.engine, 'collisionActive', function (event) {
localEvents.dispatch(new PhysicsEvent.COLLISION_ACTIVE(event));
});
MatterEvents.on(this.engine, 'collisionEnd', function (event) {
localEvents.dispatch(new PhysicsEvent.COLLISION_END(event));
});
},
/**
* Sets the bounds of the Physics world to match the given world pixel dimensions.
* You can optionally set which 'walls' to create: left, right, top or bottom.
* If none of the walls are given it will default to use the walls settings it had previously.
* I.e. if you previously told it to not have the left or right walls, and you then adjust the world size
* the newly created bounds will also not have the left and right walls.
* Explicitly state them in the parameters to override this.
*
* @method Phaser.Physics.P2#setBounds
* @param {number} x - The x coordinate of the top-left corner of the bounds.
* @param {number} y - The y coordinate of the top-left corner of the bounds.
* @param {number} width - The width of the bounds.
* @param {number} height - The height of the bounds.
* @param {boolean} [left=true] - If true will create the left bounds wall.
* @param {boolean} [right=true] - If true will create the right bounds wall.
* @param {boolean} [top=true] - If true will create the top bounds wall.
* @param {boolean} [bottom=true] - If true will create the bottom bounds wall.
*/
setBounds: function (x, y, width, height, thickness, left, right, top, bottom)
{
if (x === undefined) { x = 0; }
if (y === undefined) { y = 0; }
if (width === undefined) { width = this.scene.sys.game.config.width; }
if (height === undefined) { height = this.scene.sys.game.config.height; }
if (thickness === undefined) { thickness = 64; }
if (left === undefined) { left = true; }
if (right === undefined) { right = true; }
if (top === undefined) { top = true; }
if (bottom === undefined) { bottom = true; }
this.updateWall(left, 'left', x - thickness, y, thickness, height);
this.updateWall(right, 'right', x + width, y, thickness, height);
this.updateWall(top, 'top', x, y - thickness, width, thickness);
this.updateWall(bottom, 'bottom', x, y + height, width, thickness);
return this;
},
// position = 'left', 'right', 'top' or 'bottom'
updateWall: function (add, position, x, y, width, height)
{
var wall = this.walls[position];
if (add)
{
if (wall)
{
this.localWorld.remove(wall);
}
// adjust center
x += (width / 2);
y += (height / 2);
this.walls[position] = this.create(x, y, width, height, { isStatic: true });
}
else
{
if (wall)
{
this.localWorld.remove(wall);
}
this.walls[position] = null;
}
},
setGravity: function (x, y, scale)
{
this.localWorld.gravity.x = x;
this.localWorld.gravity.y = y;
if (scale !== undefined)
{
this.localWorld.gravity.scale = scale;
}
return this;
},
create: function (x, y, width, height, options)
{
var body = Bodies.rectangle(x, y, width, height, options);
MatterWorld.add(this.localWorld, body);
return body;
},
// body can be single or an array
add: function (body)
{
MatterWorld.add(this.localWorld, body);
return this;
},
nextGroup: function (isNonColliding)
{
return Body.nextGroup(isNonColliding);
},
nextCategory: function ()
{
return MatterBody.nextCategory();
},
update: function (time, delta)
{
var correction = 1;
Engine.update(this.engine, delta, correction);
},
postUpdate: function ()
{
// NOOP
},
shutdown: function ()
{
},
destroy: function ()
{
}
});
module.exports = World;