2013-06-26 04:44:56 +00:00
|
|
|
/// <reference path="../Game.ts" />
|
2013-06-13 16:15:16 +00:00
|
|
|
/// <reference path="Body.ts" />
|
2013-06-14 16:55:07 +00:00
|
|
|
/// <reference path="joints/Joint.ts" />
|
2013-06-13 16:15:16 +00:00
|
|
|
|
|
|
|
/**
|
2013-06-26 04:44:56 +00:00
|
|
|
* Phaser - Physics Manager
|
2013-06-13 16:15:16 +00:00
|
|
|
*
|
2013-06-26 04:44:56 +00:00
|
|
|
* The Physics Manager is responsible for looking after, creating and colliding
|
|
|
|
* all of the physics bodies and joints in the world.
|
2013-06-13 16:15:16 +00:00
|
|
|
*/
|
|
|
|
|
2013-06-26 04:44:56 +00:00
|
|
|
module Phaser.Physics {
|
2013-06-13 16:15:16 +00:00
|
|
|
|
|
|
|
export class Manager {
|
|
|
|
|
|
|
|
constructor(game: Game) {
|
|
|
|
|
|
|
|
this.game = game;
|
|
|
|
|
2013-06-26 04:44:56 +00:00
|
|
|
this.gravity = new Phaser.Vec2;
|
|
|
|
|
|
|
|
this.space = new Space(this);
|
2013-06-14 03:13:00 +00:00
|
|
|
|
2013-06-14 02:20:47 +00:00
|
|
|
Manager.collision = new Collision();
|
|
|
|
|
2013-06-13 16:15:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Local reference to Game.
|
|
|
|
*/
|
|
|
|
public game: Game;
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static debug: HTMLTextAreaElement;
|
|
|
|
|
|
|
|
public static clear() {
|
2013-06-26 04:44:56 +00:00
|
|
|
//Manager.debug.textContent = "";
|
2013-06-21 22:39:29 +00:00
|
|
|
Manager.log = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
public static write(s: string) {
|
2013-06-26 04:44:56 +00:00
|
|
|
//Manager.debug.textContent += s + "\n";
|
2013-06-21 22:39:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static writeAll() {
|
|
|
|
|
|
|
|
for (var i = 0; i < Manager.log.length; i++)
|
|
|
|
{
|
|
|
|
//Manager.debug.textContent += Manager.log[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public static log = [];
|
|
|
|
|
|
|
|
public static dump(phase: string, body: Body) {
|
|
|
|
|
2013-06-26 04:44:56 +00:00
|
|
|
/*
|
2013-06-21 22:39:29 +00:00
|
|
|
var s = "\n\nPhase: " + phase + "\n";
|
|
|
|
s += "Position: " + body.position.toString() + "\n";
|
|
|
|
s += "Velocity: " + body.velocity.toString() + "\n";
|
|
|
|
s += "Angle: " + body.angle + "\n";
|
|
|
|
s += "Force: " + body.force.toString() + "\n";
|
|
|
|
s += "Torque: " + body.torque + "\n";
|
|
|
|
s += "Bounds: " + body.bounds.toString() + "\n";
|
|
|
|
s += "Shape ***\n";
|
|
|
|
s += "Vert 0: " + body.shapes[0].verts[0].toString() + "\n";
|
|
|
|
s += "Vert 1: " + body.shapes[0].verts[1].toString() + "\n";
|
|
|
|
s += "Vert 2: " + body.shapes[0].verts[2].toString() + "\n";
|
|
|
|
s += "Vert 3: " + body.shapes[0].verts[3].toString() + "\n";
|
|
|
|
s += "TVert 0: " + body.shapes[0].tverts[0].toString() + "\n";
|
|
|
|
s += "TVert 1: " + body.shapes[0].tverts[1].toString() + "\n";
|
|
|
|
s += "TVert 2: " + body.shapes[0].tverts[2].toString() + "\n";
|
|
|
|
s += "TVert 3: " + body.shapes[0].tverts[3].toString() + "\n";
|
|
|
|
s += "Plane 0: " + body.shapes[0].planes[0].normal.toString() + "\n";
|
|
|
|
s += "Plane 1: " + body.shapes[0].planes[1].normal.toString() + "\n";
|
|
|
|
s += "Plane 2: " + body.shapes[0].planes[2].normal.toString() + "\n";
|
|
|
|
s += "Plane 3: " + body.shapes[0].planes[3].normal.toString() + "\n";
|
|
|
|
s += "TPlane 0: " + body.shapes[0].tplanes[0].normal.toString() + "\n";
|
|
|
|
s += "TPlane 1: " + body.shapes[0].tplanes[1].normal.toString() + "\n";
|
|
|
|
s += "TPlane 2: " + body.shapes[0].tplanes[2].normal.toString() + "\n";
|
|
|
|
s += "TPlane 3: " + body.shapes[0].tplanes[3].normal.toString() + "\n";
|
|
|
|
|
|
|
|
Manager.log.push(s);
|
2013-06-26 04:44:56 +00:00
|
|
|
*/
|
2013-06-21 22:39:29 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-06-14 02:20:47 +00:00
|
|
|
public static collision: Collision;
|
|
|
|
|
2013-06-13 16:15:16 +00:00
|
|
|
public static SHAPE_TYPE_CIRCLE: number = 0;
|
|
|
|
public static SHAPE_TYPE_SEGMENT: number = 1;
|
|
|
|
public static SHAPE_TYPE_POLY: number = 2;
|
|
|
|
public static SHAPE_NUM_TYPES: number = 3;
|
|
|
|
|
|
|
|
public static JOINT_TYPE_ANGLE: number = 0;
|
|
|
|
public static JOINT_TYPE_REVOLUTE: number = 1;
|
|
|
|
public static JOINT_TYPE_WELD: number = 2;
|
|
|
|
public static JOINT_TYPE_WHEEL: number = 3;
|
|
|
|
public static JOINT_TYPE_PRISMATIC: number = 4;
|
|
|
|
public static JOINT_TYPE_DISTANCE: number = 5;
|
|
|
|
public static JOINT_TYPE_ROPE: number = 6;
|
|
|
|
public static JOINT_TYPE_MOUSE: number = 7;
|
|
|
|
|
|
|
|
public static JOINT_LINEAR_SLOP: number = 0.0008;
|
2013-06-26 04:44:56 +00:00
|
|
|
public static JOINT_ANGULAR_SLOP: number = 2 * 0.017453292519943294444444444444444;
|
2013-06-13 16:15:16 +00:00
|
|
|
public static JOINT_MAX_LINEAR_CORRECTION: number = 0.5;
|
2013-06-26 04:44:56 +00:00
|
|
|
public static JOINT_MAX_ANGULAR_CORRECTION: number = 8 * 0.017453292519943294444444444444444;
|
2013-06-13 16:15:16 +00:00
|
|
|
|
|
|
|
public static JOINT_LIMIT_STATE_INACTIVE: number = 0;
|
|
|
|
public static JOINT_LIMIT_STATE_AT_LOWER: number = 1;
|
|
|
|
public static JOINT_LIMIT_STATE_AT_UPPER: number = 2;
|
|
|
|
public static JOINT_LIMIT_STATE_EQUAL_LIMITS: number = 3;
|
|
|
|
|
2013-06-14 01:42:51 +00:00
|
|
|
public static CONTACT_SOLVER_COLLISION_SLOP: number = 0.0008;
|
|
|
|
public static CONTACT_SOLVER_BAUMGARTE: number = 0.28;
|
|
|
|
public static CONTACT_SOLVER_MAX_LINEAR_CORRECTION: number = 1;//Infinity;
|
|
|
|
|
2013-06-13 16:15:16 +00:00
|
|
|
public static bodyCounter: number = 0;
|
|
|
|
public static jointCounter: number = 0;
|
|
|
|
public static shapeCounter: number = 0;
|
2013-06-14 03:13:00 +00:00
|
|
|
|
|
|
|
public space: Space;
|
2013-06-14 14:18:41 +00:00
|
|
|
public lastTime: number = Date.now();
|
2013-06-14 03:13:00 +00:00
|
|
|
public frameRateHz: number = 60;
|
|
|
|
public timeDelta: number = 0;
|
2013-06-26 04:44:56 +00:00
|
|
|
public paused: bool = false;
|
2013-06-25 14:35:45 +00:00
|
|
|
public step: bool = false; // step through the simulation (i.e. per click)
|
2013-06-26 04:44:56 +00:00
|
|
|
//public paused: bool = true;
|
|
|
|
//public step: bool = false; // step through the simulation (i.e. per click)
|
2013-06-14 03:13:00 +00:00
|
|
|
public velocityIterations: number = 8;
|
|
|
|
public positionIterations: number = 4;
|
|
|
|
public allowSleep: bool = true;
|
|
|
|
public warmStarting: bool = true;
|
|
|
|
|
2013-06-26 04:44:56 +00:00
|
|
|
public gravity: Phaser.Vec2;
|
|
|
|
|
|
|
|
|
2013-06-14 03:13:00 +00:00
|
|
|
public update() {
|
|
|
|
|
2013-06-26 04:44:56 +00:00
|
|
|
// Get these from Phaser.Time instead
|
2013-06-14 03:13:00 +00:00
|
|
|
var time = Date.now();
|
|
|
|
var frameTime = (time - this.lastTime) / 1000;
|
|
|
|
this.lastTime = time;
|
|
|
|
|
|
|
|
// if rAf - why?
|
|
|
|
frameTime = Math.floor(frameTime * 60 + 0.5) / 60;
|
|
|
|
|
|
|
|
//if (!mouseDown)
|
|
|
|
//{
|
|
|
|
// var p = canvasToWorld(mousePosition);
|
|
|
|
// var body = space.findBodyByPoint(p);
|
|
|
|
// //domCanvas.style.cursor = body ? "pointer" : "default";
|
|
|
|
//}
|
|
|
|
|
|
|
|
if (!this.paused || this.step)
|
|
|
|
{
|
2013-06-21 22:39:29 +00:00
|
|
|
Manager.clear();
|
|
|
|
|
2013-06-14 03:13:00 +00:00
|
|
|
var h = 1 / this.frameRateHz;
|
|
|
|
|
|
|
|
this.timeDelta += frameTime;
|
|
|
|
|
|
|
|
if (this.step)
|
|
|
|
{
|
|
|
|
this.step = false;
|
|
|
|
this.timeDelta = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var maxSteps = 4; maxSteps > 0 && this.timeDelta >= h; maxSteps--)
|
|
|
|
{
|
|
|
|
this.space.step(h, this.velocityIterations, this.positionIterations, this.warmStarting, this.allowSleep);
|
|
|
|
this.timeDelta -= h;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.timeDelta > h)
|
|
|
|
{
|
|
|
|
this.timeDelta = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//frameCount++;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-06-26 04:44:56 +00:00
|
|
|
public addBody(body: Body) {
|
|
|
|
this.space.addBody(body);
|
|
|
|
}
|
|
|
|
|
|
|
|
public removeBody(body: Body) {
|
|
|
|
this.space.removeBody(body);
|
|
|
|
}
|
|
|
|
|
|
|
|
public addJoint(joint: IJoint) {
|
|
|
|
this.space.addJoint(joint);
|
|
|
|
}
|
|
|
|
|
|
|
|
public removeJoint(joint: IJoint) {
|
|
|
|
this.space.removeJoint(joint);
|
|
|
|
}
|
|
|
|
|
2013-06-14 14:18:41 +00:00
|
|
|
public pixelsToMeters(value: number): number {
|
|
|
|
return value * 0.02;
|
|
|
|
}
|
2013-06-14 03:13:00 +00:00
|
|
|
|
2013-06-14 14:18:41 +00:00
|
|
|
public metersToPixels(value: number): number {
|
|
|
|
return value * 50;
|
|
|
|
}
|
2013-06-13 16:15:16 +00:00
|
|
|
|
|
|
|
public static pixelsToMeters(value: number): number {
|
|
|
|
return value * 0.02;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static metersToPixels(value: number): number {
|
|
|
|
return value * 50;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static p2m(value: number): number {
|
|
|
|
return value * 0.02;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static m2p(value: number): number {
|
|
|
|
return value * 50;
|
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static areaForCircle(radius_outer: number, radius_inner: number): number {
|
2013-06-14 01:42:51 +00:00
|
|
|
return Math.PI * (radius_outer * radius_outer - radius_inner * radius_inner);
|
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static inertiaForCircle(mass: number, center: Phaser.Vec2, radius_outer: number, radius_inner: number): number {
|
2013-06-14 03:13:00 +00:00
|
|
|
return mass * ((radius_outer * radius_outer + radius_inner * radius_inner) * 0.5 + center.lengthSq());
|
2013-06-14 01:42:51 +00:00
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static areaForSegment(a: Phaser.Vec2, b: Phaser.Vec2, radius: number): number {
|
2013-06-14 01:42:51 +00:00
|
|
|
return radius * (Math.PI * radius + 2 * Phaser.Vec2Utils.distance(a, b));
|
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static centroidForSegment(a: Phaser.Vec2, b: Phaser.Vec2): Phaser.Vec2 {
|
2013-06-14 01:42:51 +00:00
|
|
|
return Phaser.Vec2Utils.scale(Phaser.Vec2Utils.add(a, b), 0.5);
|
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static inertiaForSegment(mass: number, a: Phaser.Vec2, b: Phaser.Vec2): number {
|
2013-06-14 01:42:51 +00:00
|
|
|
|
|
|
|
var distsq = Phaser.Vec2Utils.distanceSq(b, a);
|
|
|
|
var offset: Phaser.Vec2 = Phaser.Vec2Utils.scale(Phaser.Vec2Utils.add(a, b), 0.5);
|
|
|
|
|
|
|
|
return mass * (distsq / 12 + offset.lengthSq());
|
2013-06-21 22:39:29 +00:00
|
|
|
|
2013-06-14 01:42:51 +00:00
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static areaForPoly(verts: Phaser.Vec2[]): number {
|
2013-06-14 01:42:51 +00:00
|
|
|
|
|
|
|
var area = 0;
|
|
|
|
|
|
|
|
for (var i = 0; i < verts.length; i++)
|
|
|
|
{
|
|
|
|
area += Phaser.Vec2Utils.cross(verts[i], verts[(i + 1) % verts.length]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return area / 2;
|
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static centroidForPoly(verts: Phaser.Vec2[]): Phaser.Vec2 {
|
2013-06-14 01:42:51 +00:00
|
|
|
|
|
|
|
var area = 0;
|
|
|
|
var vsum = new Phaser.Vec2;
|
|
|
|
|
|
|
|
for (var i = 0; i < verts.length; i++)
|
|
|
|
{
|
|
|
|
var v1 = verts[i];
|
|
|
|
var v2 = verts[(i + 1) % verts.length];
|
|
|
|
var cross = Phaser.Vec2Utils.cross(v1, v2);
|
|
|
|
|
|
|
|
area += cross;
|
|
|
|
|
|
|
|
// SO many vecs created here - unroll these bad boys
|
|
|
|
vsum.add(Phaser.Vec2Utils.scale(Phaser.Vec2Utils.add(v1, v2), cross));
|
|
|
|
}
|
|
|
|
|
|
|
|
return Phaser.Vec2Utils.scale(vsum, 1 / (3 * area));
|
2013-06-21 22:39:29 +00:00
|
|
|
|
2013-06-14 01:42:51 +00:00
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static inertiaForPoly(mass: number, verts: Phaser.Vec2[], offset: Phaser.Vec2): number {
|
2013-06-14 01:42:51 +00:00
|
|
|
|
|
|
|
var sum1 = 0;
|
|
|
|
var sum2 = 0;
|
|
|
|
|
|
|
|
for (var i = 0; i < verts.length; i++)
|
|
|
|
{
|
|
|
|
var v1 = Phaser.Vec2Utils.add(verts[i], offset);
|
|
|
|
var v2 = Phaser.Vec2Utils.add(verts[(i + 1) % verts.length], offset);
|
|
|
|
|
|
|
|
var a = Phaser.Vec2Utils.cross(v2, v1);
|
|
|
|
var b = Phaser.Vec2Utils.dot(v1, v1) + Phaser.Vec2Utils.dot(v1, v2) + Phaser.Vec2Utils.dot(v2, v2);
|
|
|
|
|
|
|
|
sum1 += a * b;
|
|
|
|
sum2 += a;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (mass * sum1) / (6 * sum2);
|
2013-06-21 22:39:29 +00:00
|
|
|
|
2013-06-14 01:42:51 +00:00
|
|
|
}
|
|
|
|
|
2013-06-21 22:39:29 +00:00
|
|
|
public static inertiaForBox(mass: number, w: number, h: number) {
|
2013-06-14 01:42:51 +00:00
|
|
|
return mass * (w * w + h * h) / 12;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the convex hull using the Gift wrapping algorithm (http://en.wikipedia.org/wiki/Gift_wrapping_algorithm)
|
|
|
|
public static createConvexHull(points) {
|
|
|
|
|
|
|
|
// Find the right most point on the hull
|
|
|
|
var i0 = 0;
|
|
|
|
var x0 = points[0].x;
|
|
|
|
|
|
|
|
for (var i = 1; i < points.length; i++)
|
|
|
|
{
|
|
|
|
var x = points[i].x;
|
|
|
|
|
|
|
|
if (x > x0 || (x == x0 && points[i].y < points[i0].y))
|
|
|
|
{
|
|
|
|
i0 = i;
|
|
|
|
x0 = x;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var n = points.length;
|
|
|
|
var hull = [];
|
|
|
|
var m = 0;
|
|
|
|
var ih = i0;
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
hull[m] = ih;
|
|
|
|
|
|
|
|
var ie = 0;
|
|
|
|
|
|
|
|
for (var j = 1; j < n; j++)
|
|
|
|
{
|
|
|
|
if (ie == ih)
|
|
|
|
{
|
|
|
|
ie = j;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
var r = Phaser.Vec2Utils.subtract(points[ie], points[hull[m]]);
|
|
|
|
var v = Phaser.Vec2Utils.subtract(points[j], points[hull[m]]);
|
|
|
|
var c = Phaser.Vec2Utils.cross(r, v);
|
|
|
|
|
|
|
|
if (c < 0)
|
|
|
|
{
|
|
|
|
ie = j;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Collinearity check
|
|
|
|
if (c == 0 && v.lengthSq() > r.lengthSq())
|
|
|
|
{
|
|
|
|
ie = j;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m++;
|
|
|
|
ih = ie;
|
|
|
|
|
|
|
|
if (ie == i0)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy vertices
|
|
|
|
var newPoints = [];
|
|
|
|
|
|
|
|
for (var i = 0; i < m; ++i)
|
|
|
|
{
|
|
|
|
newPoints.push(points[hull[i]]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return newPoints;
|
|
|
|
}
|
|
|
|
|
2013-06-13 16:15:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|