phaser/src/physics/arcade/Body.js

824 lines
18 KiB
JavaScript

// Phaser.Physics.Arcade.Body
var CircleContains = require('../../geom/circle/Contains');
var Class = require('../../utils/Class');
var CONST = require('./const');
var Rectangle = require('../../geom/rectangle/Rectangle');
var RectangleContains = require('../../geom/rectangle/Contains');
var Vector2 = require('../../math/Vector2');
var Body = new Class({
initialize:
function Body (world, gameObject)
{
this.world = world;
this.gameObject = gameObject;
this.debugShowBody = world.defaults.debugShowBody;
this.debugShowVelocity = world.defaults.debugShowVelocity;
this.debugBodyColor = world.defaults.bodyDebugColor;
this.enable = true;
this.isCircle = false;
this.radius = 0;
this.offset = new Vector2();
this.position = new Vector2(gameObject.x, gameObject.y);
this.prev = new Vector2(this.position.x, this.position.y);
this.allowRotation = true;
this.rotation = gameObject.angle;
this.preRotation = gameObject.angle;
this.width = gameObject.width;
this.height = gameObject.height;
this.sourceWidth = gameObject.width;
this.sourceHeight = gameObject.height;
if (gameObject.frame)
{
this.sourceWidth = gameObject.frame.realWidth;
this.sourceHeight = gameObject.frame.realHeight;
}
this.halfWidth = Math.abs(gameObject.width / 2);
this.halfHeight = Math.abs(gameObject.height / 2);
this.center = new Vector2(gameObject.x + this.halfWidth, gameObject.y + this.halfHeight);
this.velocity = new Vector2();
this.newVelocity = new Vector2();
this.deltaMax = new Vector2();
this.acceleration = new Vector2();
this.allowDrag = true;
this.drag = new Vector2();
this.allowGravity = true;
this.gravity = new Vector2();
this.bounce = new Vector2();
this.worldBounce = null;
// If true this Body will dispatch events
this.onWorldBounds = false;
this.onCollide = false;
this.onOverlap = false;
this.maxVelocity = new Vector2(10000, 10000);
this.friction = new Vector2(1, 0);
this.angularVelocity = 0;
this.angularAcceleration = 0;
this.angularDrag = 0;
this.maxAngular = 1000;
this.mass = 1;
this.angle = 0;
this.speed = 0;
this.facing = CONST.FACING_NONE;
this.immovable = false;
this.moves = true;
this.customSeparateX = false;
this.customSeparateY = false;
this.overlapX = 0;
this.overlapY = 0;
this.overlapR = 0;
this.embedded = false;
this.collideWorldBounds = false;
this.checkCollision = { none: false, up: true, down: true, left: true, right: true };
this.touching = { none: true, up: false, down: false, left: false, right: false };
this.wasTouching = { none: true, up: false, down: false, left: false, right: false };
this.blocked = { none: true, up: false, down: false, left: false, right: false };
this.dirty = false;
this.syncBounds = false;
this.isMoving = false;
this.stopVelocityOnCollide = true;
// read-only
this.physicsType = CONST.DYNAMIC_BODY;
this._reset = true;
this._sx = gameObject.scaleX;
this._sy = gameObject.scaleY;
this._dx = 0;
this._dy = 0;
this._bounds = new Rectangle();
},
updateBounds: function ()
{
var sprite = this.gameObject;
if (this.syncBounds)
{
var b = sprite.getBounds(this._bounds);
if (b.width !== this.width || b.height !== this.height)
{
this.width = b.width;
this.height = b.height;
this._reset = true;
}
}
else
{
var asx = Math.abs(sprite.scaleX);
var asy = Math.abs(sprite.scaleY);
if (asx !== this._sx || asy !== this._sy)
{
this.width = this.sourceWidth * asx;
this.height = this.sourceHeight * asy;
this._sx = asx;
this._sy = asy;
this._reset = true;
}
}
if (this._reset)
{
this.halfWidth = Math.floor(this.width / 2);
this.halfHeight = Math.floor(this.height / 2);
this.updateCenter();
}
},
updateCenter: function ()
{
this.center.set(this.position.x + this.halfWidth, this.position.y + this.halfHeight);
},
update: function (delta)
{
this.dirty = true;
// Store and reset collision flags
this.wasTouching.none = this.touching.none;
this.wasTouching.up = this.touching.up;
this.wasTouching.down = this.touching.down;
this.wasTouching.left = this.touching.left;
this.wasTouching.right = this.touching.right;
this.touching.none = true;
this.touching.up = false;
this.touching.down = false;
this.touching.left = false;
this.touching.right = false;
this.blocked.none = true;
this.blocked.up = false;
this.blocked.down = false;
this.blocked.left = false;
this.blocked.right = false;
this.overlapR = 0;
this.overlapX = 0;
this.overlapY = 0;
this.embedded = false;
this.updateBounds();
var sprite = this.gameObject;
this.position.x = sprite.x - sprite.displayOriginX + (sprite.scaleX * this.offset.x);
this.position.y = sprite.y - sprite.displayOriginY + (sprite.scaleY * this.offset.y);
this.updateCenter();
this.rotation = sprite.angle;
this.preRotation = this.rotation;
if (this._reset)
{
this.prev.x = this.position.x;
this.prev.y = this.position.y;
}
if (this.moves)
{
this.world.updateMotion(this);
this.newVelocity.set(this.velocity.x * delta, this.velocity.y * delta);
this.position.x += this.newVelocity.x;
this.position.y += this.newVelocity.y;
this.updateCenter();
if (this.position.x !== this.prev.x || this.position.y !== this.prev.y)
{
this.angle = Math.atan2(this.velocity.y, this.velocity.x);
}
this.speed = Math.sqrt(this.velocity.x * this.velocity.x + this.velocity.y * this.velocity.y);
// Now the State update will throw collision checks at the Body
// And finally we'll integrate the new position back to the Sprite in postUpdate
if (this.collideWorldBounds && this.checkWorldBounds() && this.onWorldBounds)
{
this.world.emit('worldbounds', this, this.blocked.up, this.blocked.down, this.blocked.left, this.blocked.right);
}
}
this._dx = this.deltaX();
this._dy = this.deltaY();
this._reset = false;
},
// Feeds the body results back into the parent gameobject (if there is one)
postUpdate: function ()
{
// Only allow postUpdate to be called once per frame
if (!this.enable || !this.dirty)
{
return;
}
this.dirty = false;
if (this.deltaX() < 0)
{
this.facing = CONST.FACING_LEFT;
}
else if (this.deltaX() > 0)
{
this.facing = CONST.FACING_RIGHT;
}
if (this.deltaY() < 0)
{
this.facing = CONST.FACING_UP;
}
else if (this.deltaY() > 0)
{
this.facing = CONST.FACING_DOWN;
}
if (this.moves)
{
this._dx = this.deltaX();
this._dy = this.deltaY();
if (this.deltaMax.x !== 0 && this._dx !== 0)
{
if (this._dx < 0 && this._dx < -this.deltaMax.x)
{
this._dx = -this.deltaMax.x;
}
else if (this._dx > 0 && this._dx > this.deltaMax.x)
{
this._dx = this.deltaMax.x;
}
}
if (this.deltaMax.y !== 0 && this._dy !== 0)
{
if (this._dy < 0 && this._dy < -this.deltaMax.y)
{
this._dy = -this.deltaMax.y;
}
else if (this._dy > 0 && this._dy > this.deltaMax.y)
{
this._dy = this.deltaMax.y;
}
}
this.gameObject.x += this._dx;
this.gameObject.y += this._dy;
this._reset = true;
}
this.updateCenter();
if (this.allowRotation)
{
this.gameObject.angle += this.deltaZ();
}
this.prev.x = this.position.x;
this.prev.y = this.position.y;
},
checkWorldBounds: function ()
{
var pos = this.position;
var bounds = this.world.bounds;
var check = this.world.checkCollision;
var bx = (this.worldBounce) ? -this.worldBounce.x : -this.bounce.x;
var by = (this.worldBounce) ? -this.worldBounce.y : -this.bounce.y;
if (pos.x < bounds.x && check.left)
{
pos.x = bounds.x;
this.velocity.x *= bx;
this.blocked.left = true;
this.blocked.none = false;
}
else if (this.right > bounds.right && check.right)
{
pos.x = bounds.right - this.width;
this.velocity.x *= bx;
this.blocked.right = true;
this.blocked.none = false;
}
if (pos.y < bounds.y && check.up)
{
pos.y = bounds.y;
this.velocity.y *= by;
this.blocked.up = true;
this.blocked.none = false;
}
else if (this.bottom > bounds.bottom && check.down)
{
pos.y = bounds.bottom - this.height;
this.velocity.y *= by;
this.blocked.down = true;
this.blocked.none = false;
}
return !this.blocked.none;
},
setOffset: function (x, y)
{
if (y === undefined) { y = x; }
this.offset.set(x, y);
return this;
},
setSize: function (width, height, center)
{
if (center === undefined) { center = true; }
this.sourceWidth = width;
this.sourceHeight = height;
this.width = this.sourceWidth * this._sx;
this.height = this.sourceHeight * this._sy;
this.halfWidth = Math.floor(this.width / 2);
this.halfHeight = Math.floor(this.height / 2);
this.updateCenter();
if (center && this.gameObject.getCenter)
{
var gameObject = this.gameObject;
var ox = gameObject.displayWidth / 2;
var oy = gameObject.displayHeight / 2;
this.offset.set(ox - this.halfWidth, oy - this.halfHeight);
}
this.isCircle = false;
this.radius = 0;
return this;
},
setCircle: function (radius, offsetX, offsetY)
{
if (offsetX === undefined) { offsetX = this.offset.x; }
if (offsetY === undefined) { offsetY = this.offset.y; }
if (radius > 0)
{
this.isCircle = true;
this.radius = radius;
this.sourceWidth = radius * 2;
this.sourceHeight = radius * 2;
this.width = this.sourceWidth * this._sx;
this.height = this.sourceHeight * this._sy;
this.halfWidth = Math.floor(this.width / 2);
this.halfHeight = Math.floor(this.height / 2);
this.offset.set(offsetX, offsetY);
this.updateCenter();
}
else
{
this.isCircle = false;
}
return this;
},
reset: function (x, y)
{
this.stop();
var sprite = this.gameObject;
this.position.x = x - sprite.displayOriginX + (sprite.scaleX * this.offset.x);
this.position.y = y - sprite.displayOriginY + (sprite.scaleY * this.offset.y);
this.prev.x = this.position.x;
this.prev.y = this.position.y;
this.rotation = this.gameObject.angle;
this.preRotation = this.rotation;
this.updateBounds();
this.updateCenter();
},
stop: function ()
{
this.velocity.set(0);
this.acceleration.set(0);
this.speed = 0;
this.angularVelocity = 0;
this.angularAcceleration = 0;
return this;
},
getBounds: function (obj)
{
obj.x = this.x;
obj.y = this.y;
obj.right = this.right;
obj.bottom = this.bottom;
return obj;
},
hitTest: function (x, y)
{
return (this.isCircle) ? CircleContains(this, x, y) : RectangleContains(this, x, y);
},
onFloor: function ()
{
return this.blocked.down;
},
onCeiling: function ()
{
return this.blocked.up;
},
onWall: function ()
{
return (this.blocked.left || this.blocked.right);
},
deltaAbsX: function ()
{
return (this.deltaX() > 0) ? this.deltaX() : -this.deltaX();
},
deltaAbsY: function ()
{
return (this.deltaY() > 0) ? this.deltaY() : -this.deltaY();
},
deltaX: function ()
{
return this.position.x - this.prev.x;
},
deltaY: function ()
{
return this.position.y - this.prev.y;
},
deltaZ: function ()
{
return this.rotation - this.preRotation;
},
destroy: function ()
{
this.gameObject.body = null;
this.gameObject = null;
},
drawDebug: function (graphic)
{
var pos = this.position;
var x = pos.x + this.halfWidth;
var y = pos.y + this.halfHeight;
if (this.debugShowBody)
{
graphic.lineStyle(1, this.debugBodyColor);
if (this.isCircle)
{
graphic.strokeCircle(x, y, this.radius);
}
else
{
graphic.strokeRect(pos.x, pos.y, this.width, this.height);
}
}
if (this.debugShowVelocity)
{
graphic.lineStyle(1, this.world.defaults.velocityDebugColor, 1);
graphic.lineBetween(x, y, x + this.velocity.x / 2, y + this.velocity.y / 2);
}
},
willDrawDebug: function ()
{
return (this.debugShowBody || this.debugShowVelocity);
},
setCollideWorldBounds: function (value)
{
this.collideWorldBounds = value;
return this;
},
setVelocity: function (x, y)
{
this.velocity.set(x, y);
return this;
},
setVelocityX: function (value)
{
this.velocity.x = value;
return this;
},
setVelocityY: function (value)
{
this.velocity.y = value;
return this;
},
setBounce: function (x, y)
{
this.bounce.set(x, y);
return this;
},
setBounceX: function (value)
{
this.bounce.x = value;
return this;
},
setBounceY: function (value)
{
this.bounce.y = value;
return this;
},
setAcceleration: function (x, y)
{
this.acceleration.set(x, y);
return this;
},
setAccelerationX: function (value)
{
this.acceleration.x = value;
return this;
},
setAccelerationY: function (value)
{
this.acceleration.y = value;
return this;
},
setDrag: function (x, y)
{
this.drag.set(x, y);
return this;
},
setDragX: function (value)
{
this.drag.x = value;
return this;
},
setDragY: function (value)
{
this.drag.y = value;
return this;
},
setGravity: function (x, y)
{
this.gravity.set(x, y);
return this;
},
setGravityX: function (value)
{
this.gravity.x = value;
return this;
},
setGravityY: function (value)
{
this.gravity.y = value;
return this;
},
setFriction: function (x, y)
{
this.friction.set(x, y);
return this;
},
setFrictionX: function (value)
{
this.friction.x = value;
return this;
},
setFrictionY: function (value)
{
this.friction.y = value;
return this;
},
setAngularVelocity: function (value)
{
this.angularVelocity = value;
return this;
},
setAngularAcceleration: function (value)
{
this.angularAcceleration = value;
return this;
},
setAngularDrag: function (value)
{
this.angularDrag = value;
return this;
},
setMass: function (value)
{
this.mass = value;
return this;
},
setImmovable: function (value)
{
this.immovable = value;
return this;
},
x: {
get: function ()
{
return this.position.x;
},
set: function (value)
{
this.position.x = value;
}
},
y: {
get: function ()
{
return this.position.y;
},
set: function (value)
{
this.position.y = value;
}
},
left: {
get: function ()
{
return this.position.x;
}
},
right: {
get: function ()
{
return this.position.x + this.width;
}
},
top: {
get: function ()
{
return this.position.y;
}
},
bottom: {
get: function ()
{
return this.position.y + this.height;
}
}
});
module.exports = Body;