2022-06-25 00:23:03 +08:00

22335 lines
1.0 MiB

(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
typeof define === 'function' && define.amd ? define(['exports'], factory) :
(factory((global.box2d = {})));
}(this, (function (exports) { 'use strict';
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
function b2Assert(condition) {
var args = [];
for (var _i = 1; _i < arguments.length; _i++) {
args[_i - 1] = arguments[_i];
}
if (!condition) {
// debugger;
throw new (Error.bind.apply(Error, [void 0].concat(args)))();
}
}
function b2Maybe(value, def) {
return value !== undefined ? value : def;
}
var b2_maxFloat = 1E+37; // FLT_MAX instead of Number.MAX_VALUE;
var b2_epsilon = 1E-5; // FLT_EPSILON instead of Number.MIN_VALUE;
var b2_epsilon_sq = (b2_epsilon * b2_epsilon);
var b2_pi = 3.14159265359; // Math.PI;
/// @file
/// Global tuning constants based on meters-kilograms-seconds (MKS) units.
///
// Collision
/// The maximum number of contact points between two convex shapes. Do
/// not change this value.
var b2_maxManifoldPoints = 2;
/// The maximum number of vertices on a convex polygon. You cannot increase
/// this too much because b2BlockAllocator has a maximum object size.
var b2_maxPolygonVertices = 8;
/// This is used to fatten AABBs in the dynamic tree. This allows proxies
/// to move by a small amount without triggering a tree adjustment.
/// This is in meters.
var b2_aabbExtension = 0.1;
/// This is used to fatten AABBs in the dynamic tree. This is used to predict
/// the future position based on the current displacement.
/// This is a dimensionless multiplier.
var b2_aabbMultiplier = 2;
/// A small length used as a collision and constraint tolerance. Usually it is
/// chosen to be numerically significant, but visually insignificant.
var b2_linearSlop = 0.008; // 0.005;
/// A small angle used as a collision and constraint tolerance. Usually it is
/// chosen to be numerically significant, but visually insignificant.
var b2_angularSlop = 2 / 180 * b2_pi;
/// The radius of the polygon/edge shape skin. This should not be modified. Making
/// this smaller means polygons will have an insufficient buffer for continuous collision.
/// Making it larger may create artifacts for vertex collision.
var b2_polygonRadius = 2 * b2_linearSlop;
/// Maximum number of sub-steps per contact in continuous physics simulation.
var b2_maxSubSteps = 8;
// Dynamics
/// Maximum number of contacts to be handled to solve a TOI impact.
var b2_maxTOIContacts = 32;
/// A velocity threshold for elastic collisions. Any collision with a relative linear
/// velocity below this threshold will be treated as inelastic.
var b2_velocityThreshold = 1;
/// The maximum linear position correction used when solving constraints. This helps to
/// prevent overshoot.
var b2_maxLinearCorrection = 0.2;
/// The maximum angular position correction used when solving constraints. This helps to
/// prevent overshoot.
var b2_maxAngularCorrection = 8 / 180 * b2_pi;
/// The maximum linear velocity of a body. This limit is very large and is used
/// to prevent numerical problems. You shouldn't need to adjust this.
var b2_maxTranslation = 2;
var b2_maxTranslationSquared = b2_maxTranslation * b2_maxTranslation;
/// The maximum angular velocity of a body. This limit is very large and is used
/// to prevent numerical problems. You shouldn't need to adjust this.
var b2_maxRotation = 0.5 * b2_pi;
var b2_maxRotationSquared = b2_maxRotation * b2_maxRotation;
/// This scale factor controls how fast overlap is resolved. Ideally this would be 1 so
/// that overlap is removed in one time step. However using values close to 1 often lead
/// to overshoot.
var b2_baumgarte = 0.2;
var b2_toiBaumgarte = 0.75;
// #if B2_ENABLE_PARTICLE
// Particle
/// A symbolic constant that stands for particle allocation error.
var b2_invalidParticleIndex = -1;
var b2_maxParticleIndex = 0x7FFFFFFF;
/// The default distance between particles, multiplied by the particle diameter.
var b2_particleStride = 0.75;
/// The minimum particle weight that produces pressure.
var b2_minParticleWeight = 1.0;
/// The upper limit for particle pressure.
var b2_maxParticlePressure = 0.25;
/// The upper limit for force between particles.
var b2_maxParticleForce = 0.5;
/// The maximum distance between particles in a triad, multiplied by the particle diameter.
var b2_maxTriadDistance = 2.0;
var b2_maxTriadDistanceSquared = (b2_maxTriadDistance * b2_maxTriadDistance);
/// The initial size of particle data buffers.
var b2_minParticleSystemBufferCapacity = 256;
/// The time into the future that collisions against barrier particles will be detected.
var b2_barrierCollisionTime = 2.5;
// #endif
// Sleep
/// The time that a body must be still before it will go to sleep.
var b2_timeToSleep = 0.5;
/// A body cannot sleep if its linear velocity is above this tolerance.
var b2_linearSleepTolerance = 0.01;
/// A body cannot sleep if its angular velocity is above this tolerance.
var b2_angularSleepTolerance = 2 / 180 * b2_pi;
// Memory Allocation
/// Implement this function to use your own memory allocator.
function b2Alloc(size) {
return null;
}
/// If you implement b2Alloc, you should also implement this function.
function b2Free(mem) {
}
/// Logging function.
function b2Log(message) {
var args = [];
for (var _i = 1; _i < arguments.length; _i++) {
args[_i - 1] = arguments[_i];
}
// console.log(message, ...args);
}
/// Version numbering scheme.
/// See http://en.wikipedia.org/wiki/Software_versioning
var b2Version = /** @class */ (function () {
function b2Version(major, minor, revision) {
if (major === void 0) { major = 0; }
if (minor === void 0) { minor = 0; }
if (revision === void 0) { revision = 0; }
this.major = 0; ///< significant changes
this.minor = 0; ///< incremental changes
this.revision = 0; ///< bug fixes
this.major = major;
this.minor = minor;
this.revision = revision;
}
b2Version.prototype.toString = function () {
return this.major + "." + this.minor + "." + this.revision;
};
return b2Version;
}());
/// Current version.
var b2_version = new b2Version(2, 3, 2);
var b2_branch = "master";
var b2_commit = "fbf51801d80fc389d43dc46524520e89043b6faf";
function b2ParseInt(v) {
return parseInt(v, 10);
}
function b2ParseUInt(v) {
return Math.abs(parseInt(v, 10));
}
function b2MakeArray(length, init) {
var a = [];
for (var i = 0; i < length; ++i) {
a.push(init(i));
}
return a;
}
function b2MakeNullArray(length) {
var a = [];
for (var i = 0; i < length; ++i) {
a.push(null);
}
return a;
}
function b2MakeNumberArray(length, init) {
if (init === void 0) { init = 0; }
var a = [];
for (var i = 0; i < length; ++i) {
a.push(init);
}
return a;
}
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2_pi_over_180 = b2_pi / 180;
var b2_180_over_pi = 180 / b2_pi;
var b2_two_pi = 2 * b2_pi;
var b2Abs = Math.abs;
var b2Min = Math.min;
var b2Max = Math.max;
function b2Clamp(a, lo, hi) {
return (a < lo) ? (lo) : ((a > hi) ? (hi) : (a));
}
function b2Swap(a, b) {
// DEBUG: b2Assert(false);
var tmp = a[0];
a[0] = b[0];
b[0] = tmp;
}
/// This function is used to ensure that a floating point number is
/// not a NaN or infinity.
var b2IsValid = isFinite;
function b2Sq(n) {
return n * n;
}
/// This is a approximate yet fast inverse square-root.
function b2InvSqrt(n) {
return 1 / Math.sqrt(n);
}
var b2Sqrt = Math.sqrt;
var b2Pow = Math.pow;
function b2DegToRad(degrees) {
return degrees * b2_pi_over_180;
}
function b2RadToDeg(radians) {
return radians * b2_180_over_pi;
}
var b2Cos = Math.cos;
var b2Sin = Math.sin;
var b2Acos = Math.acos;
var b2Asin = Math.asin;
var b2Atan2 = Math.atan2;
function b2NextPowerOfTwo(x) {
x |= (x >> 1) & 0x7FFFFFFF;
x |= (x >> 2) & 0x3FFFFFFF;
x |= (x >> 4) & 0x0FFFFFFF;
x |= (x >> 8) & 0x00FFFFFF;
x |= (x >> 16) & 0x0000FFFF;
return x + 1;
}
function b2IsPowerOfTwo(x) {
return x > 0 && (x & (x - 1)) === 0;
}
function b2Random() {
return Math.random() * 2 - 1;
}
function b2RandomRange(lo, hi) {
return (hi - lo) * Math.random() + lo;
}
/// A 2D column vector.
var b2Vec2 = /** @class */ (function () {
function b2Vec2(x, y) {
if (x === void 0) { x = 0; }
if (y === void 0) { y = 0; }
this.x = x;
this.y = y;
}
b2Vec2.prototype.Clone = function () {
return new b2Vec2(this.x, this.y);
};
b2Vec2.prototype.SetZero = function () {
this.x = 0;
this.y = 0;
return this;
};
b2Vec2.prototype.Set = function (x, y) {
this.x = x;
this.y = y;
return this;
};
b2Vec2.prototype.Copy = function (other) {
this.x = other.x;
this.y = other.y;
return this;
};
b2Vec2.prototype.SelfAdd = function (v) {
this.x += v.x;
this.y += v.y;
return this;
};
b2Vec2.prototype.SelfAddXY = function (x, y) {
this.x += x;
this.y += y;
return this;
};
b2Vec2.prototype.SelfSub = function (v) {
this.x -= v.x;
this.y -= v.y;
return this;
};
b2Vec2.prototype.SelfSubXY = function (x, y) {
this.x -= x;
this.y -= y;
return this;
};
b2Vec2.prototype.SelfMul = function (s) {
this.x *= s;
this.y *= s;
return this;
};
b2Vec2.prototype.SelfMulAdd = function (s, v) {
this.x += s * v.x;
this.y += s * v.y;
return this;
};
b2Vec2.prototype.SelfMulSub = function (s, v) {
this.x -= s * v.x;
this.y -= s * v.y;
return this;
};
b2Vec2.prototype.Dot = function (v) {
return this.x * v.x + this.y * v.y;
};
b2Vec2.prototype.Cross = function (v) {
return this.x * v.y - this.y * v.x;
};
b2Vec2.prototype.Length = function () {
var x = this.x, y = this.y;
return Math.sqrt(x * x + y * y);
};
b2Vec2.prototype.LengthSquared = function () {
var x = this.x, y = this.y;
return (x * x + y * y);
};
b2Vec2.prototype.Normalize = function () {
var length = this.Length();
if (length >= b2_epsilon) {
var inv_length = 1 / length;
this.x *= inv_length;
this.y *= inv_length;
}
return length;
};
b2Vec2.prototype.SelfNormalize = function () {
var length = this.Length();
if (length >= b2_epsilon) {
var inv_length = 1 / length;
this.x *= inv_length;
this.y *= inv_length;
}
return this;
};
b2Vec2.prototype.SelfRotate = function (radians) {
var c = Math.cos(radians);
var s = Math.sin(radians);
var x = this.x;
this.x = c * x - s * this.y;
this.y = s * x + c * this.y;
return this;
};
b2Vec2.prototype.IsValid = function () {
return isFinite(this.x) && isFinite(this.y);
};
b2Vec2.prototype.SelfCrossVS = function (s) {
var x = this.x;
this.x = s * this.y;
this.y = -s * x;
return this;
};
b2Vec2.prototype.SelfCrossSV = function (s) {
var x = this.x;
this.x = -s * this.y;
this.y = s * x;
return this;
};
b2Vec2.prototype.SelfMinV = function (v) {
this.x = b2Min(this.x, v.x);
this.y = b2Min(this.y, v.y);
return this;
};
b2Vec2.prototype.SelfMaxV = function (v) {
this.x = b2Max(this.x, v.x);
this.y = b2Max(this.y, v.y);
return this;
};
b2Vec2.prototype.SelfAbs = function () {
this.x = b2Abs(this.x);
this.y = b2Abs(this.y);
return this;
};
b2Vec2.prototype.SelfNeg = function () {
this.x = (-this.x);
this.y = (-this.y);
return this;
};
b2Vec2.prototype.SelfSkew = function () {
var x = this.x;
this.x = -this.y;
this.y = x;
return this;
};
b2Vec2.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2Vec2(); });
};
b2Vec2.AbsV = function (v, out) {
out.x = b2Abs(v.x);
out.y = b2Abs(v.y);
return out;
};
b2Vec2.MinV = function (a, b, out) {
out.x = b2Min(a.x, b.x);
out.y = b2Min(a.y, b.y);
return out;
};
b2Vec2.MaxV = function (a, b, out) {
out.x = b2Max(a.x, b.x);
out.y = b2Max(a.y, b.y);
return out;
};
b2Vec2.ClampV = function (v, lo, hi, out) {
out.x = b2Clamp(v.x, lo.x, hi.x);
out.y = b2Clamp(v.y, lo.y, hi.y);
return out;
};
b2Vec2.RotateV = function (v, radians, out) {
var v_x = v.x, v_y = v.y;
var c = Math.cos(radians);
var s = Math.sin(radians);
out.x = c * v_x - s * v_y;
out.y = s * v_x + c * v_y;
return out;
};
b2Vec2.DotVV = function (a, b) {
return a.x * b.x + a.y * b.y;
};
b2Vec2.CrossVV = function (a, b) {
return a.x * b.y - a.y * b.x;
};
b2Vec2.CrossVS = function (v, s, out) {
var v_x = v.x;
out.x = s * v.y;
out.y = -s * v_x;
return out;
};
b2Vec2.CrossVOne = function (v, out) {
var v_x = v.x;
out.x = v.y;
out.y = -v_x;
return out;
};
b2Vec2.CrossSV = function (s, v, out) {
var v_x = v.x;
out.x = -s * v.y;
out.y = s * v_x;
return out;
};
b2Vec2.CrossOneV = function (v, out) {
var v_x = v.x;
out.x = -v.y;
out.y = v_x;
return out;
};
b2Vec2.AddVV = function (a, b, out) { out.x = a.x + b.x; out.y = a.y + b.y; return out; };
b2Vec2.SubVV = function (a, b, out) { out.x = a.x - b.x; out.y = a.y - b.y; return out; };
b2Vec2.MulSV = function (s, v, out) { out.x = v.x * s; out.y = v.y * s; return out; };
b2Vec2.MulVS = function (v, s, out) { out.x = v.x * s; out.y = v.y * s; return out; };
b2Vec2.AddVMulSV = function (a, s, b, out) { out.x = a.x + (s * b.x); out.y = a.y + (s * b.y); return out; };
b2Vec2.SubVMulSV = function (a, s, b, out) { out.x = a.x - (s * b.x); out.y = a.y - (s * b.y); return out; };
b2Vec2.AddVCrossSV = function (a, s, v, out) {
var v_x = v.x;
out.x = a.x - (s * v.y);
out.y = a.y + (s * v_x);
return out;
};
b2Vec2.MidVV = function (a, b, out) { out.x = (a.x + b.x) * 0.5; out.y = (a.y + b.y) * 0.5; return out; };
b2Vec2.ExtVV = function (a, b, out) { out.x = (b.x - a.x) * 0.5; out.y = (b.y - a.y) * 0.5; return out; };
b2Vec2.IsEqualToV = function (a, b) {
return a.x === b.x && a.y === b.y;
};
b2Vec2.DistanceVV = function (a, b) {
var c_x = a.x - b.x;
var c_y = a.y - b.y;
return Math.sqrt(c_x * c_x + c_y * c_y);
};
b2Vec2.DistanceSquaredVV = function (a, b) {
var c_x = a.x - b.x;
var c_y = a.y - b.y;
return (c_x * c_x + c_y * c_y);
};
b2Vec2.NegV = function (v, out) { out.x = -v.x; out.y = -v.y; return out; };
b2Vec2.ZERO = new b2Vec2(0, 0);
b2Vec2.UNITX = new b2Vec2(1, 0);
b2Vec2.UNITY = new b2Vec2(0, 1);
b2Vec2.s_t0 = new b2Vec2();
b2Vec2.s_t1 = new b2Vec2();
b2Vec2.s_t2 = new b2Vec2();
b2Vec2.s_t3 = new b2Vec2();
return b2Vec2;
}());
var b2Vec2_zero = new b2Vec2(0, 0);
/// A 2D column vector with 3 elements.
var b2Vec3 = /** @class */ (function () {
function b2Vec3(x, y, z) {
if (x === void 0) { x = 0; }
if (y === void 0) { y = 0; }
if (z === void 0) { z = 0; }
this.x = x;
this.y = y;
this.z = z;
}
b2Vec3.prototype.Clone = function () {
return new b2Vec3(this.x, this.y, this.z);
};
b2Vec3.prototype.SetZero = function () {
this.x = 0;
this.y = 0;
this.z = 0;
return this;
};
b2Vec3.prototype.SetXYZ = function (x, y, z) {
this.x = x;
this.y = y;
this.z = z;
return this;
};
b2Vec3.prototype.Copy = function (other) {
this.x = other.x;
this.y = other.y;
this.z = other.z;
return this;
};
b2Vec3.prototype.SelfNeg = function () {
this.x = (-this.x);
this.y = (-this.y);
this.z = (-this.z);
return this;
};
b2Vec3.prototype.SelfAdd = function (v) {
this.x += v.x;
this.y += v.y;
this.z += v.z;
return this;
};
b2Vec3.prototype.SelfAddXYZ = function (x, y, z) {
this.x += x;
this.y += y;
this.z += z;
return this;
};
b2Vec3.prototype.SelfSub = function (v) {
this.x -= v.x;
this.y -= v.y;
this.z -= v.z;
return this;
};
b2Vec3.prototype.SelfSubXYZ = function (x, y, z) {
this.x -= x;
this.y -= y;
this.z -= z;
return this;
};
b2Vec3.prototype.SelfMul = function (s) {
this.x *= s;
this.y *= s;
this.z *= s;
return this;
};
b2Vec3.DotV3V3 = function (a, b) {
return a.x * b.x + a.y * b.y + a.z * b.z;
};
b2Vec3.CrossV3V3 = function (a, b, out) {
var a_x = a.x, a_y = a.y, a_z = a.z;
var b_x = b.x, b_y = b.y, b_z = b.z;
out.x = a_y * b_z - a_z * b_y;
out.y = a_z * b_x - a_x * b_z;
out.z = a_x * b_y - a_y * b_x;
return out;
};
b2Vec3.ZERO = new b2Vec3(0, 0, 0);
b2Vec3.s_t0 = new b2Vec3();
return b2Vec3;
}());
/// A 2-by-2 matrix. Stored in column-major order.
var b2Mat22 = /** @class */ (function () {
function b2Mat22() {
this.ex = new b2Vec2(1, 0);
this.ey = new b2Vec2(0, 1);
}
b2Mat22.prototype.Clone = function () {
return new b2Mat22().Copy(this);
};
b2Mat22.FromVV = function (c1, c2) {
return new b2Mat22().SetVV(c1, c2);
};
b2Mat22.FromSSSS = function (r1c1, r1c2, r2c1, r2c2) {
return new b2Mat22().SetSSSS(r1c1, r1c2, r2c1, r2c2);
};
b2Mat22.FromAngle = function (radians) {
return new b2Mat22().SetAngle(radians);
};
b2Mat22.prototype.SetSSSS = function (r1c1, r1c2, r2c1, r2c2) {
this.ex.Set(r1c1, r2c1);
this.ey.Set(r1c2, r2c2);
return this;
};
b2Mat22.prototype.SetVV = function (c1, c2) {
this.ex.Copy(c1);
this.ey.Copy(c2);
return this;
};
b2Mat22.prototype.SetAngle = function (radians) {
var c = Math.cos(radians);
var s = Math.sin(radians);
this.ex.Set(c, s);
this.ey.Set(-s, c);
return this;
};
b2Mat22.prototype.Copy = function (other) {
this.ex.Copy(other.ex);
this.ey.Copy(other.ey);
return this;
};
b2Mat22.prototype.SetIdentity = function () {
this.ex.Set(1, 0);
this.ey.Set(0, 1);
return this;
};
b2Mat22.prototype.SetZero = function () {
this.ex.SetZero();
this.ey.SetZero();
return this;
};
b2Mat22.prototype.GetAngle = function () {
return Math.atan2(this.ex.y, this.ex.x);
};
b2Mat22.prototype.GetInverse = function (out) {
var a = this.ex.x;
var b = this.ey.x;
var c = this.ex.y;
var d = this.ey.y;
var det = a * d - b * c;
if (det !== 0) {
det = 1 / det;
}
out.ex.x = det * d;
out.ey.x = (-det * b);
out.ex.y = (-det * c);
out.ey.y = det * a;
return out;
};
b2Mat22.prototype.Solve = function (b_x, b_y, out) {
var a11 = this.ex.x, a12 = this.ey.x;
var a21 = this.ex.y, a22 = this.ey.y;
var det = a11 * a22 - a12 * a21;
if (det !== 0) {
det = 1 / det;
}
out.x = det * (a22 * b_x - a12 * b_y);
out.y = det * (a11 * b_y - a21 * b_x);
return out;
};
b2Mat22.prototype.SelfAbs = function () {
this.ex.SelfAbs();
this.ey.SelfAbs();
return this;
};
b2Mat22.prototype.SelfInv = function () {
this.GetInverse(this);
return this;
};
b2Mat22.prototype.SelfAddM = function (M) {
this.ex.SelfAdd(M.ex);
this.ey.SelfAdd(M.ey);
return this;
};
b2Mat22.prototype.SelfSubM = function (M) {
this.ex.SelfSub(M.ex);
this.ey.SelfSub(M.ey);
return this;
};
b2Mat22.AbsM = function (M, out) {
var M_ex = M.ex, M_ey = M.ey;
out.ex.x = b2Abs(M_ex.x);
out.ex.y = b2Abs(M_ex.y);
out.ey.x = b2Abs(M_ey.x);
out.ey.y = b2Abs(M_ey.y);
return out;
};
b2Mat22.MulMV = function (M, v, out) {
var M_ex = M.ex, M_ey = M.ey;
var v_x = v.x, v_y = v.y;
out.x = M_ex.x * v_x + M_ey.x * v_y;
out.y = M_ex.y * v_x + M_ey.y * v_y;
return out;
};
b2Mat22.MulTMV = function (M, v, out) {
var M_ex = M.ex, M_ey = M.ey;
var v_x = v.x, v_y = v.y;
out.x = M_ex.x * v_x + M_ex.y * v_y;
out.y = M_ey.x * v_x + M_ey.y * v_y;
return out;
};
b2Mat22.AddMM = function (A, B, out) {
var A_ex = A.ex, A_ey = A.ey;
var B_ex = B.ex, B_ey = B.ey;
out.ex.x = A_ex.x + B_ex.x;
out.ex.y = A_ex.y + B_ex.y;
out.ey.x = A_ey.x + B_ey.x;
out.ey.y = A_ey.y + B_ey.y;
return out;
};
b2Mat22.MulMM = function (A, B, out) {
var A_ex_x = A.ex.x, A_ex_y = A.ex.y;
var A_ey_x = A.ey.x, A_ey_y = A.ey.y;
var B_ex_x = B.ex.x, B_ex_y = B.ex.y;
var B_ey_x = B.ey.x, B_ey_y = B.ey.y;
out.ex.x = A_ex_x * B_ex_x + A_ey_x * B_ex_y;
out.ex.y = A_ex_y * B_ex_x + A_ey_y * B_ex_y;
out.ey.x = A_ex_x * B_ey_x + A_ey_x * B_ey_y;
out.ey.y = A_ex_y * B_ey_x + A_ey_y * B_ey_y;
return out;
};
b2Mat22.MulTMM = function (A, B, out) {
var A_ex_x = A.ex.x, A_ex_y = A.ex.y;
var A_ey_x = A.ey.x, A_ey_y = A.ey.y;
var B_ex_x = B.ex.x, B_ex_y = B.ex.y;
var B_ey_x = B.ey.x, B_ey_y = B.ey.y;
out.ex.x = A_ex_x * B_ex_x + A_ex_y * B_ex_y;
out.ex.y = A_ey_x * B_ex_x + A_ey_y * B_ex_y;
out.ey.x = A_ex_x * B_ey_x + A_ex_y * B_ey_y;
out.ey.y = A_ey_x * B_ey_x + A_ey_y * B_ey_y;
return out;
};
b2Mat22.IDENTITY = new b2Mat22();
return b2Mat22;
}());
/// A 3-by-3 matrix. Stored in column-major order.
var b2Mat33 = /** @class */ (function () {
function b2Mat33() {
this.ex = new b2Vec3(1, 0, 0);
this.ey = new b2Vec3(0, 1, 0);
this.ez = new b2Vec3(0, 0, 1);
}
b2Mat33.prototype.Clone = function () {
return new b2Mat33().Copy(this);
};
b2Mat33.prototype.SetVVV = function (c1, c2, c3) {
this.ex.Copy(c1);
this.ey.Copy(c2);
this.ez.Copy(c3);
return this;
};
b2Mat33.prototype.Copy = function (other) {
this.ex.Copy(other.ex);
this.ey.Copy(other.ey);
this.ez.Copy(other.ez);
return this;
};
b2Mat33.prototype.SetIdentity = function () {
this.ex.SetXYZ(1, 0, 0);
this.ey.SetXYZ(0, 1, 0);
this.ez.SetXYZ(0, 0, 1);
return this;
};
b2Mat33.prototype.SetZero = function () {
this.ex.SetZero();
this.ey.SetZero();
this.ez.SetZero();
return this;
};
b2Mat33.prototype.SelfAddM = function (M) {
this.ex.SelfAdd(M.ex);
this.ey.SelfAdd(M.ey);
this.ez.SelfAdd(M.ez);
return this;
};
b2Mat33.prototype.Solve33 = function (b_x, b_y, b_z, out) {
var a11 = this.ex.x, a21 = this.ex.y, a31 = this.ex.z;
var a12 = this.ey.x, a22 = this.ey.y, a32 = this.ey.z;
var a13 = this.ez.x, a23 = this.ez.y, a33 = this.ez.z;
var det = a11 * (a22 * a33 - a32 * a23) + a21 * (a32 * a13 - a12 * a33) + a31 * (a12 * a23 - a22 * a13);
if (det !== 0) {
det = 1 / det;
}
out.x = det * (b_x * (a22 * a33 - a32 * a23) + b_y * (a32 * a13 - a12 * a33) + b_z * (a12 * a23 - a22 * a13));
out.y = det * (a11 * (b_y * a33 - b_z * a23) + a21 * (b_z * a13 - b_x * a33) + a31 * (b_x * a23 - b_y * a13));
out.z = det * (a11 * (a22 * b_z - a32 * b_y) + a21 * (a32 * b_x - a12 * b_z) + a31 * (a12 * b_y - a22 * b_x));
return out;
};
b2Mat33.prototype.Solve22 = function (b_x, b_y, out) {
var a11 = this.ex.x, a12 = this.ey.x;
var a21 = this.ex.y, a22 = this.ey.y;
var det = a11 * a22 - a12 * a21;
if (det !== 0) {
det = 1 / det;
}
out.x = det * (a22 * b_x - a12 * b_y);
out.y = det * (a11 * b_y - a21 * b_x);
return out;
};
b2Mat33.prototype.GetInverse22 = function (M) {
var a = this.ex.x, b = this.ey.x, c = this.ex.y, d = this.ey.y;
var det = a * d - b * c;
if (det !== 0) {
det = 1 / det;
}
M.ex.x = det * d;
M.ey.x = -det * b;
M.ex.z = 0;
M.ex.y = -det * c;
M.ey.y = det * a;
M.ey.z = 0;
M.ez.x = 0;
M.ez.y = 0;
M.ez.z = 0;
};
b2Mat33.prototype.GetSymInverse33 = function (M) {
var det = b2Vec3.DotV3V3(this.ex, b2Vec3.CrossV3V3(this.ey, this.ez, b2Vec3.s_t0));
if (det !== 0) {
det = 1 / det;
}
var a11 = this.ex.x, a12 = this.ey.x, a13 = this.ez.x;
var a22 = this.ey.y, a23 = this.ez.y;
var a33 = this.ez.z;
M.ex.x = det * (a22 * a33 - a23 * a23);
M.ex.y = det * (a13 * a23 - a12 * a33);
M.ex.z = det * (a12 * a23 - a13 * a22);
M.ey.x = M.ex.y;
M.ey.y = det * (a11 * a33 - a13 * a13);
M.ey.z = det * (a13 * a12 - a11 * a23);
M.ez.x = M.ex.z;
M.ez.y = M.ey.z;
M.ez.z = det * (a11 * a22 - a12 * a12);
};
b2Mat33.MulM33V3 = function (A, v, out) {
var v_x = v.x, v_y = v.y, v_z = v.z;
out.x = A.ex.x * v_x + A.ey.x * v_y + A.ez.x * v_z;
out.y = A.ex.y * v_x + A.ey.y * v_y + A.ez.y * v_z;
out.z = A.ex.z * v_x + A.ey.z * v_y + A.ez.z * v_z;
return out;
};
b2Mat33.MulM33XYZ = function (A, x, y, z, out) {
out.x = A.ex.x * x + A.ey.x * y + A.ez.x * z;
out.y = A.ex.y * x + A.ey.y * y + A.ez.y * z;
out.z = A.ex.z * x + A.ey.z * y + A.ez.z * z;
return out;
};
b2Mat33.MulM33V2 = function (A, v, out) {
var v_x = v.x, v_y = v.y;
out.x = A.ex.x * v_x + A.ey.x * v_y;
out.y = A.ex.y * v_x + A.ey.y * v_y;
return out;
};
b2Mat33.MulM33XY = function (A, x, y, out) {
out.x = A.ex.x * x + A.ey.x * y;
out.y = A.ex.y * x + A.ey.y * y;
return out;
};
b2Mat33.IDENTITY = new b2Mat33();
return b2Mat33;
}());
/// Rotation
var b2Rot = /** @class */ (function () {
function b2Rot(angle) {
if (angle === void 0) { angle = 0; }
this.s = 0;
this.c = 1;
if (angle) {
this.s = Math.sin(angle);
this.c = Math.cos(angle);
}
}
b2Rot.prototype.Clone = function () {
return new b2Rot().Copy(this);
};
b2Rot.prototype.Copy = function (other) {
this.s = other.s;
this.c = other.c;
return this;
};
b2Rot.prototype.SetAngle = function (angle) {
this.s = Math.sin(angle);
this.c = Math.cos(angle);
return this;
};
b2Rot.prototype.SetIdentity = function () {
this.s = 0;
this.c = 1;
return this;
};
b2Rot.prototype.GetAngle = function () {
return Math.atan2(this.s, this.c);
};
b2Rot.prototype.GetXAxis = function (out) {
out.x = this.c;
out.y = this.s;
return out;
};
b2Rot.prototype.GetYAxis = function (out) {
out.x = -this.s;
out.y = this.c;
return out;
};
b2Rot.MulRR = function (q, r, out) {
// [qc -qs] * [rc -rs] = [qc*rc-qs*rs -qc*rs-qs*rc]
// [qs qc] [rs rc] [qs*rc+qc*rs -qs*rs+qc*rc]
// s = qs * rc + qc * rs
// c = qc * rc - qs * rs
var q_c = q.c, q_s = q.s;
var r_c = r.c, r_s = r.s;
out.s = q_s * r_c + q_c * r_s;
out.c = q_c * r_c - q_s * r_s;
return out;
};
b2Rot.MulTRR = function (q, r, out) {
// [ qc qs] * [rc -rs] = [qc*rc+qs*rs -qc*rs+qs*rc]
// [-qs qc] [rs rc] [-qs*rc+qc*rs qs*rs+qc*rc]
// s = qc * rs - qs * rc
// c = qc * rc + qs * rs
var q_c = q.c, q_s = q.s;
var r_c = r.c, r_s = r.s;
out.s = q_c * r_s - q_s * r_c;
out.c = q_c * r_c + q_s * r_s;
return out;
};
b2Rot.MulRV = function (q, v, out) {
var q_c = q.c, q_s = q.s;
var v_x = v.x, v_y = v.y;
out.x = q_c * v_x - q_s * v_y;
out.y = q_s * v_x + q_c * v_y;
return out;
};
b2Rot.MulTRV = function (q, v, out) {
var q_c = q.c, q_s = q.s;
var v_x = v.x, v_y = v.y;
out.x = q_c * v_x + q_s * v_y;
out.y = -q_s * v_x + q_c * v_y;
return out;
};
b2Rot.IDENTITY = new b2Rot();
return b2Rot;
}());
/// A transform contains translation and rotation. It is used to represent
/// the position and orientation of rigid frames.
var b2Transform = /** @class */ (function () {
function b2Transform() {
this.p = new b2Vec2();
this.q = new b2Rot();
}
b2Transform.prototype.Clone = function () {
return new b2Transform().Copy(this);
};
b2Transform.prototype.Copy = function (other) {
this.p.Copy(other.p);
this.q.Copy(other.q);
return this;
};
b2Transform.prototype.SetIdentity = function () {
this.p.SetZero();
this.q.SetIdentity();
return this;
};
b2Transform.prototype.SetPositionRotation = function (position, q) {
this.p.Copy(position);
this.q.Copy(q);
return this;
};
b2Transform.prototype.SetPositionAngle = function (pos, a) {
this.p.Copy(pos);
this.q.SetAngle(a);
return this;
};
b2Transform.prototype.SetPosition = function (position) {
this.p.Copy(position);
return this;
};
b2Transform.prototype.SetPositionXY = function (x, y) {
this.p.Set(x, y);
return this;
};
b2Transform.prototype.SetRotation = function (rotation) {
this.q.Copy(rotation);
return this;
};
b2Transform.prototype.SetRotationAngle = function (radians) {
this.q.SetAngle(radians);
return this;
};
b2Transform.prototype.GetPosition = function () {
return this.p;
};
b2Transform.prototype.GetRotation = function () {
return this.q;
};
b2Transform.prototype.GetRotationAngle = function () {
return this.q.GetAngle();
};
b2Transform.prototype.GetAngle = function () {
return this.q.GetAngle();
};
b2Transform.MulXV = function (T, v, out) {
// float32 x = (T.q.c * v.x - T.q.s * v.y) + T.p.x;
// float32 y = (T.q.s * v.x + T.q.c * v.y) + T.p.y;
// return b2Vec2(x, y);
var T_q_c = T.q.c, T_q_s = T.q.s;
var v_x = v.x, v_y = v.y;
out.x = (T_q_c * v_x - T_q_s * v_y) + T.p.x;
out.y = (T_q_s * v_x + T_q_c * v_y) + T.p.y;
return out;
};
b2Transform.MulTXV = function (T, v, out) {
// float32 px = v.x - T.p.x;
// float32 py = v.y - T.p.y;
// float32 x = (T.q.c * px + T.q.s * py);
// float32 y = (-T.q.s * px + T.q.c * py);
// return b2Vec2(x, y);
var T_q_c = T.q.c, T_q_s = T.q.s;
var p_x = v.x - T.p.x;
var p_y = v.y - T.p.y;
out.x = (T_q_c * p_x + T_q_s * p_y);
out.y = (-T_q_s * p_x + T_q_c * p_y);
return out;
};
b2Transform.MulXX = function (A, B, out) {
b2Rot.MulRR(A.q, B.q, out.q);
b2Vec2.AddVV(b2Rot.MulRV(A.q, B.p, out.p), A.p, out.p);
return out;
};
b2Transform.MulTXX = function (A, B, out) {
b2Rot.MulTRR(A.q, B.q, out.q);
b2Rot.MulTRV(A.q, b2Vec2.SubVV(B.p, A.p, out.p), out.p);
return out;
};
b2Transform.IDENTITY = new b2Transform();
return b2Transform;
}());
/// This describes the motion of a body/shape for TOI computation.
/// Shapes are defined with respect to the body origin, which may
/// no coincide with the center of mass. However, to support dynamics
/// we must interpolate the center of mass position.
var b2Sweep = /** @class */ (function () {
function b2Sweep() {
this.localCenter = new b2Vec2();
this.c0 = new b2Vec2();
this.c = new b2Vec2();
this.a0 = 0;
this.a = 0;
this.alpha0 = 0;
}
b2Sweep.prototype.Clone = function () {
return new b2Sweep().Copy(this);
};
b2Sweep.prototype.Copy = function (other) {
this.localCenter.Copy(other.localCenter);
this.c0.Copy(other.c0);
this.c.Copy(other.c);
this.a0 = other.a0;
this.a = other.a;
this.alpha0 = other.alpha0;
return this;
};
b2Sweep.prototype.GetTransform = function (xf, beta) {
var one_minus_beta = (1 - beta);
xf.p.x = one_minus_beta * this.c0.x + beta * this.c.x;
xf.p.y = one_minus_beta * this.c0.y + beta * this.c.y;
var angle = one_minus_beta * this.a0 + beta * this.a;
xf.q.SetAngle(angle);
xf.p.SelfSub(b2Rot.MulRV(xf.q, this.localCenter, b2Vec2.s_t0));
return xf;
};
b2Sweep.prototype.Advance = function (alpha) {
// DEBUG: b2Assert(this.alpha0 < 1);
var beta = (alpha - this.alpha0) / (1 - this.alpha0);
var one_minus_beta = (1 - beta);
this.c0.x = one_minus_beta * this.c0.x + beta * this.c.x;
this.c0.y = one_minus_beta * this.c0.y + beta * this.c.y;
this.a0 = one_minus_beta * this.a0 + beta * this.a;
this.alpha0 = alpha;
};
b2Sweep.prototype.Normalize = function () {
var d = b2_two_pi * Math.floor(this.a0 / b2_two_pi);
this.a0 -= d;
this.a -= d;
};
return b2Sweep;
}());
/*
* Copyright (c) 2011 Erin Catto http://box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Color for debug drawing. Each value has the range [0,1].
var b2Color = /** @class */ (function () {
function b2Color(rr, gg, bb, aa) {
if (rr === void 0) { rr = 0.5; }
if (gg === void 0) { gg = 0.5; }
if (bb === void 0) { bb = 0.5; }
if (aa === void 0) { aa = 1.0; }
this.r = rr;
this.g = gg;
this.b = bb;
this.a = aa;
}
b2Color.prototype.Clone = function () {
return new b2Color().Copy(this);
};
b2Color.prototype.Copy = function (other) {
this.r = other.r;
this.g = other.g;
this.b = other.b;
this.a = other.a;
return this;
};
b2Color.prototype.IsEqual = function (color) {
return (this.r === color.r) && (this.g === color.g) && (this.b === color.b) && (this.a === color.a);
};
b2Color.prototype.IsZero = function () {
return (this.r === 0) && (this.g === 0) && (this.b === 0) && (this.a === 0);
};
b2Color.prototype.Set = function (r, g, b, a) {
if (a === void 0) { a = this.a; }
this.SetRGBA(r, g, b, a);
};
b2Color.prototype.SetByteRGB = function (r, g, b) {
this.r = r / 0xff;
this.g = g / 0xff;
this.b = b / 0xff;
return this;
};
b2Color.prototype.SetByteRGBA = function (r, g, b, a) {
this.r = r / 0xff;
this.g = g / 0xff;
this.b = b / 0xff;
this.a = a / 0xff;
return this;
};
b2Color.prototype.SetRGB = function (rr, gg, bb) {
this.r = rr;
this.g = gg;
this.b = bb;
return this;
};
b2Color.prototype.SetRGBA = function (rr, gg, bb, aa) {
this.r = rr;
this.g = gg;
this.b = bb;
this.a = aa;
return this;
};
b2Color.prototype.SelfAdd = function (color) {
this.r += color.r;
this.g += color.g;
this.b += color.b;
this.a += color.a;
return this;
};
b2Color.prototype.Add = function (color, out) {
out.r = this.r + color.r;
out.g = this.g + color.g;
out.b = this.b + color.b;
out.a = this.a + color.a;
return out;
};
b2Color.prototype.SelfSub = function (color) {
this.r -= color.r;
this.g -= color.g;
this.b -= color.b;
this.a -= color.a;
return this;
};
b2Color.prototype.Sub = function (color, out) {
out.r = this.r - color.r;
out.g = this.g - color.g;
out.b = this.b - color.b;
out.a = this.a - color.a;
return out;
};
b2Color.prototype.SelfMul = function (s) {
this.r *= s;
this.g *= s;
this.b *= s;
this.a *= s;
return this;
};
b2Color.prototype.Mul = function (s, out) {
out.r = this.r * s;
out.g = this.g * s;
out.b = this.b * s;
out.a = this.a * s;
return out;
};
b2Color.prototype.Mix = function (mixColor, strength) {
b2Color.MixColors(this, mixColor, strength);
};
b2Color.MixColors = function (colorA, colorB, strength) {
var dr = (strength * (colorB.r - colorA.r));
var dg = (strength * (colorB.g - colorA.g));
var db = (strength * (colorB.b - colorA.b));
var da = (strength * (colorB.a - colorA.a));
colorA.r += dr;
colorA.g += dg;
colorA.b += db;
colorA.a += da;
colorB.r -= dr;
colorB.g -= dg;
colorB.b -= db;
colorB.a -= da;
};
b2Color.prototype.MakeStyleString = function (alpha) {
if (alpha === void 0) { alpha = this.a; }
return b2Color.MakeStyleString(this.r, this.g, this.b, alpha);
};
b2Color.MakeStyleString = function (r, g, b, a) {
if (a === void 0) { a = 1.0; }
// function clamp(x: number, lo: number, hi: number) { return x < lo ? lo : hi < x ? hi : x; }
r *= 255; // r = clamp(r, 0, 255);
g *= 255; // g = clamp(g, 0, 255);
b *= 255; // b = clamp(b, 0, 255);
// a = clamp(a, 0, 1);
if (a < 1) {
return "rgba(" + r + "," + g + "," + b + "," + a + ")";
}
else {
return "rgb(" + r + "," + g + "," + b + ")";
}
};
b2Color.ZERO = new b2Color(0, 0, 0, 0);
b2Color.RED = new b2Color(1, 0, 0);
b2Color.GREEN = new b2Color(0, 1, 0);
b2Color.BLUE = new b2Color(0, 0, 1);
return b2Color;
}());
(function (b2DrawFlags) {
b2DrawFlags[b2DrawFlags["e_none"] = 0] = "e_none";
b2DrawFlags[b2DrawFlags["e_shapeBit"] = 1] = "e_shapeBit";
b2DrawFlags[b2DrawFlags["e_jointBit"] = 2] = "e_jointBit";
b2DrawFlags[b2DrawFlags["e_aabbBit"] = 4] = "e_aabbBit";
b2DrawFlags[b2DrawFlags["e_pairBit"] = 8] = "e_pairBit";
b2DrawFlags[b2DrawFlags["e_centerOfMassBit"] = 16] = "e_centerOfMassBit";
// #if B2_ENABLE_PARTICLE
b2DrawFlags[b2DrawFlags["e_particleBit"] = 32] = "e_particleBit";
// #endif
b2DrawFlags[b2DrawFlags["e_controllerBit"] = 64] = "e_controllerBit";
b2DrawFlags[b2DrawFlags["e_all"] = 63] = "e_all";
})(exports.b2DrawFlags || (exports.b2DrawFlags = {}));
/// Implement and register this class with a b2World to provide debug drawing of physics
/// entities in your game.
var b2Draw = /** @class */ (function () {
function b2Draw() {
this.m_drawFlags = 0;
}
b2Draw.prototype.SetFlags = function (flags) {
this.m_drawFlags = flags;
};
b2Draw.prototype.GetFlags = function () {
return this.m_drawFlags;
};
b2Draw.prototype.AppendFlags = function (flags) {
this.m_drawFlags |= flags;
};
b2Draw.prototype.ClearFlags = function (flags) {
this.m_drawFlags &= ~flags;
};
return b2Draw;
}());
/*
* Copyright (c) 2011 Erin Catto http://box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Timer for profiling. This has platform specific code and may
/// not work on every platform.
var b2Timer = /** @class */ (function () {
function b2Timer() {
this.m_start = Date.now();
}
/// Reset the timer.
b2Timer.prototype.Reset = function () {
this.m_start = Date.now();
return this;
};
/// Get the time since construction or the last reset.
b2Timer.prototype.GetMilliseconds = function () {
return Date.now() - this.m_start;
};
return b2Timer;
}());
var b2Counter = /** @class */ (function () {
function b2Counter() {
this.m_count = 0;
this.m_min_count = 0;
this.m_max_count = 0;
}
b2Counter.prototype.GetCount = function () {
return this.m_count;
};
b2Counter.prototype.GetMinCount = function () {
return this.m_min_count;
};
b2Counter.prototype.GetMaxCount = function () {
return this.m_max_count;
};
b2Counter.prototype.ResetCount = function () {
var count = this.m_count;
this.m_count = 0;
return count;
};
b2Counter.prototype.ResetMinCount = function () {
this.m_min_count = 0;
};
b2Counter.prototype.ResetMaxCount = function () {
this.m_max_count = 0;
};
b2Counter.prototype.Increment = function () {
this.m_count++;
if (this.m_max_count < this.m_count) {
this.m_max_count = this.m_count;
}
};
b2Counter.prototype.Decrement = function () {
this.m_count--;
if (this.m_min_count > this.m_count) {
this.m_min_count = this.m_count;
}
};
return b2Counter;
}());
/*
* Copyright (c) 2010 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// This is a growable LIFO stack with an initial capacity of N.
/// If the stack size exceeds the initial capacity, the heap is used
/// to increase the size of the stack.
var b2GrowableStack = /** @class */ (function () {
function b2GrowableStack(N) {
this.m_stack = [];
this.m_count = 0;
this.m_stack = b2MakeArray(N, function (index) { return null; });
this.m_count = 0;
}
b2GrowableStack.prototype.Reset = function () {
this.m_count = 0;
return this;
};
b2GrowableStack.prototype.Push = function (element) {
this.m_stack[this.m_count] = element;
this.m_count++;
};
b2GrowableStack.prototype.Pop = function () {
// DEBUG: b2Assert(this.m_count > 0);
this.m_count--;
var element = this.m_stack[this.m_count];
this.m_stack[this.m_count] = null;
if (element === null) {
throw new Error();
}
return element;
};
b2GrowableStack.prototype.GetCount = function () {
return this.m_count;
};
return b2GrowableStack;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2BlockAllocator = /** @class */ (function () {
function b2BlockAllocator() {
}
return b2BlockAllocator;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2StackAllocator = /** @class */ (function () {
function b2StackAllocator() {
}
return b2StackAllocator;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// A distance proxy is used by the GJK algorithm.
/// It encapsulates any shape.
var b2DistanceProxy = /** @class */ (function () {
function b2DistanceProxy() {
this.m_buffer = b2Vec2.MakeArray(2);
this.m_vertices = this.m_buffer;
this.m_count = 0;
this.m_radius = 0;
}
b2DistanceProxy.prototype.Copy = function (other) {
if (other.m_vertices === other.m_buffer) {
this.m_vertices = this.m_buffer;
this.m_buffer[0].Copy(other.m_buffer[0]);
this.m_buffer[1].Copy(other.m_buffer[1]);
}
else {
this.m_vertices = other.m_vertices;
}
this.m_count = other.m_count;
this.m_radius = other.m_radius;
return this;
};
b2DistanceProxy.prototype.Reset = function () {
this.m_vertices = this.m_buffer;
this.m_count = 0;
this.m_radius = 0;
return this;
};
b2DistanceProxy.prototype.SetShape = function (shape, index) {
shape.SetupDistanceProxy(this, index);
};
b2DistanceProxy.prototype.SetVerticesRadius = function (vertices, count, radius) {
this.m_vertices = vertices;
this.m_count = count;
this.m_radius = radius;
};
b2DistanceProxy.prototype.GetSupport = function (d) {
var bestIndex = 0;
var bestValue = b2Vec2.DotVV(this.m_vertices[0], d);
for (var i = 1; i < this.m_count; ++i) {
var value = b2Vec2.DotVV(this.m_vertices[i], d);
if (value > bestValue) {
bestIndex = i;
bestValue = value;
}
}
return bestIndex;
};
b2DistanceProxy.prototype.GetSupportVertex = function (d) {
var bestIndex = 0;
var bestValue = b2Vec2.DotVV(this.m_vertices[0], d);
for (var i = 1; i < this.m_count; ++i) {
var value = b2Vec2.DotVV(this.m_vertices[i], d);
if (value > bestValue) {
bestIndex = i;
bestValue = value;
}
}
return this.m_vertices[bestIndex];
};
b2DistanceProxy.prototype.GetVertexCount = function () {
return this.m_count;
};
b2DistanceProxy.prototype.GetVertex = function (index) {
// DEBUG: b2Assert(0 <= index && index < this.m_count);
return this.m_vertices[index];
};
return b2DistanceProxy;
}());
var b2SimplexCache = /** @class */ (function () {
function b2SimplexCache() {
this.metric = 0;
this.count = 0;
this.indexA = [0, 0, 0];
this.indexB = [0, 0, 0];
}
b2SimplexCache.prototype.Reset = function () {
this.metric = 0;
this.count = 0;
return this;
};
return b2SimplexCache;
}());
var b2DistanceInput = /** @class */ (function () {
function b2DistanceInput() {
this.proxyA = new b2DistanceProxy();
this.proxyB = new b2DistanceProxy();
this.transformA = new b2Transform();
this.transformB = new b2Transform();
this.useRadii = false;
}
b2DistanceInput.prototype.Reset = function () {
this.proxyA.Reset();
this.proxyB.Reset();
this.transformA.SetIdentity();
this.transformB.SetIdentity();
this.useRadii = false;
return this;
};
return b2DistanceInput;
}());
var b2DistanceOutput = /** @class */ (function () {
function b2DistanceOutput() {
this.pointA = new b2Vec2();
this.pointB = new b2Vec2();
this.distance = 0;
this.iterations = 0; ///< number of GJK iterations used
}
b2DistanceOutput.prototype.Reset = function () {
this.pointA.SetZero();
this.pointB.SetZero();
this.distance = 0;
this.iterations = 0;
return this;
};
return b2DistanceOutput;
}());
/// Input parameters for b2ShapeCast
var b2ShapeCastInput = /** @class */ (function () {
function b2ShapeCastInput() {
this.proxyA = new b2DistanceProxy();
this.proxyB = new b2DistanceProxy();
this.transformA = new b2Transform();
this.transformB = new b2Transform();
this.translationB = new b2Vec2();
}
return b2ShapeCastInput;
}());
/// Output results for b2ShapeCast
var b2ShapeCastOutput = /** @class */ (function () {
function b2ShapeCastOutput() {
this.point = new b2Vec2();
this.normal = new b2Vec2();
this.lambda = 0.0;
this.iterations = 0;
}
return b2ShapeCastOutput;
}());
exports.b2_gjkCalls = 0;
exports.b2_gjkIters = 0;
exports.b2_gjkMaxIters = 0;
function b2_gjk_reset() {
exports.b2_gjkCalls = 0;
exports.b2_gjkIters = 0;
exports.b2_gjkMaxIters = 0;
}
var b2SimplexVertex = /** @class */ (function () {
function b2SimplexVertex() {
this.wA = new b2Vec2(); // support point in proxyA
this.wB = new b2Vec2(); // support point in proxyB
this.w = new b2Vec2(); // wB - wA
this.a = 0; // barycentric coordinate for closest point
this.indexA = 0; // wA index
this.indexB = 0; // wB index
}
b2SimplexVertex.prototype.Copy = function (other) {
this.wA.Copy(other.wA); // support point in proxyA
this.wB.Copy(other.wB); // support point in proxyB
this.w.Copy(other.w); // wB - wA
this.a = other.a; // barycentric coordinate for closest point
this.indexA = other.indexA; // wA index
this.indexB = other.indexB; // wB index
return this;
};
return b2SimplexVertex;
}());
var b2Simplex = /** @class */ (function () {
function b2Simplex() {
this.m_v1 = new b2SimplexVertex();
this.m_v2 = new b2SimplexVertex();
this.m_v3 = new b2SimplexVertex();
this.m_vertices = [ /*3*/];
this.m_count = 0;
this.m_vertices[0] = this.m_v1;
this.m_vertices[1] = this.m_v2;
this.m_vertices[2] = this.m_v3;
}
b2Simplex.prototype.ReadCache = function (cache, proxyA, transformA, proxyB, transformB) {
// DEBUG: b2Assert(0 <= cache.count && cache.count <= 3);
// Copy data from cache.
this.m_count = cache.count;
var vertices = this.m_vertices;
for (var i = 0; i < this.m_count; ++i) {
var v = vertices[i];
v.indexA = cache.indexA[i];
v.indexB = cache.indexB[i];
var wALocal = proxyA.GetVertex(v.indexA);
var wBLocal = proxyB.GetVertex(v.indexB);
b2Transform.MulXV(transformA, wALocal, v.wA);
b2Transform.MulXV(transformB, wBLocal, v.wB);
b2Vec2.SubVV(v.wB, v.wA, v.w);
v.a = 0;
}
// Compute the new simplex metric, if it is substantially different than
// old metric then flush the simplex.
if (this.m_count > 1) {
var metric1 = cache.metric;
var metric2 = this.GetMetric();
if (metric2 < 0.5 * metric1 || 2 * metric1 < metric2 || metric2 < b2_epsilon) {
// Reset the simplex.
this.m_count = 0;
}
}
// If the cache is empty or invalid ...
if (this.m_count === 0) {
var v = vertices[0];
v.indexA = 0;
v.indexB = 0;
var wALocal = proxyA.GetVertex(0);
var wBLocal = proxyB.GetVertex(0);
b2Transform.MulXV(transformA, wALocal, v.wA);
b2Transform.MulXV(transformB, wBLocal, v.wB);
b2Vec2.SubVV(v.wB, v.wA, v.w);
v.a = 1;
this.m_count = 1;
}
};
b2Simplex.prototype.WriteCache = function (cache) {
cache.metric = this.GetMetric();
cache.count = this.m_count;
var vertices = this.m_vertices;
for (var i = 0; i < this.m_count; ++i) {
cache.indexA[i] = vertices[i].indexA;
cache.indexB[i] = vertices[i].indexB;
}
};
b2Simplex.prototype.GetSearchDirection = function (out) {
switch (this.m_count) {
case 1:
return b2Vec2.NegV(this.m_v1.w, out);
case 2: {
var e12 = b2Vec2.SubVV(this.m_v2.w, this.m_v1.w, out);
var sgn = b2Vec2.CrossVV(e12, b2Vec2.NegV(this.m_v1.w, b2Vec2.s_t0));
if (sgn > 0) {
// Origin is left of e12.
return b2Vec2.CrossOneV(e12, out);
}
else {
// Origin is right of e12.
return b2Vec2.CrossVOne(e12, out);
}
}
default:
// DEBUG: b2Assert(false);
return out.SetZero();
}
};
b2Simplex.prototype.GetClosestPoint = function (out) {
switch (this.m_count) {
case 0:
// DEBUG: b2Assert(false);
return out.SetZero();
case 1:
return out.Copy(this.m_v1.w);
case 2:
return out.Set(this.m_v1.a * this.m_v1.w.x + this.m_v2.a * this.m_v2.w.x, this.m_v1.a * this.m_v1.w.y + this.m_v2.a * this.m_v2.w.y);
case 3:
return out.SetZero();
default:
// DEBUG: b2Assert(false);
return out.SetZero();
}
};
b2Simplex.prototype.GetWitnessPoints = function (pA, pB) {
switch (this.m_count) {
case 0:
// DEBUG: b2Assert(false);
break;
case 1:
pA.Copy(this.m_v1.wA);
pB.Copy(this.m_v1.wB);
break;
case 2:
pA.x = this.m_v1.a * this.m_v1.wA.x + this.m_v2.a * this.m_v2.wA.x;
pA.y = this.m_v1.a * this.m_v1.wA.y + this.m_v2.a * this.m_v2.wA.y;
pB.x = this.m_v1.a * this.m_v1.wB.x + this.m_v2.a * this.m_v2.wB.x;
pB.y = this.m_v1.a * this.m_v1.wB.y + this.m_v2.a * this.m_v2.wB.y;
break;
case 3:
pB.x = pA.x = this.m_v1.a * this.m_v1.wA.x + this.m_v2.a * this.m_v2.wA.x + this.m_v3.a * this.m_v3.wA.x;
pB.y = pA.y = this.m_v1.a * this.m_v1.wA.y + this.m_v2.a * this.m_v2.wA.y + this.m_v3.a * this.m_v3.wA.y;
break;
default:
// DEBUG: b2Assert(false);
break;
}
};
b2Simplex.prototype.GetMetric = function () {
switch (this.m_count) {
case 0:
// DEBUG: b2Assert(false);
return 0;
case 1:
return 0;
case 2:
return b2Vec2.DistanceVV(this.m_v1.w, this.m_v2.w);
case 3:
return b2Vec2.CrossVV(b2Vec2.SubVV(this.m_v2.w, this.m_v1.w, b2Vec2.s_t0), b2Vec2.SubVV(this.m_v3.w, this.m_v1.w, b2Vec2.s_t1));
default:
// DEBUG: b2Assert(false);
return 0;
}
};
b2Simplex.prototype.Solve2 = function () {
var w1 = this.m_v1.w;
var w2 = this.m_v2.w;
var e12 = b2Vec2.SubVV(w2, w1, b2Simplex.s_e12);
// w1 region
var d12_2 = (-b2Vec2.DotVV(w1, e12));
if (d12_2 <= 0) {
// a2 <= 0, so we clamp it to 0
this.m_v1.a = 1;
this.m_count = 1;
return;
}
// w2 region
var d12_1 = b2Vec2.DotVV(w2, e12);
if (d12_1 <= 0) {
// a1 <= 0, so we clamp it to 0
this.m_v2.a = 1;
this.m_count = 1;
this.m_v1.Copy(this.m_v2);
return;
}
// Must be in e12 region.
var inv_d12 = 1 / (d12_1 + d12_2);
this.m_v1.a = d12_1 * inv_d12;
this.m_v2.a = d12_2 * inv_d12;
this.m_count = 2;
};
b2Simplex.prototype.Solve3 = function () {
var w1 = this.m_v1.w;
var w2 = this.m_v2.w;
var w3 = this.m_v3.w;
// Edge12
// [1 1 ][a1] = [1]
// [w1.e12 w2.e12][a2] = [0]
// a3 = 0
var e12 = b2Vec2.SubVV(w2, w1, b2Simplex.s_e12);
var w1e12 = b2Vec2.DotVV(w1, e12);
var w2e12 = b2Vec2.DotVV(w2, e12);
var d12_1 = w2e12;
var d12_2 = (-w1e12);
// Edge13
// [1 1 ][a1] = [1]
// [w1.e13 w3.e13][a3] = [0]
// a2 = 0
var e13 = b2Vec2.SubVV(w3, w1, b2Simplex.s_e13);
var w1e13 = b2Vec2.DotVV(w1, e13);
var w3e13 = b2Vec2.DotVV(w3, e13);
var d13_1 = w3e13;
var d13_2 = (-w1e13);
// Edge23
// [1 1 ][a2] = [1]
// [w2.e23 w3.e23][a3] = [0]
// a1 = 0
var e23 = b2Vec2.SubVV(w3, w2, b2Simplex.s_e23);
var w2e23 = b2Vec2.DotVV(w2, e23);
var w3e23 = b2Vec2.DotVV(w3, e23);
var d23_1 = w3e23;
var d23_2 = (-w2e23);
// Triangle123
var n123 = b2Vec2.CrossVV(e12, e13);
var d123_1 = n123 * b2Vec2.CrossVV(w2, w3);
var d123_2 = n123 * b2Vec2.CrossVV(w3, w1);
var d123_3 = n123 * b2Vec2.CrossVV(w1, w2);
// w1 region
if (d12_2 <= 0 && d13_2 <= 0) {
this.m_v1.a = 1;
this.m_count = 1;
return;
}
// e12
if (d12_1 > 0 && d12_2 > 0 && d123_3 <= 0) {
var inv_d12 = 1 / (d12_1 + d12_2);
this.m_v1.a = d12_1 * inv_d12;
this.m_v2.a = d12_2 * inv_d12;
this.m_count = 2;
return;
}
// e13
if (d13_1 > 0 && d13_2 > 0 && d123_2 <= 0) {
var inv_d13 = 1 / (d13_1 + d13_2);
this.m_v1.a = d13_1 * inv_d13;
this.m_v3.a = d13_2 * inv_d13;
this.m_count = 2;
this.m_v2.Copy(this.m_v3);
return;
}
// w2 region
if (d12_1 <= 0 && d23_2 <= 0) {
this.m_v2.a = 1;
this.m_count = 1;
this.m_v1.Copy(this.m_v2);
return;
}
// w3 region
if (d13_1 <= 0 && d23_1 <= 0) {
this.m_v3.a = 1;
this.m_count = 1;
this.m_v1.Copy(this.m_v3);
return;
}
// e23
if (d23_1 > 0 && d23_2 > 0 && d123_1 <= 0) {
var inv_d23 = 1 / (d23_1 + d23_2);
this.m_v2.a = d23_1 * inv_d23;
this.m_v3.a = d23_2 * inv_d23;
this.m_count = 2;
this.m_v1.Copy(this.m_v3);
return;
}
// Must be in triangle123
var inv_d123 = 1 / (d123_1 + d123_2 + d123_3);
this.m_v1.a = d123_1 * inv_d123;
this.m_v2.a = d123_2 * inv_d123;
this.m_v3.a = d123_3 * inv_d123;
this.m_count = 3;
};
b2Simplex.s_e12 = new b2Vec2();
b2Simplex.s_e13 = new b2Vec2();
b2Simplex.s_e23 = new b2Vec2();
return b2Simplex;
}());
var b2Distance_s_simplex = new b2Simplex();
var b2Distance_s_saveA = [0, 0, 0];
var b2Distance_s_saveB = [0, 0, 0];
var b2Distance_s_p = new b2Vec2();
var b2Distance_s_d = new b2Vec2();
var b2Distance_s_normal = new b2Vec2();
var b2Distance_s_supportA = new b2Vec2();
var b2Distance_s_supportB = new b2Vec2();
function b2Distance(output, cache, input) {
++exports.b2_gjkCalls;
var proxyA = input.proxyA;
var proxyB = input.proxyB;
var transformA = input.transformA;
var transformB = input.transformB;
// Initialize the simplex.
var simplex = b2Distance_s_simplex;
simplex.ReadCache(cache, proxyA, transformA, proxyB, transformB);
// Get simplex vertices as an array.
var vertices = simplex.m_vertices;
var k_maxIters = 20;
// These store the vertices of the last simplex so that we
// can check for duplicates and prevent cycling.
var saveA = b2Distance_s_saveA;
var saveB = b2Distance_s_saveB;
var saveCount = 0;
// Main iteration loop.
var iter = 0;
while (iter < k_maxIters) {
// Copy simplex so we can identify duplicates.
saveCount = simplex.m_count;
for (var i = 0; i < saveCount; ++i) {
saveA[i] = vertices[i].indexA;
saveB[i] = vertices[i].indexB;
}
switch (simplex.m_count) {
case 1:
break;
case 2:
simplex.Solve2();
break;
case 3:
simplex.Solve3();
break;
default:
// DEBUG: b2Assert(false);
break;
}
// If we have 3 points, then the origin is in the corresponding triangle.
if (simplex.m_count === 3) {
break;
}
// Get search direction.
var d = simplex.GetSearchDirection(b2Distance_s_d);
// Ensure the search direction is numerically fit.
if (d.LengthSquared() < b2_epsilon_sq) {
// The origin is probably contained by a line segment
// or triangle. Thus the shapes are overlapped.
// We can't return zero here even though there may be overlap.
// In case the simplex is a point, segment, or triangle it is difficult
// to determine if the origin is contained in the CSO or very close to it.
break;
}
// Compute a tentative new simplex vertex using support points.
var vertex = vertices[simplex.m_count];
vertex.indexA = proxyA.GetSupport(b2Rot.MulTRV(transformA.q, b2Vec2.NegV(d, b2Vec2.s_t0), b2Distance_s_supportA));
b2Transform.MulXV(transformA, proxyA.GetVertex(vertex.indexA), vertex.wA);
vertex.indexB = proxyB.GetSupport(b2Rot.MulTRV(transformB.q, d, b2Distance_s_supportB));
b2Transform.MulXV(transformB, proxyB.GetVertex(vertex.indexB), vertex.wB);
b2Vec2.SubVV(vertex.wB, vertex.wA, vertex.w);
// Iteration count is equated to the number of support point calls.
++iter;
++exports.b2_gjkIters;
// Check for duplicate support points. This is the main termination criteria.
var duplicate = false;
for (var i = 0; i < saveCount; ++i) {
if (vertex.indexA === saveA[i] && vertex.indexB === saveB[i]) {
duplicate = true;
break;
}
}
// If we found a duplicate support point we must exit to avoid cycling.
if (duplicate) {
break;
}
// New vertex is ok and needed.
++simplex.m_count;
}
exports.b2_gjkMaxIters = b2Max(exports.b2_gjkMaxIters, iter);
// Prepare output.
simplex.GetWitnessPoints(output.pointA, output.pointB);
output.distance = b2Vec2.DistanceVV(output.pointA, output.pointB);
output.iterations = iter;
// Cache the simplex.
simplex.WriteCache(cache);
// Apply radii if requested.
if (input.useRadii) {
var rA = proxyA.m_radius;
var rB = proxyB.m_radius;
if (output.distance > (rA + rB) && output.distance > b2_epsilon) {
// Shapes are still no overlapped.
// Move the witness points to the outer surface.
output.distance -= rA + rB;
var normal = b2Vec2.SubVV(output.pointB, output.pointA, b2Distance_s_normal);
normal.Normalize();
output.pointA.SelfMulAdd(rA, normal);
output.pointB.SelfMulSub(rB, normal);
}
else {
// Shapes are overlapped when radii are considered.
// Move the witness points to the middle.
var p = b2Vec2.MidVV(output.pointA, output.pointB, b2Distance_s_p);
output.pointA.Copy(p);
output.pointB.Copy(p);
output.distance = 0;
}
}
}
/// Perform a linear shape cast of shape B moving and shape A fixed. Determines the hit point, normal, and translation fraction.
// GJK-raycast
// Algorithm by Gino van den Bergen.
// "Smooth Mesh Contacts with GJK" in Game Physics Pearls. 2010
// bool b2ShapeCast(b2ShapeCastOutput* output, const b2ShapeCastInput* input);
var b2ShapeCast_s_n = new b2Vec2();
var b2ShapeCast_s_simplex = new b2Simplex();
var b2ShapeCast_s_wA = new b2Vec2();
var b2ShapeCast_s_wB = new b2Vec2();
var b2ShapeCast_s_v = new b2Vec2();
var b2ShapeCast_s_p = new b2Vec2();
var b2ShapeCast_s_pointA = new b2Vec2();
var b2ShapeCast_s_pointB = new b2Vec2();
function b2ShapeCast(output, input) {
output.iterations = 0;
output.lambda = 1.0;
output.normal.SetZero();
output.point.SetZero();
// const b2DistanceProxy* proxyA = &input.proxyA;
var proxyA = input.proxyA;
// const b2DistanceProxy* proxyB = &input.proxyB;
var proxyB = input.proxyB;
// float32 radiusA = b2Max(proxyA.m_radius, b2_polygonRadius);
var radiusA = b2Max(proxyA.m_radius, b2_polygonRadius);
// float32 radiusB = b2Max(proxyB.m_radius, b2_polygonRadius);
var radiusB = b2Max(proxyB.m_radius, b2_polygonRadius);
// float32 radius = radiusA + radiusB;
var radius = radiusA + radiusB;
// b2Transform xfA = input.transformA;
var xfA = input.transformA;
// b2Transform xfB = input.transformB;
var xfB = input.transformB;
// b2Vec2 r = input.translationB;
var r = input.translationB;
// b2Vec2 n(0.0f, 0.0f);
var n = b2ShapeCast_s_n.Set(0.0, 0.0);
// float32 lambda = 0.0f;
var lambda = 0.0;
// Initial simplex
var simplex = b2ShapeCast_s_simplex;
simplex.m_count = 0;
// Get simplex vertices as an array.
// b2SimplexVertex* vertices = &simplex.m_v1;
var vertices = simplex.m_vertices;
// Get support point in -r direction
// int32 indexA = proxyA.GetSupport(b2MulT(xfA.q, -r));
var indexA = proxyA.GetSupport(b2Rot.MulTRV(xfA.q, b2Vec2.NegV(r, b2Vec2.s_t1), b2Vec2.s_t0));
// b2Vec2 wA = b2Mul(xfA, proxyA.GetVertex(indexA));
var wA = b2Transform.MulXV(xfA, proxyA.GetVertex(indexA), b2ShapeCast_s_wA);
// int32 indexB = proxyB.GetSupport(b2MulT(xfB.q, r));
var indexB = proxyB.GetSupport(b2Rot.MulTRV(xfB.q, r, b2Vec2.s_t0));
// b2Vec2 wB = b2Mul(xfB, proxyB.GetVertex(indexB));
var wB = b2Transform.MulXV(xfB, proxyB.GetVertex(indexB), b2ShapeCast_s_wB);
// b2Vec2 v = wA - wB;
var v = b2Vec2.SubVV(wA, wB, b2ShapeCast_s_v);
// Sigma is the target distance between polygons
// float32 sigma = b2Max(b2_polygonRadius, radius - b2_polygonRadius);
var sigma = b2Max(b2_polygonRadius, radius - b2_polygonRadius);
// const float32 tolerance = 0.5f * b2_linearSlop;
var tolerance = 0.5 * b2_linearSlop;
// Main iteration loop.
// const int32 k_maxIters = 20;
var k_maxIters = 20;
// int32 iter = 0;
var iter = 0;
// while (iter < k_maxIters && b2Abs(v.Length() - sigma) > tolerance)
while (iter < k_maxIters && b2Abs(v.Length() - sigma) > tolerance) {
// DEBUG: b2Assert(simplex.m_count < 3);
output.iterations += 1;
// Support in direction -v (A - B)
// indexA = proxyA.GetSupport(b2MulT(xfA.q, -v));
indexA = proxyA.GetSupport(b2Rot.MulTRV(xfA.q, b2Vec2.NegV(v, b2Vec2.s_t1), b2Vec2.s_t0));
// wA = b2Mul(xfA, proxyA.GetVertex(indexA));
wA = b2Transform.MulXV(xfA, proxyA.GetVertex(indexA), b2ShapeCast_s_wA);
// indexB = proxyB.GetSupport(b2MulT(xfB.q, v));
indexB = proxyB.GetSupport(b2Rot.MulTRV(xfB.q, v, b2Vec2.s_t0));
// wB = b2Mul(xfB, proxyB.GetVertex(indexB));
wB = b2Transform.MulXV(xfB, proxyB.GetVertex(indexB), b2ShapeCast_s_wB);
// b2Vec2 p = wA - wB;
var p = b2Vec2.SubVV(wA, wB, b2ShapeCast_s_p);
// -v is a normal at p
v.Normalize();
// Intersect ray with plane
var vp = b2Vec2.DotVV(v, p);
var vr = b2Vec2.DotVV(v, r);
if (vp - sigma > lambda * vr) {
if (vr <= 0.0) {
return false;
}
lambda = (vp - sigma) / vr;
if (lambda > 1.0) {
return false;
}
// n = -v;
n.Copy(v).SelfNeg();
simplex.m_count = 0;
}
// Reverse simplex since it works with B - A.
// Shift by lambda * r because we want the closest point to the current clip point.
// Note that the support point p is not shifted because we want the plane equation
// to be formed in unshifted space.
// b2SimplexVertex* vertex = vertices + simplex.m_count;
var vertex = vertices[simplex.m_count];
vertex.indexA = indexB;
// vertex.wA = wB + lambda * r;
vertex.wA.Copy(wB).SelfMulAdd(lambda, r);
vertex.indexB = indexA;
// vertex.wB = wA;
vertex.wB.Copy(wA);
// vertex.w = vertex.wB - vertex.wA;
vertex.w.Copy(vertex.wB).SelfSub(vertex.wA);
vertex.a = 1.0;
simplex.m_count += 1;
switch (simplex.m_count) {
case 1:
break;
case 2:
simplex.Solve2();
break;
case 3:
simplex.Solve3();
break;
default:
// DEBUG: b2Assert(false);
}
// If we have 3 points, then the origin is in the corresponding triangle.
if (simplex.m_count === 3) {
// Overlap
return false;
}
// Get search direction.
// v = simplex.GetClosestPoint();
simplex.GetClosestPoint(v);
// Iteration count is equated to the number of support point calls.
++iter;
}
// Prepare output.
var pointA = b2ShapeCast_s_pointA;
var pointB = b2ShapeCast_s_pointB;
simplex.GetWitnessPoints(pointA, pointB);
if (v.LengthSquared() > 0.0) {
// n = -v;
n.Copy(v).SelfNeg();
n.Normalize();
}
// output.point = pointA + radiusA * n;
output.normal.Copy(n);
output.lambda = lambda;
output.iterations = iter;
return true;
}
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
(function (b2ContactFeatureType) {
b2ContactFeatureType[b2ContactFeatureType["e_vertex"] = 0] = "e_vertex";
b2ContactFeatureType[b2ContactFeatureType["e_face"] = 1] = "e_face";
})(exports.b2ContactFeatureType || (exports.b2ContactFeatureType = {}));
/// The features that intersect to form the contact point
/// This must be 4 bytes or less.
var b2ContactFeature = /** @class */ (function () {
function b2ContactFeature() {
this._key = 0;
this._key_invalid = false;
this._indexA = 0;
this._indexB = 0;
this._typeA = 0;
this._typeB = 0;
}
Object.defineProperty(b2ContactFeature.prototype, "key", {
get: function () {
if (this._key_invalid) {
this._key_invalid = false;
this._key = this._indexA | (this._indexB << 8) | (this._typeA << 16) | (this._typeB << 24);
}
return this._key;
},
set: function (value) {
this._key = value;
this._key_invalid = false;
this._indexA = this._key & 0xff;
this._indexB = (this._key >> 8) & 0xff;
this._typeA = (this._key >> 16) & 0xff;
this._typeB = (this._key >> 24) & 0xff;
},
enumerable: true,
configurable: true
});
Object.defineProperty(b2ContactFeature.prototype, "indexA", {
get: function () {
return this._indexA;
},
set: function (value) {
this._indexA = value;
this._key_invalid = true;
},
enumerable: true,
configurable: true
});
Object.defineProperty(b2ContactFeature.prototype, "indexB", {
get: function () {
return this._indexB;
},
set: function (value) {
this._indexB = value;
this._key_invalid = true;
},
enumerable: true,
configurable: true
});
Object.defineProperty(b2ContactFeature.prototype, "typeA", {
get: function () {
return this._typeA;
},
set: function (value) {
this._typeA = value;
this._key_invalid = true;
},
enumerable: true,
configurable: true
});
Object.defineProperty(b2ContactFeature.prototype, "typeB", {
get: function () {
return this._typeB;
},
set: function (value) {
this._typeB = value;
this._key_invalid = true;
},
enumerable: true,
configurable: true
});
return b2ContactFeature;
}());
/// Contact ids to facilitate warm starting.
var b2ContactID = /** @class */ (function () {
function b2ContactID() {
this.cf = new b2ContactFeature();
}
b2ContactID.prototype.Copy = function (o) {
this.key = o.key;
return this;
};
b2ContactID.prototype.Clone = function () {
return new b2ContactID().Copy(this);
};
Object.defineProperty(b2ContactID.prototype, "key", {
get: function () {
return this.cf.key;
},
set: function (value) {
this.cf.key = value;
},
enumerable: true,
configurable: true
});
return b2ContactID;
}());
/// A manifold point is a contact point belonging to a contact
/// manifold. It holds details related to the geometry and dynamics
/// of the contact points.
/// The local point usage depends on the manifold type:
/// -e_circles: the local center of circleB
/// -e_faceA: the local center of cirlceB or the clip point of polygonB
/// -e_faceB: the clip point of polygonA
/// This structure is stored across time steps, so we keep it small.
/// Note: the impulses are used for internal caching and may not
/// provide reliable contact forces, especially for high speed collisions.
var b2ManifoldPoint = /** @class */ (function () {
function b2ManifoldPoint() {
this.localPoint = new b2Vec2(); ///< usage depends on manifold type
this.normalImpulse = 0; ///< the non-penetration impulse
this.tangentImpulse = 0; ///< the friction impulse
this.id = new b2ContactID(); // TODO: readonly ///< uniquely identifies a contact point between two shapes
}
b2ManifoldPoint.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2ManifoldPoint(); });
};
b2ManifoldPoint.prototype.Reset = function () {
this.localPoint.SetZero();
this.normalImpulse = 0;
this.tangentImpulse = 0;
this.id.key = 0;
};
b2ManifoldPoint.prototype.Copy = function (o) {
this.localPoint.Copy(o.localPoint);
this.normalImpulse = o.normalImpulse;
this.tangentImpulse = o.tangentImpulse;
this.id.Copy(o.id);
return this;
};
return b2ManifoldPoint;
}());
(function (b2ManifoldType) {
b2ManifoldType[b2ManifoldType["e_unknown"] = -1] = "e_unknown";
b2ManifoldType[b2ManifoldType["e_circles"] = 0] = "e_circles";
b2ManifoldType[b2ManifoldType["e_faceA"] = 1] = "e_faceA";
b2ManifoldType[b2ManifoldType["e_faceB"] = 2] = "e_faceB";
})(exports.b2ManifoldType || (exports.b2ManifoldType = {}));
/// A manifold for two touching convex shapes.
/// Box2D supports multiple types of contact:
/// - clip point versus plane with radius
/// - point versus point with radius (circles)
/// The local point usage depends on the manifold type:
/// -e_circles: the local center of circleA
/// -e_faceA: the center of faceA
/// -e_faceB: the center of faceB
/// Similarly the local normal usage:
/// -e_circles: not used
/// -e_faceA: the normal on polygonA
/// -e_faceB: the normal on polygonB
/// We store contacts in this way so that position correction can
/// account for movement, which is critical for continuous physics.
/// All contact scenarios must be expressed in one of these types.
/// This structure is stored across time steps, so we keep it small.
var b2Manifold = /** @class */ (function () {
function b2Manifold() {
this.points = b2ManifoldPoint.MakeArray(b2_maxManifoldPoints);
this.localNormal = new b2Vec2();
this.localPoint = new b2Vec2();
this.type = exports.b2ManifoldType.e_unknown;
this.pointCount = 0;
}
b2Manifold.prototype.Reset = function () {
for (var i = 0; i < b2_maxManifoldPoints; ++i) {
// DEBUG: b2Assert(this.points[i] instanceof b2ManifoldPoint);
this.points[i].Reset();
}
this.localNormal.SetZero();
this.localPoint.SetZero();
this.type = exports.b2ManifoldType.e_unknown;
this.pointCount = 0;
};
b2Manifold.prototype.Copy = function (o) {
this.pointCount = o.pointCount;
for (var i = 0; i < b2_maxManifoldPoints; ++i) {
// DEBUG: b2Assert(this.points[i] instanceof b2ManifoldPoint);
this.points[i].Copy(o.points[i]);
}
this.localNormal.Copy(o.localNormal);
this.localPoint.Copy(o.localPoint);
this.type = o.type;
return this;
};
b2Manifold.prototype.Clone = function () {
return new b2Manifold().Copy(this);
};
return b2Manifold;
}());
var b2WorldManifold = /** @class */ (function () {
function b2WorldManifold() {
this.normal = new b2Vec2();
this.points = b2Vec2.MakeArray(b2_maxManifoldPoints);
this.separations = b2MakeNumberArray(b2_maxManifoldPoints);
}
b2WorldManifold.prototype.Initialize = function (manifold, xfA, radiusA, xfB, radiusB) {
if (manifold.pointCount === 0) {
return;
}
switch (manifold.type) {
case exports.b2ManifoldType.e_circles: {
this.normal.Set(1, 0);
var pointA = b2Transform.MulXV(xfA, manifold.localPoint, b2WorldManifold.Initialize_s_pointA);
var pointB = b2Transform.MulXV(xfB, manifold.points[0].localPoint, b2WorldManifold.Initialize_s_pointB);
if (b2Vec2.DistanceSquaredVV(pointA, pointB) > b2_epsilon_sq) {
b2Vec2.SubVV(pointB, pointA, this.normal).SelfNormalize();
}
var cA = b2Vec2.AddVMulSV(pointA, radiusA, this.normal, b2WorldManifold.Initialize_s_cA);
var cB = b2Vec2.SubVMulSV(pointB, radiusB, this.normal, b2WorldManifold.Initialize_s_cB);
b2Vec2.MidVV(cA, cB, this.points[0]);
this.separations[0] = b2Vec2.DotVV(b2Vec2.SubVV(cB, cA, b2Vec2.s_t0), this.normal); // b2Dot(cB - cA, normal);
break;
}
case exports.b2ManifoldType.e_faceA: {
b2Rot.MulRV(xfA.q, manifold.localNormal, this.normal);
var planePoint = b2Transform.MulXV(xfA, manifold.localPoint, b2WorldManifold.Initialize_s_planePoint);
for (var i = 0; i < manifold.pointCount; ++i) {
var clipPoint = b2Transform.MulXV(xfB, manifold.points[i].localPoint, b2WorldManifold.Initialize_s_clipPoint);
var s = radiusA - b2Vec2.DotVV(b2Vec2.SubVV(clipPoint, planePoint, b2Vec2.s_t0), this.normal);
var cA = b2Vec2.AddVMulSV(clipPoint, s, this.normal, b2WorldManifold.Initialize_s_cA);
var cB = b2Vec2.SubVMulSV(clipPoint, radiusB, this.normal, b2WorldManifold.Initialize_s_cB);
b2Vec2.MidVV(cA, cB, this.points[i]);
this.separations[i] = b2Vec2.DotVV(b2Vec2.SubVV(cB, cA, b2Vec2.s_t0), this.normal); // b2Dot(cB - cA, normal);
}
break;
}
case exports.b2ManifoldType.e_faceB: {
b2Rot.MulRV(xfB.q, manifold.localNormal, this.normal);
var planePoint = b2Transform.MulXV(xfB, manifold.localPoint, b2WorldManifold.Initialize_s_planePoint);
for (var i = 0; i < manifold.pointCount; ++i) {
var clipPoint = b2Transform.MulXV(xfA, manifold.points[i].localPoint, b2WorldManifold.Initialize_s_clipPoint);
var s = radiusB - b2Vec2.DotVV(b2Vec2.SubVV(clipPoint, planePoint, b2Vec2.s_t0), this.normal);
var cB = b2Vec2.AddVMulSV(clipPoint, s, this.normal, b2WorldManifold.Initialize_s_cB);
var cA = b2Vec2.SubVMulSV(clipPoint, radiusA, this.normal, b2WorldManifold.Initialize_s_cA);
b2Vec2.MidVV(cA, cB, this.points[i]);
this.separations[i] = b2Vec2.DotVV(b2Vec2.SubVV(cA, cB, b2Vec2.s_t0), this.normal); // b2Dot(cA - cB, normal);
}
// Ensure normal points from A to B.
this.normal.SelfNeg();
break;
}
}
};
b2WorldManifold.Initialize_s_pointA = new b2Vec2();
b2WorldManifold.Initialize_s_pointB = new b2Vec2();
b2WorldManifold.Initialize_s_cA = new b2Vec2();
b2WorldManifold.Initialize_s_cB = new b2Vec2();
b2WorldManifold.Initialize_s_planePoint = new b2Vec2();
b2WorldManifold.Initialize_s_clipPoint = new b2Vec2();
return b2WorldManifold;
}());
(function (b2PointState) {
b2PointState[b2PointState["b2_nullState"] = 0] = "b2_nullState";
b2PointState[b2PointState["b2_addState"] = 1] = "b2_addState";
b2PointState[b2PointState["b2_persistState"] = 2] = "b2_persistState";
b2PointState[b2PointState["b2_removeState"] = 3] = "b2_removeState";
})(exports.b2PointState || (exports.b2PointState = {}));
/// Compute the point states given two manifolds. The states pertain to the transition from manifold1
/// to manifold2. So state1 is either persist or remove while state2 is either add or persist.
function b2GetPointStates(state1, state2, manifold1, manifold2) {
// Detect persists and removes.
var i;
for (i = 0; i < manifold1.pointCount; ++i) {
var id = manifold1.points[i].id;
var key = id.key;
state1[i] = exports.b2PointState.b2_removeState;
for (var j = 0, jct = manifold2.pointCount; j < jct; ++j) {
if (manifold2.points[j].id.key === key) {
state1[i] = exports.b2PointState.b2_persistState;
break;
}
}
}
for (; i < b2_maxManifoldPoints; ++i) {
state1[i] = exports.b2PointState.b2_nullState;
}
// Detect persists and adds.
for (i = 0; i < manifold2.pointCount; ++i) {
var id = manifold2.points[i].id;
var key = id.key;
state2[i] = exports.b2PointState.b2_addState;
for (var j = 0, jct = manifold1.pointCount; j < jct; ++j) {
if (manifold1.points[j].id.key === key) {
state2[i] = exports.b2PointState.b2_persistState;
break;
}
}
}
for (; i < b2_maxManifoldPoints; ++i) {
state2[i] = exports.b2PointState.b2_nullState;
}
}
/// Used for computing contact manifolds.
var b2ClipVertex = /** @class */ (function () {
function b2ClipVertex() {
this.v = new b2Vec2();
this.id = new b2ContactID();
}
b2ClipVertex.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2ClipVertex(); });
};
b2ClipVertex.prototype.Copy = function (other) {
this.v.Copy(other.v);
this.id.Copy(other.id);
return this;
};
return b2ClipVertex;
}());
/// Ray-cast input data. The ray extends from p1 to p1 + maxFraction * (p2 - p1).
var b2RayCastInput = /** @class */ (function () {
function b2RayCastInput() {
this.p1 = new b2Vec2();
this.p2 = new b2Vec2();
this.maxFraction = 1;
}
b2RayCastInput.prototype.Copy = function (o) {
this.p1.Copy(o.p1);
this.p2.Copy(o.p2);
this.maxFraction = o.maxFraction;
return this;
};
return b2RayCastInput;
}());
/// Ray-cast output data. The ray hits at p1 + fraction * (p2 - p1), where p1 and p2
/// come from b2RayCastInput.
var b2RayCastOutput = /** @class */ (function () {
function b2RayCastOutput() {
this.normal = new b2Vec2();
this.fraction = 0;
}
b2RayCastOutput.prototype.Copy = function (o) {
this.normal.Copy(o.normal);
this.fraction = o.fraction;
return this;
};
return b2RayCastOutput;
}());
/// An axis aligned bounding box.
var b2AABB = /** @class */ (function () {
function b2AABB() {
this.lowerBound = new b2Vec2(); ///< the lower vertex
this.upperBound = new b2Vec2(); ///< the upper vertex
this.m_cache_center = new b2Vec2(); // access using GetCenter()
this.m_cache_extent = new b2Vec2(); // access using GetExtents()
}
b2AABB.prototype.Copy = function (o) {
this.lowerBound.Copy(o.lowerBound);
this.upperBound.Copy(o.upperBound);
return this;
};
/// Verify that the bounds are sorted.
b2AABB.prototype.IsValid = function () {
var d_x = this.upperBound.x - this.lowerBound.x;
var d_y = this.upperBound.y - this.lowerBound.y;
var valid = d_x >= 0 && d_y >= 0;
valid = valid && this.lowerBound.IsValid() && this.upperBound.IsValid();
return valid;
};
/// Get the center of the AABB.
b2AABB.prototype.GetCenter = function () {
return b2Vec2.MidVV(this.lowerBound, this.upperBound, this.m_cache_center);
};
/// Get the extents of the AABB (half-widths).
b2AABB.prototype.GetExtents = function () {
return b2Vec2.ExtVV(this.lowerBound, this.upperBound, this.m_cache_extent);
};
/// Get the perimeter length
b2AABB.prototype.GetPerimeter = function () {
var wx = this.upperBound.x - this.lowerBound.x;
var wy = this.upperBound.y - this.lowerBound.y;
return 2 * (wx + wy);
};
/// Combine an AABB into this one.
b2AABB.prototype.Combine1 = function (aabb) {
this.lowerBound.x = b2Min(this.lowerBound.x, aabb.lowerBound.x);
this.lowerBound.y = b2Min(this.lowerBound.y, aabb.lowerBound.y);
this.upperBound.x = b2Max(this.upperBound.x, aabb.upperBound.x);
this.upperBound.y = b2Max(this.upperBound.y, aabb.upperBound.y);
return this;
};
/// Combine two AABBs into this one.
b2AABB.prototype.Combine2 = function (aabb1, aabb2) {
this.lowerBound.x = b2Min(aabb1.lowerBound.x, aabb2.lowerBound.x);
this.lowerBound.y = b2Min(aabb1.lowerBound.y, aabb2.lowerBound.y);
this.upperBound.x = b2Max(aabb1.upperBound.x, aabb2.upperBound.x);
this.upperBound.y = b2Max(aabb1.upperBound.y, aabb2.upperBound.y);
return this;
};
b2AABB.Combine = function (aabb1, aabb2, out) {
out.Combine2(aabb1, aabb2);
return out;
};
/// Does this aabb contain the provided AABB.
b2AABB.prototype.Contains = function (aabb) {
var result = true;
result = result && this.lowerBound.x <= aabb.lowerBound.x;
result = result && this.lowerBound.y <= aabb.lowerBound.y;
result = result && aabb.upperBound.x <= this.upperBound.x;
result = result && aabb.upperBound.y <= this.upperBound.y;
return result;
};
// From Real-time Collision Detection, p179.
b2AABB.prototype.RayCast = function (output, input) {
var tmin = (-b2_maxFloat);
var tmax = b2_maxFloat;
var p_x = input.p1.x;
var p_y = input.p1.y;
var d_x = input.p2.x - input.p1.x;
var d_y = input.p2.y - input.p1.y;
var absD_x = b2Abs(d_x);
var absD_y = b2Abs(d_y);
var normal = output.normal;
if (absD_x < b2_epsilon) {
// Parallel.
if (p_x < this.lowerBound.x || this.upperBound.x < p_x) {
return false;
}
}
else {
var inv_d = 1 / d_x;
var t1 = (this.lowerBound.x - p_x) * inv_d;
var t2 = (this.upperBound.x - p_x) * inv_d;
// Sign of the normal vector.
var s = (-1);
if (t1 > t2) {
var t3 = t1;
t1 = t2;
t2 = t3;
s = 1;
}
// Push the min up
if (t1 > tmin) {
normal.x = s;
normal.y = 0;
tmin = t1;
}
// Pull the max down
tmax = b2Min(tmax, t2);
if (tmin > tmax) {
return false;
}
}
if (absD_y < b2_epsilon) {
// Parallel.
if (p_y < this.lowerBound.y || this.upperBound.y < p_y) {
return false;
}
}
else {
var inv_d = 1 / d_y;
var t1 = (this.lowerBound.y - p_y) * inv_d;
var t2 = (this.upperBound.y - p_y) * inv_d;
// Sign of the normal vector.
var s = (-1);
if (t1 > t2) {
var t3 = t1;
t1 = t2;
t2 = t3;
s = 1;
}
// Push the min up
if (t1 > tmin) {
normal.x = 0;
normal.y = s;
tmin = t1;
}
// Pull the max down
tmax = b2Min(tmax, t2);
if (tmin > tmax) {
return false;
}
}
// Does the ray start inside the box?
// Does the ray intersect beyond the max fraction?
if (tmin < 0 || input.maxFraction < tmin) {
return false;
}
// Intersection.
output.fraction = tmin;
return true;
};
b2AABB.prototype.TestContain = function (point) {
if (point.x < this.lowerBound.x || this.upperBound.x < point.x) {
return false;
}
if (point.y < this.lowerBound.y || this.upperBound.y < point.y) {
return false;
}
return true;
};
b2AABB.prototype.TestOverlap = function (other) {
var d1_x = other.lowerBound.x - this.upperBound.x;
var d1_y = other.lowerBound.y - this.upperBound.y;
var d2_x = this.lowerBound.x - other.upperBound.x;
var d2_y = this.lowerBound.y - other.upperBound.y;
if (d1_x > 0 || d1_y > 0) {
return false;
}
if (d2_x > 0 || d2_y > 0) {
return false;
}
return true;
};
return b2AABB;
}());
function b2TestOverlapAABB(a, b) {
var d1_x = b.lowerBound.x - a.upperBound.x;
var d1_y = b.lowerBound.y - a.upperBound.y;
var d2_x = a.lowerBound.x - b.upperBound.x;
var d2_y = a.lowerBound.y - b.upperBound.y;
if (d1_x > 0 || d1_y > 0) {
return false;
}
if (d2_x > 0 || d2_y > 0) {
return false;
}
return true;
}
/// Clipping for contact manifolds.
function b2ClipSegmentToLine(vOut, vIn, normal, offset, vertexIndexA) {
// Start with no output points
var numOut = 0;
var vIn0 = vIn[0];
var vIn1 = vIn[1];
// Calculate the distance of end points to the line
var distance0 = b2Vec2.DotVV(normal, vIn0.v) - offset;
var distance1 = b2Vec2.DotVV(normal, vIn1.v) - offset;
// If the points are behind the plane
if (distance0 <= 0) {
vOut[numOut++].Copy(vIn0);
}
if (distance1 <= 0) {
vOut[numOut++].Copy(vIn1);
}
// If the points are on different sides of the plane
if (distance0 * distance1 < 0) {
// Find intersection point of edge and plane
var interp = distance0 / (distance0 - distance1);
var v = vOut[numOut].v;
v.x = vIn0.v.x + interp * (vIn1.v.x - vIn0.v.x);
v.y = vIn0.v.y + interp * (vIn1.v.y - vIn0.v.y);
// VertexA is hitting edgeB.
var id = vOut[numOut].id;
id.cf.indexA = vertexIndexA;
id.cf.indexB = vIn0.id.cf.indexB;
id.cf.typeA = exports.b2ContactFeatureType.e_vertex;
id.cf.typeB = exports.b2ContactFeatureType.e_face;
++numOut;
}
return numOut;
}
/// Determine if two generic shapes overlap.
var b2TestOverlapShape_s_input = new b2DistanceInput();
var b2TestOverlapShape_s_simplexCache = new b2SimplexCache();
var b2TestOverlapShape_s_output = new b2DistanceOutput();
function b2TestOverlapShape(shapeA, indexA, shapeB, indexB, xfA, xfB) {
var input = b2TestOverlapShape_s_input.Reset();
input.proxyA.SetShape(shapeA, indexA);
input.proxyB.SetShape(shapeB, indexB);
input.transformA.Copy(xfA);
input.transformB.Copy(xfB);
input.useRadii = true;
var simplexCache = b2TestOverlapShape_s_simplexCache.Reset();
simplexCache.count = 0;
var output = b2TestOverlapShape_s_output.Reset();
b2Distance(output, simplexCache, input);
return output.distance < 10 * b2_epsilon;
}
/*
* Copyright (c) 2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
function verify(value) {
if (value === null) {
throw new Error();
}
return value;
}
/// A node in the dynamic tree. The client does not interact with this directly.
var b2TreeNode = /** @class */ (function () {
function b2TreeNode(id) {
if (id === void 0) { id = 0; }
this.m_id = 0;
this.aabb = new b2AABB();
this.parent = null; // or next
this.child1 = null;
this.child2 = null;
this.height = 0; // leaf = 0, free node = -1
this.m_id = id;
}
b2TreeNode.prototype.IsLeaf = function () {
return this.child1 === null;
};
return b2TreeNode;
}());
var b2DynamicTree = /** @class */ (function () {
function b2DynamicTree() {
this.m_root = null;
// b2TreeNode* public m_nodes;
// int32 public m_nodeCount;
// int32 public m_nodeCapacity;
this.m_freeList = null;
this.m_path = 0;
this.m_insertionCount = 0;
this.m_stack = new b2GrowableStack(256);
}
// public GetUserData(proxy: b2TreeNode<T>): any {
// // DEBUG: b2Assert(proxy !== null);
// return proxy.userData;
// }
// public GetFatAABB(proxy: b2TreeNode<T>): b2AABB {
// // DEBUG: b2Assert(proxy !== null);
// return proxy.aabb;
// }
b2DynamicTree.prototype.Query = function (aabb, callback) {
if (this.m_root === null) {
return;
}
var stack = this.m_stack.Reset();
stack.Push(this.m_root);
while (stack.GetCount() > 0) {
var node = stack.Pop();
// if (node === null) {
// continue;
// }
if (node.aabb.TestOverlap(aabb)) {
if (node.IsLeaf()) {
var proceed = callback(node);
if (!proceed) {
return;
}
}
else {
stack.Push(verify(node.child1));
stack.Push(verify(node.child2));
}
}
}
};
b2DynamicTree.prototype.QueryPoint = function (point, callback) {
if (this.m_root === null) {
return;
}
var stack = this.m_stack.Reset();
stack.Push(this.m_root);
while (stack.GetCount() > 0) {
var node = stack.Pop();
// if (node === null) {
// continue;
// }
if (node.aabb.TestContain(point)) {
if (node.IsLeaf()) {
var proceed = callback(node);
if (!proceed) {
return;
}
}
else {
stack.Push(verify(node.child1));
stack.Push(verify(node.child2));
}
}
}
};
b2DynamicTree.prototype.RayCast = function (input, callback) {
if (this.m_root === null) {
return;
}
var p1 = input.p1;
var p2 = input.p2;
var r = b2Vec2.SubVV(p2, p1, b2DynamicTree.s_r);
// DEBUG: b2Assert(r.LengthSquared() > 0);
r.Normalize();
// v is perpendicular to the segment.
var v = b2Vec2.CrossOneV(r, b2DynamicTree.s_v);
var abs_v = b2Vec2.AbsV(v, b2DynamicTree.s_abs_v);
// Separating axis for segment (Gino, p80).
// |dot(v, p1 - c)| > dot(|v|, h)
var maxFraction = input.maxFraction;
// Build a bounding box for the segment.
var segmentAABB = b2DynamicTree.s_segmentAABB;
var t_x = p1.x + maxFraction * (p2.x - p1.x);
var t_y = p1.y + maxFraction * (p2.y - p1.y);
segmentAABB.lowerBound.x = b2Min(p1.x, t_x);
segmentAABB.lowerBound.y = b2Min(p1.y, t_y);
segmentAABB.upperBound.x = b2Max(p1.x, t_x);
segmentAABB.upperBound.y = b2Max(p1.y, t_y);
var stack = this.m_stack.Reset();
stack.Push(this.m_root);
while (stack.GetCount() > 0) {
var node = stack.Pop();
// if (node === null) {
// continue;
// }
if (!b2TestOverlapAABB(node.aabb, segmentAABB)) {
continue;
}
// Separating axis for segment (Gino, p80).
// |dot(v, p1 - c)| > dot(|v|, h)
var c = node.aabb.GetCenter();
var h = node.aabb.GetExtents();
var separation = b2Abs(b2Vec2.DotVV(v, b2Vec2.SubVV(p1, c, b2Vec2.s_t0))) - b2Vec2.DotVV(abs_v, h);
if (separation > 0) {
continue;
}
if (node.IsLeaf()) {
var subInput = b2DynamicTree.s_subInput;
subInput.p1.Copy(input.p1);
subInput.p2.Copy(input.p2);
subInput.maxFraction = maxFraction;
var value = callback(subInput, node);
if (value === 0) {
// The client has terminated the ray cast.
return;
}
if (value > 0) {
// Update segment bounding box.
maxFraction = value;
t_x = p1.x + maxFraction * (p2.x - p1.x);
t_y = p1.y + maxFraction * (p2.y - p1.y);
segmentAABB.lowerBound.x = b2Min(p1.x, t_x);
segmentAABB.lowerBound.y = b2Min(p1.y, t_y);
segmentAABB.upperBound.x = b2Max(p1.x, t_x);
segmentAABB.upperBound.y = b2Max(p1.y, t_y);
}
}
else {
stack.Push(verify(node.child1));
stack.Push(verify(node.child2));
}
}
};
b2DynamicTree.prototype.AllocateNode = function () {
// Expand the node pool as needed.
if (this.m_freeList) {
var node = this.m_freeList;
this.m_freeList = node.parent; // this.m_freeList = node.next;
node.parent = null;
node.child1 = null;
node.child2 = null;
node.height = 0;
delete node.userData; // = null;
return node;
}
return new b2TreeNode(b2DynamicTree.s_node_id++);
};
b2DynamicTree.prototype.FreeNode = function (node) {
node.parent = this.m_freeList; // node.next = this.m_freeList;
node.child1 = null;
node.child2 = null;
node.height = -1;
delete node.userData; // = null;
this.m_freeList = node;
};
b2DynamicTree.prototype.CreateProxy = function (aabb, userData) {
var node = this.AllocateNode();
// Fatten the aabb.
var r_x = b2_aabbExtension;
var r_y = b2_aabbExtension;
node.aabb.lowerBound.x = aabb.lowerBound.x - r_x;
node.aabb.lowerBound.y = aabb.lowerBound.y - r_y;
node.aabb.upperBound.x = aabb.upperBound.x + r_x;
node.aabb.upperBound.y = aabb.upperBound.y + r_y;
node.userData = userData;
node.height = 0;
this.InsertLeaf(node);
return node;
};
b2DynamicTree.prototype.DestroyProxy = function (proxy) {
// DEBUG: b2Assert(proxy.IsLeaf());
this.RemoveLeaf(proxy);
this.FreeNode(proxy);
};
b2DynamicTree.prototype.MoveProxy = function (proxy, aabb, displacement) {
// DEBUG: b2Assert(proxy.IsLeaf());
if (proxy.aabb.Contains(aabb)) {
return false;
}
this.RemoveLeaf(proxy);
// Extend AABB.
// Predict AABB displacement.
var r_x = b2_aabbExtension + b2_aabbMultiplier * (displacement.x > 0 ? displacement.x : (-displacement.x));
var r_y = b2_aabbExtension + b2_aabbMultiplier * (displacement.y > 0 ? displacement.y : (-displacement.y));
proxy.aabb.lowerBound.x = aabb.lowerBound.x - r_x;
proxy.aabb.lowerBound.y = aabb.lowerBound.y - r_y;
proxy.aabb.upperBound.x = aabb.upperBound.x + r_x;
proxy.aabb.upperBound.y = aabb.upperBound.y + r_y;
this.InsertLeaf(proxy);
return true;
};
b2DynamicTree.prototype.InsertLeaf = function (leaf) {
++this.m_insertionCount;
if (this.m_root === null) {
this.m_root = leaf;
this.m_root.parent = null;
return;
}
// Find the best sibling for this node
var leafAABB = leaf.aabb;
///const center: b2Vec2 = leafAABB.GetCenter();
var index = this.m_root;
while (!index.IsLeaf()) {
var child1 = verify(index.child1);
var child2 = verify(index.child2);
var area = index.aabb.GetPerimeter();
var combinedAABB = b2DynamicTree.s_combinedAABB;
combinedAABB.Combine2(index.aabb, leafAABB);
var combinedArea = combinedAABB.GetPerimeter();
// Cost of creating a new parent for this node and the new leaf
var cost = 2 * combinedArea;
// Minimum cost of pushing the leaf further down the tree
var inheritanceCost = 2 * (combinedArea - area);
// Cost of descending into child1
var cost1 = void 0;
var aabb = b2DynamicTree.s_aabb;
var oldArea = void 0;
var newArea = void 0;
if (child1.IsLeaf()) {
aabb.Combine2(leafAABB, child1.aabb);
cost1 = aabb.GetPerimeter() + inheritanceCost;
}
else {
aabb.Combine2(leafAABB, child1.aabb);
oldArea = child1.aabb.GetPerimeter();
newArea = aabb.GetPerimeter();
cost1 = (newArea - oldArea) + inheritanceCost;
}
// Cost of descending into child2
var cost2 = void 0;
if (child2.IsLeaf()) {
aabb.Combine2(leafAABB, child2.aabb);
cost2 = aabb.GetPerimeter() + inheritanceCost;
}
else {
aabb.Combine2(leafAABB, child2.aabb);
oldArea = child2.aabb.GetPerimeter();
newArea = aabb.GetPerimeter();
cost2 = newArea - oldArea + inheritanceCost;
}
// Descend according to the minimum cost.
if (cost < cost1 && cost < cost2) {
break;
}
// Descend
if (cost1 < cost2) {
index = child1;
}
else {
index = child2;
}
}
var sibling = index;
// Create a parent for the siblings.
var oldParent = sibling.parent;
var newParent = this.AllocateNode();
newParent.parent = oldParent;
delete newParent.userData; // = null;
newParent.aabb.Combine2(leafAABB, sibling.aabb);
newParent.height = sibling.height + 1;
if (oldParent) {
// The sibling was not the root.
if (oldParent.child1 === sibling) {
oldParent.child1 = newParent;
}
else {
oldParent.child2 = newParent;
}
newParent.child1 = sibling;
newParent.child2 = leaf;
sibling.parent = newParent;
leaf.parent = newParent;
}
else {
// The sibling was the root.
newParent.child1 = sibling;
newParent.child2 = leaf;
sibling.parent = newParent;
leaf.parent = newParent;
this.m_root = newParent;
}
// Walk back up the tree fixing heights and AABBs
var index2 = leaf.parent;
while (index2 !== null) {
index2 = this.Balance(index2);
var child1 = verify(index2.child1);
var child2 = verify(index2.child2);
index2.height = 1 + b2Max(child1.height, child2.height);
index2.aabb.Combine2(child1.aabb, child2.aabb);
index2 = index2.parent;
}
// this.Validate();
};
b2DynamicTree.prototype.RemoveLeaf = function (leaf) {
if (leaf === this.m_root) {
this.m_root = null;
return;
}
var parent = verify(leaf.parent);
var grandParent = parent && parent.parent;
var sibling;
if (parent.child1 === leaf) {
sibling = verify(parent.child2);
}
else {
sibling = verify(parent.child1);
}
if (grandParent) {
// Destroy parent and connect sibling to grandParent.
if (grandParent.child1 === parent) {
grandParent.child1 = sibling;
}
else {
grandParent.child2 = sibling;
}
sibling.parent = grandParent;
this.FreeNode(parent);
// Adjust ancestor bounds.
var index = grandParent;
while (index) {
index = this.Balance(index);
var child1 = verify(index.child1);
var child2 = verify(index.child2);
index.aabb.Combine2(child1.aabb, child2.aabb);
index.height = 1 + b2Max(child1.height, child2.height);
index = index.parent;
}
}
else {
this.m_root = sibling;
sibling.parent = null;
this.FreeNode(parent);
}
// this.Validate();
};
b2DynamicTree.prototype.Balance = function (A) {
// DEBUG: b2Assert(A !== null);
if (A.IsLeaf() || A.height < 2) {
return A;
}
var B = verify(A.child1);
var C = verify(A.child2);
var balance = C.height - B.height;
// Rotate C up
if (balance > 1) {
var F = verify(C.child1);
var G = verify(C.child2);
// Swap A and C
C.child1 = A;
C.parent = A.parent;
A.parent = C;
// A's old parent should point to C
if (C.parent !== null) {
if (C.parent.child1 === A) {
C.parent.child1 = C;
}
else {
// DEBUG: b2Assert(C.parent.child2 === A);
C.parent.child2 = C;
}
}
else {
this.m_root = C;
}
// Rotate
if (F.height > G.height) {
C.child2 = F;
A.child2 = G;
G.parent = A;
A.aabb.Combine2(B.aabb, G.aabb);
C.aabb.Combine2(A.aabb, F.aabb);
A.height = 1 + b2Max(B.height, G.height);
C.height = 1 + b2Max(A.height, F.height);
}
else {
C.child2 = G;
A.child2 = F;
F.parent = A;
A.aabb.Combine2(B.aabb, F.aabb);
C.aabb.Combine2(A.aabb, G.aabb);
A.height = 1 + b2Max(B.height, F.height);
C.height = 1 + b2Max(A.height, G.height);
}
return C;
}
// Rotate B up
if (balance < -1) {
var D = verify(B.child1);
var E = verify(B.child2);
// Swap A and B
B.child1 = A;
B.parent = A.parent;
A.parent = B;
// A's old parent should point to B
if (B.parent !== null) {
if (B.parent.child1 === A) {
B.parent.child1 = B;
}
else {
// DEBUG: b2Assert(B.parent.child2 === A);
B.parent.child2 = B;
}
}
else {
this.m_root = B;
}
// Rotate
if (D.height > E.height) {
B.child2 = D;
A.child1 = E;
E.parent = A;
A.aabb.Combine2(C.aabb, E.aabb);
B.aabb.Combine2(A.aabb, D.aabb);
A.height = 1 + b2Max(C.height, E.height);
B.height = 1 + b2Max(A.height, D.height);
}
else {
B.child2 = E;
A.child1 = D;
D.parent = A;
A.aabb.Combine2(C.aabb, D.aabb);
B.aabb.Combine2(A.aabb, E.aabb);
A.height = 1 + b2Max(C.height, D.height);
B.height = 1 + b2Max(A.height, E.height);
}
return B;
}
return A;
};
b2DynamicTree.prototype.GetHeight = function () {
if (this.m_root === null) {
return 0;
}
return this.m_root.height;
};
b2DynamicTree.GetAreaNode = function (node) {
if (node === null) {
return 0;
}
if (node.IsLeaf()) {
return 0;
}
var area = node.aabb.GetPerimeter();
area += b2DynamicTree.GetAreaNode(node.child1);
area += b2DynamicTree.GetAreaNode(node.child2);
return area;
};
b2DynamicTree.prototype.GetAreaRatio = function () {
if (this.m_root === null) {
return 0;
}
var root = this.m_root;
var rootArea = root.aabb.GetPerimeter();
var totalArea = b2DynamicTree.GetAreaNode(this.m_root);
/*
float32 totalArea = 0.0;
for (int32 i = 0; i < m_nodeCapacity; ++i) {
const b2TreeNode<T>* node = m_nodes + i;
if (node.height < 0) {
// Free node in pool
continue;
}
totalArea += node.aabb.GetPerimeter();
}
*/
return totalArea / rootArea;
};
b2DynamicTree.prototype.ComputeHeightNode = function (node) {
if (!node || node.IsLeaf()) {
return 0;
}
var height1 = this.ComputeHeightNode(node.child1);
var height2 = this.ComputeHeightNode(node.child2);
return 1 + b2Max(height1, height2);
};
b2DynamicTree.prototype.ComputeHeight = function () {
var height = this.ComputeHeightNode(this.m_root);
return height;
};
b2DynamicTree.prototype.ValidateStructure = function (index) {
if (index === null) {
return;
}
if (index === this.m_root) ;
var node = index;
if (node.IsLeaf()) {
// DEBUG: b2Assert(node.child1 === null);
// DEBUG: b2Assert(node.child2 === null);
// DEBUG: b2Assert(node.height === 0);
return;
}
var child1 = verify(node.child1);
var child2 = verify(node.child2);
// DEBUG: b2Assert(child1.parent === index);
// DEBUG: b2Assert(child2.parent === index);
this.ValidateStructure(child1);
this.ValidateStructure(child2);
};
b2DynamicTree.prototype.ValidateMetrics = function (index) {
if (index === null) {
return;
}
var node = index;
if (node.IsLeaf()) {
// DEBUG: b2Assert(node.child1 === null);
// DEBUG: b2Assert(node.child2 === null);
// DEBUG: b2Assert(node.height === 0);
return;
}
var child1 = verify(node.child1);
var child2 = verify(node.child2);
// DEBUG: const height1: number = child1.height;
// DEBUG: const height2: number = child2.height;
// DEBUG: const height: number = 1 + b2Max(height1, height2);
// DEBUG: b2Assert(node.height === height);
var aabb = b2DynamicTree.s_aabb;
aabb.Combine2(child1.aabb, child2.aabb);
// DEBUG: b2Assert(aabb.lowerBound === node.aabb.lowerBound);
// DEBUG: b2Assert(aabb.upperBound === node.aabb.upperBound);
this.ValidateMetrics(child1);
this.ValidateMetrics(child2);
};
b2DynamicTree.prototype.Validate = function () {
// DEBUG: this.ValidateStructure(this.m_root);
// DEBUG: this.ValidateMetrics(this.m_root);
// let freeCount: number = 0;
// let freeIndex: b2TreeNode<T> | null = this.m_freeList;
// while (freeIndex !== null) {
// freeIndex = freeIndex.parent; // freeIndex = freeIndex.next;
// ++freeCount;
// }
// DEBUG: b2Assert(this.GetHeight() === this.ComputeHeight());
// b2Assert(this.m_nodeCount + freeCount === this.m_nodeCapacity);
};
b2DynamicTree.GetMaxBalanceNode = function (node, maxBalance) {
if (node === null) {
return maxBalance;
}
if (node.height <= 1) {
return maxBalance;
}
// DEBUG: b2Assert(!node.IsLeaf());
var child1 = verify(node.child1);
var child2 = verify(node.child2);
var balance = b2Abs(child2.height - child1.height);
return b2Max(maxBalance, balance);
};
b2DynamicTree.prototype.GetMaxBalance = function () {
var maxBalance = b2DynamicTree.GetMaxBalanceNode(this.m_root, 0);
/*
int32 maxBalance = 0;
for (int32 i = 0; i < m_nodeCapacity; ++i) {
const b2TreeNode<T>* node = m_nodes + i;
if (node.height <= 1) {
continue;
}
b2Assert(!node.IsLeaf());
int32 child1 = node.child1;
int32 child2 = node.child2;
int32 balance = b2Abs(m_nodes[child2].height - m_nodes[child1].height);
maxBalance = b2Max(maxBalance, balance);
}
*/
return maxBalance;
};
b2DynamicTree.prototype.RebuildBottomUp = function () {
/*
int32* nodes = (int32*)b2Alloc(m_nodeCount * sizeof(int32));
int32 count = 0;
// Build array of leaves. Free the rest.
for (int32 i = 0; i < m_nodeCapacity; ++i) {
if (m_nodes[i].height < 0) {
// free node in pool
continue;
}
if (m_nodes[i].IsLeaf()) {
m_nodes[i].parent = b2_nullNode;
nodes[count] = i;
++count;
} else {
FreeNode(i);
}
}
while (count > 1) {
float32 minCost = b2_maxFloat;
int32 iMin = -1, jMin = -1;
for (int32 i = 0; i < count; ++i) {
b2AABB aabbi = m_nodes[nodes[i]].aabb;
for (int32 j = i + 1; j < count; ++j) {
b2AABB aabbj = m_nodes[nodes[j]].aabb;
b2AABB b;
b.Combine(aabbi, aabbj);
float32 cost = b.GetPerimeter();
if (cost < minCost) {
iMin = i;
jMin = j;
minCost = cost;
}
}
}
int32 index1 = nodes[iMin];
int32 index2 = nodes[jMin];
b2TreeNode<T>* child1 = m_nodes + index1;
b2TreeNode<T>* child2 = m_nodes + index2;
int32 parentIndex = AllocateNode();
b2TreeNode<T>* parent = m_nodes + parentIndex;
parent.child1 = index1;
parent.child2 = index2;
parent.height = 1 + b2Max(child1.height, child2.height);
parent.aabb.Combine(child1.aabb, child2.aabb);
parent.parent = b2_nullNode;
child1.parent = parentIndex;
child2.parent = parentIndex;
nodes[jMin] = nodes[count-1];
nodes[iMin] = parentIndex;
--count;
}
m_root = nodes[0];
b2Free(nodes);
*/
this.Validate();
};
b2DynamicTree.ShiftOriginNode = function (node, newOrigin) {
if (node === null) {
return;
}
if (node.height <= 1) {
return;
}
// DEBUG: b2Assert(!node.IsLeaf());
var child1 = node.child1;
var child2 = node.child2;
b2DynamicTree.ShiftOriginNode(child1, newOrigin);
b2DynamicTree.ShiftOriginNode(child2, newOrigin);
node.aabb.lowerBound.SelfSub(newOrigin);
node.aabb.upperBound.SelfSub(newOrigin);
};
b2DynamicTree.prototype.ShiftOrigin = function (newOrigin) {
b2DynamicTree.ShiftOriginNode(this.m_root, newOrigin);
/*
// Build array of leaves. Free the rest.
for (int32 i = 0; i < m_nodeCapacity; ++i) {
m_nodes[i].aabb.lowerBound -= newOrigin;
m_nodes[i].aabb.upperBound -= newOrigin;
}
*/
};
b2DynamicTree.s_r = new b2Vec2();
b2DynamicTree.s_v = new b2Vec2();
b2DynamicTree.s_abs_v = new b2Vec2();
b2DynamicTree.s_segmentAABB = new b2AABB();
b2DynamicTree.s_subInput = new b2RayCastInput();
b2DynamicTree.s_combinedAABB = new b2AABB();
b2DynamicTree.s_aabb = new b2AABB();
b2DynamicTree.s_node_id = 0;
return b2DynamicTree;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2Pair = /** @class */ (function () {
function b2Pair(proxyA, proxyB) {
this.proxyA = proxyA;
this.proxyB = proxyB;
}
return b2Pair;
}());
/// The broad-phase is used for computing pairs and performing volume queries and ray casts.
/// This broad-phase does not persist pairs. Instead, this reports potentially new pairs.
/// It is up to the client to consume the new pairs and to track subsequent overlap.
var b2BroadPhase = /** @class */ (function () {
function b2BroadPhase() {
this.m_tree = new b2DynamicTree();
this.m_proxyCount = 0;
// public m_moveCapacity: number = 16;
this.m_moveCount = 0;
this.m_moveBuffer = [];
// public m_pairCapacity: number = 16;
this.m_pairCount = 0;
this.m_pairBuffer = [];
}
// public m_queryProxyId: number = 0;
/// Create a proxy with an initial AABB. Pairs are not reported until
/// UpdatePairs is called.
b2BroadPhase.prototype.CreateProxy = function (aabb, userData) {
var proxy = this.m_tree.CreateProxy(aabb, userData);
++this.m_proxyCount;
this.BufferMove(proxy);
return proxy;
};
/// Destroy a proxy. It is up to the client to remove any pairs.
b2BroadPhase.prototype.DestroyProxy = function (proxy) {
this.UnBufferMove(proxy);
--this.m_proxyCount;
this.m_tree.DestroyProxy(proxy);
};
/// Call MoveProxy as many times as you like, then when you are done
/// call UpdatePairs to finalized the proxy pairs (for your time step).
b2BroadPhase.prototype.MoveProxy = function (proxy, aabb, displacement) {
var buffer = this.m_tree.MoveProxy(proxy, aabb, displacement);
if (buffer) {
this.BufferMove(proxy);
}
};
/// Call to trigger a re-processing of it's pairs on the next call to UpdatePairs.
b2BroadPhase.prototype.TouchProxy = function (proxy) {
this.BufferMove(proxy);
};
/// Get the fat AABB for a proxy.
// public GetFatAABB(proxy: b2TreeNode<T>): b2AABB {
// return this.m_tree.GetFatAABB(proxy);
// }
/// Get user data from a proxy. Returns NULL if the id is invalid.
// public GetUserData(proxy: b2TreeNode<T>): T {
// return this.m_tree.GetUserData(proxy);
// }
/// Test overlap of fat AABBs.
// public TestOverlap(proxyA: b2TreeNode<T>, proxyB: b2TreeNode<T>): boolean {
// const aabbA: b2AABB = this.m_tree.GetFatAABB(proxyA);
// const aabbB: b2AABB = this.m_tree.GetFatAABB(proxyB);
// return b2TestOverlapAABB(aabbA, aabbB);
// }
/// Get the number of proxies.
b2BroadPhase.prototype.GetProxyCount = function () {
return this.m_proxyCount;
};
/// Update the pairs. This results in pair callbacks. This can only add pairs.
b2BroadPhase.prototype.UpdatePairs = function (callback) {
var _this = this;
// Reset pair buffer
this.m_pairCount = 0;
var _loop_1 = function (i_1) {
var queryProxy = this_1.m_moveBuffer[i_1];
if (queryProxy === null) {
return "continue";
}
// This is called from box2d.b2DynamicTree::Query when we are gathering pairs.
// boolean b2BroadPhase::QueryCallback(int32 proxyId);
// We have to query the tree with the fat AABB so that
// we don't fail to create a pair that may touch later.
var fatAABB = queryProxy.aabb; // this.m_tree.GetFatAABB(queryProxy);
// Query tree, create pairs and add them pair buffer.
this_1.m_tree.Query(fatAABB, function (proxy) {
// A proxy cannot form a pair with itself.
if (proxy.m_id === queryProxy.m_id) {
return true;
}
// const proxyA = proxy < queryProxy ? proxy : queryProxy;
// const proxyB = proxy >= queryProxy ? proxy : queryProxy;
var proxyA;
var proxyB;
if (proxy.m_id < queryProxy.m_id) {
proxyA = proxy;
proxyB = queryProxy;
}
else {
proxyA = queryProxy;
proxyB = proxy;
}
// Grow the pair buffer as needed.
if (_this.m_pairCount === _this.m_pairBuffer.length) {
_this.m_pairBuffer[_this.m_pairCount] = new b2Pair(proxyA, proxyB);
}
else {
var pair = _this.m_pairBuffer[_this.m_pairCount];
pair.proxyA = proxyA;
pair.proxyB = proxyB;
}
++_this.m_pairCount;
return true;
});
};
var this_1 = this;
// Perform tree queries for all moving proxies.
for (var i_1 = 0; i_1 < this.m_moveCount; ++i_1) {
_loop_1(i_1);
}
// Reset move buffer
this.m_moveCount = 0;
// Sort the pair buffer to expose duplicates.
this.m_pairBuffer.length = this.m_pairCount;
this.m_pairBuffer.sort(b2PairLessThan);
// Send the pairs back to the client.
var i = 0;
while (i < this.m_pairCount) {
var primaryPair = this.m_pairBuffer[i];
var userDataA = primaryPair.proxyA.userData; // this.m_tree.GetUserData(primaryPair.proxyA);
var userDataB = primaryPair.proxyB.userData; // this.m_tree.GetUserData(primaryPair.proxyB);
if (userDataA && userDataB) {
callback(userDataA, userDataB);
}
++i;
// Skip any duplicate pairs.
while (i < this.m_pairCount) {
var pair = this.m_pairBuffer[i];
if (pair.proxyA.m_id !== primaryPair.proxyA.m_id || pair.proxyB.m_id !== primaryPair.proxyB.m_id) {
break;
}
++i;
}
}
// Try to keep the tree balanced.
// this.m_tree.Rebalance(4);
};
/// Query an AABB for overlapping proxies. The callback class
/// is called for each proxy that overlaps the supplied AABB.
b2BroadPhase.prototype.Query = function (aabb, callback) {
this.m_tree.Query(aabb, callback);
};
b2BroadPhase.prototype.QueryPoint = function (point, callback) {
this.m_tree.QueryPoint(point, callback);
};
/// Ray-cast against the proxies in the tree. This relies on the callback
/// to perform a exact ray-cast in the case were the proxy contains a shape.
/// The callback also performs the any collision filtering. This has performance
/// roughly equal to k * log(n), where k is the number of collisions and n is the
/// number of proxies in the tree.
/// @param input the ray-cast input data. The ray extends from p1 to p1 + maxFraction * (p2 - p1).
/// @param callback a callback class that is called for each proxy that is hit by the ray.
b2BroadPhase.prototype.RayCast = function (input, callback) {
this.m_tree.RayCast(input, callback);
};
/// Get the height of the embedded tree.
b2BroadPhase.prototype.GetTreeHeight = function () {
return this.m_tree.GetHeight();
};
/// Get the balance of the embedded tree.
b2BroadPhase.prototype.GetTreeBalance = function () {
return this.m_tree.GetMaxBalance();
};
/// Get the quality metric of the embedded tree.
b2BroadPhase.prototype.GetTreeQuality = function () {
return this.m_tree.GetAreaRatio();
};
/// Shift the world origin. Useful for large worlds.
/// The shift formula is: position -= newOrigin
/// @param newOrigin the new origin with respect to the old origin
b2BroadPhase.prototype.ShiftOrigin = function (newOrigin) {
this.m_tree.ShiftOrigin(newOrigin);
};
b2BroadPhase.prototype.BufferMove = function (proxy) {
this.m_moveBuffer[this.m_moveCount] = proxy;
++this.m_moveCount;
};
b2BroadPhase.prototype.UnBufferMove = function (proxy) {
var i = this.m_moveBuffer.indexOf(proxy);
this.m_moveBuffer[i] = null;
};
return b2BroadPhase;
}());
/// This is used to sort pairs.
function b2PairLessThan(pair1, pair2) {
if (pair1.proxyA.m_id === pair2.proxyA.m_id) {
return pair1.proxyB.m_id - pair2.proxyB.m_id;
}
return pair1.proxyA.m_id - pair2.proxyA.m_id;
}
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
exports.b2_toiTime = 0;
exports.b2_toiMaxTime = 0;
exports.b2_toiCalls = 0;
exports.b2_toiIters = 0;
exports.b2_toiMaxIters = 0;
exports.b2_toiRootIters = 0;
exports.b2_toiMaxRootIters = 0;
function b2_toi_reset() {
exports.b2_toiTime = 0;
exports.b2_toiMaxTime = 0;
exports.b2_toiCalls = 0;
exports.b2_toiIters = 0;
exports.b2_toiMaxIters = 0;
exports.b2_toiRootIters = 0;
exports.b2_toiMaxRootIters = 0;
}
var b2TimeOfImpact_s_xfA = new b2Transform();
var b2TimeOfImpact_s_xfB = new b2Transform();
var b2TimeOfImpact_s_pointA = new b2Vec2();
var b2TimeOfImpact_s_pointB = new b2Vec2();
var b2TimeOfImpact_s_normal = new b2Vec2();
var b2TimeOfImpact_s_axisA = new b2Vec2();
var b2TimeOfImpact_s_axisB = new b2Vec2();
/// Input parameters for b2TimeOfImpact
var b2TOIInput = /** @class */ (function () {
function b2TOIInput() {
this.proxyA = new b2DistanceProxy();
this.proxyB = new b2DistanceProxy();
this.sweepA = new b2Sweep();
this.sweepB = new b2Sweep();
this.tMax = 0; // defines sweep interval [0, tMax]
}
return b2TOIInput;
}());
(function (b2TOIOutputState) {
b2TOIOutputState[b2TOIOutputState["e_unknown"] = 0] = "e_unknown";
b2TOIOutputState[b2TOIOutputState["e_failed"] = 1] = "e_failed";
b2TOIOutputState[b2TOIOutputState["e_overlapped"] = 2] = "e_overlapped";
b2TOIOutputState[b2TOIOutputState["e_touching"] = 3] = "e_touching";
b2TOIOutputState[b2TOIOutputState["e_separated"] = 4] = "e_separated";
})(exports.b2TOIOutputState || (exports.b2TOIOutputState = {}));
var b2TOIOutput = /** @class */ (function () {
function b2TOIOutput() {
this.state = exports.b2TOIOutputState.e_unknown;
this.t = 0;
}
return b2TOIOutput;
}());
(function (b2SeparationFunctionType) {
b2SeparationFunctionType[b2SeparationFunctionType["e_unknown"] = -1] = "e_unknown";
b2SeparationFunctionType[b2SeparationFunctionType["e_points"] = 0] = "e_points";
b2SeparationFunctionType[b2SeparationFunctionType["e_faceA"] = 1] = "e_faceA";
b2SeparationFunctionType[b2SeparationFunctionType["e_faceB"] = 2] = "e_faceB";
})(exports.b2SeparationFunctionType || (exports.b2SeparationFunctionType = {}));
var b2SeparationFunction = /** @class */ (function () {
function b2SeparationFunction() {
this.m_sweepA = new b2Sweep();
this.m_sweepB = new b2Sweep();
this.m_type = exports.b2SeparationFunctionType.e_unknown;
this.m_localPoint = new b2Vec2();
this.m_axis = new b2Vec2();
}
b2SeparationFunction.prototype.Initialize = function (cache, proxyA, sweepA, proxyB, sweepB, t1) {
this.m_proxyA = proxyA;
this.m_proxyB = proxyB;
var count = cache.count;
// DEBUG: b2Assert(0 < count && count < 3);
this.m_sweepA.Copy(sweepA);
this.m_sweepB.Copy(sweepB);
var xfA = b2TimeOfImpact_s_xfA;
var xfB = b2TimeOfImpact_s_xfB;
this.m_sweepA.GetTransform(xfA, t1);
this.m_sweepB.GetTransform(xfB, t1);
if (count === 1) {
this.m_type = exports.b2SeparationFunctionType.e_points;
var localPointA = this.m_proxyA.GetVertex(cache.indexA[0]);
var localPointB = this.m_proxyB.GetVertex(cache.indexB[0]);
var pointA = b2Transform.MulXV(xfA, localPointA, b2TimeOfImpact_s_pointA);
var pointB = b2Transform.MulXV(xfB, localPointB, b2TimeOfImpact_s_pointB);
b2Vec2.SubVV(pointB, pointA, this.m_axis);
var s = this.m_axis.Normalize();
// #if B2_ENABLE_PARTICLE
this.m_localPoint.SetZero();
// #endif
return s;
}
else if (cache.indexA[0] === cache.indexA[1]) {
// Two points on B and one on A.
this.m_type = exports.b2SeparationFunctionType.e_faceB;
var localPointB1 = this.m_proxyB.GetVertex(cache.indexB[0]);
var localPointB2 = this.m_proxyB.GetVertex(cache.indexB[1]);
b2Vec2.CrossVOne(b2Vec2.SubVV(localPointB2, localPointB1, b2Vec2.s_t0), this.m_axis).SelfNormalize();
var normal = b2Rot.MulRV(xfB.q, this.m_axis, b2TimeOfImpact_s_normal);
b2Vec2.MidVV(localPointB1, localPointB2, this.m_localPoint);
var pointB = b2Transform.MulXV(xfB, this.m_localPoint, b2TimeOfImpact_s_pointB);
var localPointA = this.m_proxyA.GetVertex(cache.indexA[0]);
var pointA = b2Transform.MulXV(xfA, localPointA, b2TimeOfImpact_s_pointA);
var s = b2Vec2.DotVV(b2Vec2.SubVV(pointA, pointB, b2Vec2.s_t0), normal);
if (s < 0) {
this.m_axis.SelfNeg();
s = -s;
}
return s;
}
else {
// Two points on A and one or two points on B.
this.m_type = exports.b2SeparationFunctionType.e_faceA;
var localPointA1 = this.m_proxyA.GetVertex(cache.indexA[0]);
var localPointA2 = this.m_proxyA.GetVertex(cache.indexA[1]);
b2Vec2.CrossVOne(b2Vec2.SubVV(localPointA2, localPointA1, b2Vec2.s_t0), this.m_axis).SelfNormalize();
var normal = b2Rot.MulRV(xfA.q, this.m_axis, b2TimeOfImpact_s_normal);
b2Vec2.MidVV(localPointA1, localPointA2, this.m_localPoint);
var pointA = b2Transform.MulXV(xfA, this.m_localPoint, b2TimeOfImpact_s_pointA);
var localPointB = this.m_proxyB.GetVertex(cache.indexB[0]);
var pointB = b2Transform.MulXV(xfB, localPointB, b2TimeOfImpact_s_pointB);
var s = b2Vec2.DotVV(b2Vec2.SubVV(pointB, pointA, b2Vec2.s_t0), normal);
if (s < 0) {
this.m_axis.SelfNeg();
s = -s;
}
return s;
}
};
b2SeparationFunction.prototype.FindMinSeparation = function (indexA, indexB, t) {
var xfA = b2TimeOfImpact_s_xfA;
var xfB = b2TimeOfImpact_s_xfB;
this.m_sweepA.GetTransform(xfA, t);
this.m_sweepB.GetTransform(xfB, t);
switch (this.m_type) {
case exports.b2SeparationFunctionType.e_points: {
var axisA = b2Rot.MulTRV(xfA.q, this.m_axis, b2TimeOfImpact_s_axisA);
var axisB = b2Rot.MulTRV(xfB.q, b2Vec2.NegV(this.m_axis, b2Vec2.s_t0), b2TimeOfImpact_s_axisB);
indexA[0] = this.m_proxyA.GetSupport(axisA);
indexB[0] = this.m_proxyB.GetSupport(axisB);
var localPointA = this.m_proxyA.GetVertex(indexA[0]);
var localPointB = this.m_proxyB.GetVertex(indexB[0]);
var pointA = b2Transform.MulXV(xfA, localPointA, b2TimeOfImpact_s_pointA);
var pointB = b2Transform.MulXV(xfB, localPointB, b2TimeOfImpact_s_pointB);
var separation = b2Vec2.DotVV(b2Vec2.SubVV(pointB, pointA, b2Vec2.s_t0), this.m_axis);
return separation;
}
case exports.b2SeparationFunctionType.e_faceA: {
var normal = b2Rot.MulRV(xfA.q, this.m_axis, b2TimeOfImpact_s_normal);
var pointA = b2Transform.MulXV(xfA, this.m_localPoint, b2TimeOfImpact_s_pointA);
var axisB = b2Rot.MulTRV(xfB.q, b2Vec2.NegV(normal, b2Vec2.s_t0), b2TimeOfImpact_s_axisB);
indexA[0] = -1;
indexB[0] = this.m_proxyB.GetSupport(axisB);
var localPointB = this.m_proxyB.GetVertex(indexB[0]);
var pointB = b2Transform.MulXV(xfB, localPointB, b2TimeOfImpact_s_pointB);
var separation = b2Vec2.DotVV(b2Vec2.SubVV(pointB, pointA, b2Vec2.s_t0), normal);
return separation;
}
case exports.b2SeparationFunctionType.e_faceB: {
var normal = b2Rot.MulRV(xfB.q, this.m_axis, b2TimeOfImpact_s_normal);
var pointB = b2Transform.MulXV(xfB, this.m_localPoint, b2TimeOfImpact_s_pointB);
var axisA = b2Rot.MulTRV(xfA.q, b2Vec2.NegV(normal, b2Vec2.s_t0), b2TimeOfImpact_s_axisA);
indexB[0] = -1;
indexA[0] = this.m_proxyA.GetSupport(axisA);
var localPointA = this.m_proxyA.GetVertex(indexA[0]);
var pointA = b2Transform.MulXV(xfA, localPointA, b2TimeOfImpact_s_pointA);
var separation = b2Vec2.DotVV(b2Vec2.SubVV(pointA, pointB, b2Vec2.s_t0), normal);
return separation;
}
default:
// DEBUG: b2Assert(false);
indexA[0] = -1;
indexB[0] = -1;
return 0;
}
};
b2SeparationFunction.prototype.Evaluate = function (indexA, indexB, t) {
var xfA = b2TimeOfImpact_s_xfA;
var xfB = b2TimeOfImpact_s_xfB;
this.m_sweepA.GetTransform(xfA, t);
this.m_sweepB.GetTransform(xfB, t);
switch (this.m_type) {
case exports.b2SeparationFunctionType.e_points: {
var localPointA = this.m_proxyA.GetVertex(indexA);
var localPointB = this.m_proxyB.GetVertex(indexB);
var pointA = b2Transform.MulXV(xfA, localPointA, b2TimeOfImpact_s_pointA);
var pointB = b2Transform.MulXV(xfB, localPointB, b2TimeOfImpact_s_pointB);
var separation = b2Vec2.DotVV(b2Vec2.SubVV(pointB, pointA, b2Vec2.s_t0), this.m_axis);
return separation;
}
case exports.b2SeparationFunctionType.e_faceA: {
var normal = b2Rot.MulRV(xfA.q, this.m_axis, b2TimeOfImpact_s_normal);
var pointA = b2Transform.MulXV(xfA, this.m_localPoint, b2TimeOfImpact_s_pointA);
var localPointB = this.m_proxyB.GetVertex(indexB);
var pointB = b2Transform.MulXV(xfB, localPointB, b2TimeOfImpact_s_pointB);
var separation = b2Vec2.DotVV(b2Vec2.SubVV(pointB, pointA, b2Vec2.s_t0), normal);
return separation;
}
case exports.b2SeparationFunctionType.e_faceB: {
var normal = b2Rot.MulRV(xfB.q, this.m_axis, b2TimeOfImpact_s_normal);
var pointB = b2Transform.MulXV(xfB, this.m_localPoint, b2TimeOfImpact_s_pointB);
var localPointA = this.m_proxyA.GetVertex(indexA);
var pointA = b2Transform.MulXV(xfA, localPointA, b2TimeOfImpact_s_pointA);
var separation = b2Vec2.DotVV(b2Vec2.SubVV(pointA, pointB, b2Vec2.s_t0), normal);
return separation;
}
default:
// DEBUG: b2Assert(false);
return 0;
}
};
return b2SeparationFunction;
}());
var b2TimeOfImpact_s_timer = new b2Timer();
var b2TimeOfImpact_s_cache = new b2SimplexCache();
var b2TimeOfImpact_s_distanceInput = new b2DistanceInput();
var b2TimeOfImpact_s_distanceOutput = new b2DistanceOutput();
var b2TimeOfImpact_s_fcn = new b2SeparationFunction();
var b2TimeOfImpact_s_indexA = [0];
var b2TimeOfImpact_s_indexB = [0];
var b2TimeOfImpact_s_sweepA = new b2Sweep();
var b2TimeOfImpact_s_sweepB = new b2Sweep();
function b2TimeOfImpact(output, input) {
var timer = b2TimeOfImpact_s_timer.Reset();
++exports.b2_toiCalls;
output.state = exports.b2TOIOutputState.e_unknown;
output.t = input.tMax;
var proxyA = input.proxyA;
var proxyB = input.proxyB;
var sweepA = b2TimeOfImpact_s_sweepA.Copy(input.sweepA);
var sweepB = b2TimeOfImpact_s_sweepB.Copy(input.sweepB);
// Large rotations can make the root finder fail, so we normalize the
// sweep angles.
sweepA.Normalize();
sweepB.Normalize();
var tMax = input.tMax;
var totalRadius = proxyA.m_radius + proxyB.m_radius;
var target = b2Max(b2_linearSlop, totalRadius - 3 * b2_linearSlop);
var tolerance = 0.25 * b2_linearSlop;
// DEBUG: b2Assert(target > tolerance);
var t1 = 0;
var k_maxIterations = 20; // TODO_ERIN b2Settings
var iter = 0;
// Prepare input for distance query.
var cache = b2TimeOfImpact_s_cache;
cache.count = 0;
var distanceInput = b2TimeOfImpact_s_distanceInput;
distanceInput.proxyA.Copy(input.proxyA);
distanceInput.proxyB.Copy(input.proxyB);
distanceInput.useRadii = false;
// The outer loop progressively attempts to compute new separating axes.
// This loop terminates when an axis is repeated (no progress is made).
for (;;) {
var xfA = b2TimeOfImpact_s_xfA;
var xfB = b2TimeOfImpact_s_xfB;
sweepA.GetTransform(xfA, t1);
sweepB.GetTransform(xfB, t1);
// Get the distance between shapes. We can also use the results
// to get a separating axis.
distanceInput.transformA.Copy(xfA);
distanceInput.transformB.Copy(xfB);
var distanceOutput = b2TimeOfImpact_s_distanceOutput;
b2Distance(distanceOutput, cache, distanceInput);
// If the shapes are overlapped, we give up on continuous collision.
if (distanceOutput.distance <= 0) {
// Failure!
output.state = exports.b2TOIOutputState.e_overlapped;
output.t = 0;
break;
}
if (distanceOutput.distance < target + tolerance) {
// Victory!
output.state = exports.b2TOIOutputState.e_touching;
output.t = t1;
break;
}
// Initialize the separating axis.
var fcn = b2TimeOfImpact_s_fcn;
fcn.Initialize(cache, proxyA, sweepA, proxyB, sweepB, t1);
/*
#if 0
// Dump the curve seen by the root finder {
const int32 N = 100;
float32 dx = 1.0f / N;
float32 xs[N+1];
float32 fs[N+1];
float32 x = 0.0f;
for (int32 i = 0; i <= N; ++i) {
sweepA.GetTransform(&xfA, x);
sweepB.GetTransform(&xfB, x);
float32 f = fcn.Evaluate(xfA, xfB) - target;
printf("%g %g\n", x, f);
xs[i] = x;
fs[i] = f;
x += dx;
}
}
#endif
*/
// Compute the TOI on the separating axis. We do this by successively
// resolving the deepest point. This loop is bounded by the number of vertices.
var done = false;
var t2 = tMax;
var pushBackIter = 0;
for (;;) {
// Find the deepest point at t2. Store the witness point indices.
var indexA = b2TimeOfImpact_s_indexA;
var indexB = b2TimeOfImpact_s_indexB;
var s2 = fcn.FindMinSeparation(indexA, indexB, t2);
// Is the final configuration separated?
if (s2 > (target + tolerance)) {
// Victory!
output.state = exports.b2TOIOutputState.e_separated;
output.t = tMax;
done = true;
break;
}
// Has the separation reached tolerance?
if (s2 > (target - tolerance)) {
// Advance the sweeps
t1 = t2;
break;
}
// Compute the initial separation of the witness points.
var s1 = fcn.Evaluate(indexA[0], indexB[0], t1);
// Check for initial overlap. This might happen if the root finder
// runs out of iterations.
if (s1 < (target - tolerance)) {
output.state = exports.b2TOIOutputState.e_failed;
output.t = t1;
done = true;
break;
}
// Check for touching
if (s1 <= (target + tolerance)) {
// Victory! t1 should hold the TOI (could be 0.0).
output.state = exports.b2TOIOutputState.e_touching;
output.t = t1;
done = true;
break;
}
// Compute 1D root of: f(x) - target = 0
var rootIterCount = 0;
var a1 = t1;
var a2 = t2;
for (;;) {
// Use a mix of the secant rule and bisection.
var t = 0;
if (rootIterCount & 1) {
// Secant rule to improve convergence.
t = a1 + (target - s1) * (a2 - a1) / (s2 - s1);
}
else {
// Bisection to guarantee progress.
t = 0.5 * (a1 + a2);
}
++rootIterCount;
++exports.b2_toiRootIters;
var s = fcn.Evaluate(indexA[0], indexB[0], t);
if (b2Abs(s - target) < tolerance) {
// t2 holds a tentative value for t1
t2 = t;
break;
}
// Ensure we continue to bracket the root.
if (s > target) {
a1 = t;
s1 = s;
}
else {
a2 = t;
s2 = s;
}
if (rootIterCount === 50) {
break;
}
}
exports.b2_toiMaxRootIters = b2Max(exports.b2_toiMaxRootIters, rootIterCount);
++pushBackIter;
if (pushBackIter === b2_maxPolygonVertices) {
break;
}
}
++iter;
++exports.b2_toiIters;
if (done) {
break;
}
if (iter === k_maxIterations) {
// Root finder got stuck. Semi-victory.
output.state = exports.b2TOIOutputState.e_failed;
output.t = t1;
break;
}
}
exports.b2_toiMaxIters = b2Max(exports.b2_toiMaxIters, iter);
var time = timer.GetMilliseconds();
exports.b2_toiMaxTime = b2Max(exports.b2_toiMaxTime, time);
exports.b2_toiTime += time;
}
var b2CollideCircles_s_pA = new b2Vec2();
var b2CollideCircles_s_pB = new b2Vec2();
function b2CollideCircles(manifold, circleA, xfA, circleB, xfB) {
manifold.pointCount = 0;
var pA = b2Transform.MulXV(xfA, circleA.m_p, b2CollideCircles_s_pA);
var pB = b2Transform.MulXV(xfB, circleB.m_p, b2CollideCircles_s_pB);
var distSqr = b2Vec2.DistanceSquaredVV(pA, pB);
var radius = circleA.m_radius + circleB.m_radius;
if (distSqr > radius * radius) {
return;
}
manifold.type = exports.b2ManifoldType.e_circles;
manifold.localPoint.Copy(circleA.m_p);
manifold.localNormal.SetZero();
manifold.pointCount = 1;
manifold.points[0].localPoint.Copy(circleB.m_p);
manifold.points[0].id.key = 0;
}
var b2CollidePolygonAndCircle_s_c = new b2Vec2();
var b2CollidePolygonAndCircle_s_cLocal = new b2Vec2();
var b2CollidePolygonAndCircle_s_faceCenter = new b2Vec2();
function b2CollidePolygonAndCircle(manifold, polygonA, xfA, circleB, xfB) {
manifold.pointCount = 0;
// Compute circle position in the frame of the polygon.
var c = b2Transform.MulXV(xfB, circleB.m_p, b2CollidePolygonAndCircle_s_c);
var cLocal = b2Transform.MulTXV(xfA, c, b2CollidePolygonAndCircle_s_cLocal);
// Find the min separating edge.
var normalIndex = 0;
var separation = (-b2_maxFloat);
var radius = polygonA.m_radius + circleB.m_radius;
var vertexCount = polygonA.m_count;
var vertices = polygonA.m_vertices;
var normals = polygonA.m_normals;
for (var i = 0; i < vertexCount; ++i) {
var s = b2Vec2.DotVV(normals[i], b2Vec2.SubVV(cLocal, vertices[i], b2Vec2.s_t0));
if (s > radius) {
// Early out.
return;
}
if (s > separation) {
separation = s;
normalIndex = i;
}
}
// Vertices that subtend the incident face.
var vertIndex1 = normalIndex;
var vertIndex2 = (vertIndex1 + 1) % vertexCount;
var v1 = vertices[vertIndex1];
var v2 = vertices[vertIndex2];
// If the center is inside the polygon ...
if (separation < b2_epsilon) {
manifold.pointCount = 1;
manifold.type = exports.b2ManifoldType.e_faceA;
manifold.localNormal.Copy(normals[normalIndex]);
b2Vec2.MidVV(v1, v2, manifold.localPoint);
manifold.points[0].localPoint.Copy(circleB.m_p);
manifold.points[0].id.key = 0;
return;
}
// Compute barycentric coordinates
var u1 = b2Vec2.DotVV(b2Vec2.SubVV(cLocal, v1, b2Vec2.s_t0), b2Vec2.SubVV(v2, v1, b2Vec2.s_t1));
var u2 = b2Vec2.DotVV(b2Vec2.SubVV(cLocal, v2, b2Vec2.s_t0), b2Vec2.SubVV(v1, v2, b2Vec2.s_t1));
if (u1 <= 0) {
if (b2Vec2.DistanceSquaredVV(cLocal, v1) > radius * radius) {
return;
}
manifold.pointCount = 1;
manifold.type = exports.b2ManifoldType.e_faceA;
b2Vec2.SubVV(cLocal, v1, manifold.localNormal).SelfNormalize();
manifold.localPoint.Copy(v1);
manifold.points[0].localPoint.Copy(circleB.m_p);
manifold.points[0].id.key = 0;
}
else if (u2 <= 0) {
if (b2Vec2.DistanceSquaredVV(cLocal, v2) > radius * radius) {
return;
}
manifold.pointCount = 1;
manifold.type = exports.b2ManifoldType.e_faceA;
b2Vec2.SubVV(cLocal, v2, manifold.localNormal).SelfNormalize();
manifold.localPoint.Copy(v2);
manifold.points[0].localPoint.Copy(circleB.m_p);
manifold.points[0].id.key = 0;
}
else {
var faceCenter = b2Vec2.MidVV(v1, v2, b2CollidePolygonAndCircle_s_faceCenter);
var separation_1 = b2Vec2.DotVV(b2Vec2.SubVV(cLocal, faceCenter, b2Vec2.s_t1), normals[vertIndex1]);
if (separation_1 > radius) {
return;
}
manifold.pointCount = 1;
manifold.type = exports.b2ManifoldType.e_faceA;
manifold.localNormal.Copy(normals[vertIndex1]).SelfNormalize();
manifold.localPoint.Copy(faceCenter);
manifold.points[0].localPoint.Copy(circleB.m_p);
manifold.points[0].id.key = 0;
}
}
// DEBUG: import { b2Assert } from "../Common/b2Settings";
var b2EdgeSeparation_s_normal1World = new b2Vec2();
var b2EdgeSeparation_s_normal1 = new b2Vec2();
var b2EdgeSeparation_s_v1 = new b2Vec2();
var b2EdgeSeparation_s_v2 = new b2Vec2();
function b2EdgeSeparation(poly1, xf1, edge1, poly2, xf2) {
// DEBUG: const count1: number = poly1.m_count;
var vertices1 = poly1.m_vertices;
var normals1 = poly1.m_normals;
var count2 = poly2.m_count;
var vertices2 = poly2.m_vertices;
// DEBUG: b2Assert(0 <= edge1 && edge1 < count1);
// Convert normal from poly1's frame into poly2's frame.
var normal1World = b2Rot.MulRV(xf1.q, normals1[edge1], b2EdgeSeparation_s_normal1World);
var normal1 = b2Rot.MulTRV(xf2.q, normal1World, b2EdgeSeparation_s_normal1);
// Find support vertex on poly2 for -normal.
var index = 0;
var minDot = b2_maxFloat;
for (var i = 0; i < count2; ++i) {
var dot = b2Vec2.DotVV(vertices2[i], normal1);
if (dot < minDot) {
minDot = dot;
index = i;
}
}
var v1 = b2Transform.MulXV(xf1, vertices1[edge1], b2EdgeSeparation_s_v1);
var v2 = b2Transform.MulXV(xf2, vertices2[index], b2EdgeSeparation_s_v2);
var separation = b2Vec2.DotVV(b2Vec2.SubVV(v2, v1, b2Vec2.s_t0), normal1World);
return separation;
}
var b2FindMaxSeparation_s_d = new b2Vec2();
var b2FindMaxSeparation_s_dLocal1 = new b2Vec2();
function b2FindMaxSeparation(edgeIndex, poly1, xf1, poly2, xf2) {
var count1 = poly1.m_count;
var normals1 = poly1.m_normals;
// Vector pointing from the centroid of poly1 to the centroid of poly2.
var d = b2Vec2.SubVV(b2Transform.MulXV(xf2, poly2.m_centroid, b2Vec2.s_t0), b2Transform.MulXV(xf1, poly1.m_centroid, b2Vec2.s_t1), b2FindMaxSeparation_s_d);
var dLocal1 = b2Rot.MulTRV(xf1.q, d, b2FindMaxSeparation_s_dLocal1);
// Find edge normal on poly1 that has the largest projection onto d.
var edge = 0;
var maxDot = (-b2_maxFloat);
for (var i = 0; i < count1; ++i) {
var dot = b2Vec2.DotVV(normals1[i], dLocal1);
if (dot > maxDot) {
maxDot = dot;
edge = i;
}
}
// Get the separation for the edge normal.
var s = b2EdgeSeparation(poly1, xf1, edge, poly2, xf2);
// Check the separation for the previous edge normal.
var prevEdge = (edge + count1 - 1) % count1;
var sPrev = b2EdgeSeparation(poly1, xf1, prevEdge, poly2, xf2);
// Check the separation for the next edge normal.
var nextEdge = (edge + 1) % count1;
var sNext = b2EdgeSeparation(poly1, xf1, nextEdge, poly2, xf2);
// Find the best edge and the search direction.
var bestEdge = 0;
var bestSeparation = 0;
var increment = 0;
if (sPrev > s && sPrev > sNext) {
increment = -1;
bestEdge = prevEdge;
bestSeparation = sPrev;
}
else if (sNext > s) {
increment = 1;
bestEdge = nextEdge;
bestSeparation = sNext;
}
else {
edgeIndex[0] = edge;
return s;
}
// Perform a local search for the best edge normal.
while (true) {
if (increment === -1) {
edge = (bestEdge + count1 - 1) % count1;
}
else {
edge = (bestEdge + 1) % count1;
}
s = b2EdgeSeparation(poly1, xf1, edge, poly2, xf2);
if (s > bestSeparation) {
bestEdge = edge;
bestSeparation = s;
}
else {
break;
}
}
edgeIndex[0] = bestEdge;
return bestSeparation;
}
var b2FindIncidentEdge_s_normal1 = new b2Vec2();
function b2FindIncidentEdge(c, poly1, xf1, edge1, poly2, xf2) {
// DEBUG: const count1: number = poly1.m_count;
var normals1 = poly1.m_normals;
var count2 = poly2.m_count;
var vertices2 = poly2.m_vertices;
var normals2 = poly2.m_normals;
// DEBUG: b2Assert(0 <= edge1 && edge1 < count1);
// Get the normal of the reference edge in poly2's frame.
var normal1 = b2Rot.MulTRV(xf2.q, b2Rot.MulRV(xf1.q, normals1[edge1], b2Vec2.s_t0), b2FindIncidentEdge_s_normal1);
// Find the incident edge on poly2.
var index = 0;
var minDot = b2_maxFloat;
for (var i = 0; i < count2; ++i) {
var dot = b2Vec2.DotVV(normal1, normals2[i]);
if (dot < minDot) {
minDot = dot;
index = i;
}
}
// Build the clip vertices for the incident edge.
var i1 = index;
var i2 = (i1 + 1) % count2;
var c0 = c[0];
b2Transform.MulXV(xf2, vertices2[i1], c0.v);
var cf0 = c0.id.cf;
cf0.indexA = edge1;
cf0.indexB = i1;
cf0.typeA = exports.b2ContactFeatureType.e_face;
cf0.typeB = exports.b2ContactFeatureType.e_vertex;
var c1 = c[1];
b2Transform.MulXV(xf2, vertices2[i2], c1.v);
var cf1 = c1.id.cf;
cf1.indexA = edge1;
cf1.indexB = i2;
cf1.typeA = exports.b2ContactFeatureType.e_face;
cf1.typeB = exports.b2ContactFeatureType.e_vertex;
}
var b2CollidePolygons_s_incidentEdge = b2ClipVertex.MakeArray(2);
var b2CollidePolygons_s_clipPoints1 = b2ClipVertex.MakeArray(2);
var b2CollidePolygons_s_clipPoints2 = b2ClipVertex.MakeArray(2);
var b2CollidePolygons_s_edgeA = [0];
var b2CollidePolygons_s_edgeB = [0];
var b2CollidePolygons_s_localTangent = new b2Vec2();
var b2CollidePolygons_s_localNormal = new b2Vec2();
var b2CollidePolygons_s_planePoint = new b2Vec2();
var b2CollidePolygons_s_normal = new b2Vec2();
var b2CollidePolygons_s_tangent = new b2Vec2();
var b2CollidePolygons_s_ntangent = new b2Vec2();
var b2CollidePolygons_s_v11 = new b2Vec2();
var b2CollidePolygons_s_v12 = new b2Vec2();
function b2CollidePolygons(manifold, polyA, xfA, polyB, xfB) {
manifold.pointCount = 0;
var totalRadius = polyA.m_radius + polyB.m_radius;
var edgeA = b2CollidePolygons_s_edgeA;
edgeA[0] = 0;
var separationA = b2FindMaxSeparation(edgeA, polyA, xfA, polyB, xfB);
if (separationA > totalRadius) {
return;
}
var edgeB = b2CollidePolygons_s_edgeB;
edgeB[0] = 0;
var separationB = b2FindMaxSeparation(edgeB, polyB, xfB, polyA, xfA);
if (separationB > totalRadius) {
return;
}
var poly1; // reference polygon
var poly2; // incident polygon
var xf1, xf2;
var edge1 = 0; // reference edge
var flip = 0;
var k_relativeTol = 0.98;
var k_absoluteTol = 0.001;
if (separationB > k_relativeTol * separationA + k_absoluteTol) {
poly1 = polyB;
poly2 = polyA;
xf1 = xfB;
xf2 = xfA;
edge1 = edgeB[0];
manifold.type = exports.b2ManifoldType.e_faceB;
flip = 1;
}
else {
poly1 = polyA;
poly2 = polyB;
xf1 = xfA;
xf2 = xfB;
edge1 = edgeA[0];
manifold.type = exports.b2ManifoldType.e_faceA;
flip = 0;
}
var incidentEdge = b2CollidePolygons_s_incidentEdge;
b2FindIncidentEdge(incidentEdge, poly1, xf1, edge1, poly2, xf2);
var count1 = poly1.m_count;
var vertices1 = poly1.m_vertices;
var iv1 = edge1;
var iv2 = (edge1 + 1) % count1;
var local_v11 = vertices1[iv1];
var local_v12 = vertices1[iv2];
var localTangent = b2Vec2.SubVV(local_v12, local_v11, b2CollidePolygons_s_localTangent);
localTangent.Normalize();
var localNormal = b2Vec2.CrossVOne(localTangent, b2CollidePolygons_s_localNormal);
var planePoint = b2Vec2.MidVV(local_v11, local_v12, b2CollidePolygons_s_planePoint);
var tangent = b2Rot.MulRV(xf1.q, localTangent, b2CollidePolygons_s_tangent);
var normal = b2Vec2.CrossVOne(tangent, b2CollidePolygons_s_normal);
var v11 = b2Transform.MulXV(xf1, local_v11, b2CollidePolygons_s_v11);
var v12 = b2Transform.MulXV(xf1, local_v12, b2CollidePolygons_s_v12);
// Face offset.
var frontOffset = b2Vec2.DotVV(normal, v11);
// Side offsets, extended by polytope skin thickness.
var sideOffset1 = -b2Vec2.DotVV(tangent, v11) + totalRadius;
var sideOffset2 = b2Vec2.DotVV(tangent, v12) + totalRadius;
// Clip incident edge against extruded edge1 side edges.
var clipPoints1 = b2CollidePolygons_s_clipPoints1;
var clipPoints2 = b2CollidePolygons_s_clipPoints2;
var np;
// Clip to box side 1
var ntangent = b2Vec2.NegV(tangent, b2CollidePolygons_s_ntangent);
np = b2ClipSegmentToLine(clipPoints1, incidentEdge, ntangent, sideOffset1, iv1);
if (np < 2) {
return;
}
// Clip to negative box side 1
np = b2ClipSegmentToLine(clipPoints2, clipPoints1, tangent, sideOffset2, iv2);
if (np < 2) {
return;
}
// Now clipPoints2 contains the clipped points.
manifold.localNormal.Copy(localNormal);
manifold.localPoint.Copy(planePoint);
var pointCount = 0;
for (var i = 0; i < b2_maxManifoldPoints; ++i) {
var cv = clipPoints2[i];
var separation = b2Vec2.DotVV(normal, cv.v) - frontOffset;
if (separation <= totalRadius) {
var cp = manifold.points[pointCount];
b2Transform.MulTXV(xf2, cv.v, cp.localPoint);
cp.id.Copy(cv.id);
if (flip) {
// Swap features
var cf = cp.id.cf;
cp.id.cf.indexA = cf.indexB;
cp.id.cf.indexB = cf.indexA;
cp.id.cf.typeA = cf.typeB;
cp.id.cf.typeB = cf.typeA;
}
++pointCount;
}
}
manifold.pointCount = pointCount;
}
// DEBUG: import { b2Assert } from "../Common/b2Settings";
var b2CollideEdgeAndCircle_s_Q = new b2Vec2();
var b2CollideEdgeAndCircle_s_e = new b2Vec2();
var b2CollideEdgeAndCircle_s_d = new b2Vec2();
var b2CollideEdgeAndCircle_s_e1 = new b2Vec2();
var b2CollideEdgeAndCircle_s_e2 = new b2Vec2();
var b2CollideEdgeAndCircle_s_P = new b2Vec2();
var b2CollideEdgeAndCircle_s_n = new b2Vec2();
var b2CollideEdgeAndCircle_s_id = new b2ContactID();
function b2CollideEdgeAndCircle(manifold, edgeA, xfA, circleB, xfB) {
manifold.pointCount = 0;
// Compute circle in frame of edge
var Q = b2Transform.MulTXV(xfA, b2Transform.MulXV(xfB, circleB.m_p, b2Vec2.s_t0), b2CollideEdgeAndCircle_s_Q);
var A = edgeA.m_vertex1;
var B = edgeA.m_vertex2;
var e = b2Vec2.SubVV(B, A, b2CollideEdgeAndCircle_s_e);
// Barycentric coordinates
var u = b2Vec2.DotVV(e, b2Vec2.SubVV(B, Q, b2Vec2.s_t0));
var v = b2Vec2.DotVV(e, b2Vec2.SubVV(Q, A, b2Vec2.s_t0));
var radius = edgeA.m_radius + circleB.m_radius;
// const cf: b2ContactFeature = new b2ContactFeature();
var id = b2CollideEdgeAndCircle_s_id;
id.cf.indexB = 0;
id.cf.typeB = exports.b2ContactFeatureType.e_vertex;
// Region A
if (v <= 0) {
var P_1 = A;
var d_1 = b2Vec2.SubVV(Q, P_1, b2CollideEdgeAndCircle_s_d);
var dd_1 = b2Vec2.DotVV(d_1, d_1);
if (dd_1 > radius * radius) {
return;
}
// Is there an edge connected to A?
if (edgeA.m_hasVertex0) {
var A1 = edgeA.m_vertex0;
var B1 = A;
var e1 = b2Vec2.SubVV(B1, A1, b2CollideEdgeAndCircle_s_e1);
var u1 = b2Vec2.DotVV(e1, b2Vec2.SubVV(B1, Q, b2Vec2.s_t0));
// Is the circle in Region AB of the previous edge?
if (u1 > 0) {
return;
}
}
id.cf.indexA = 0;
id.cf.typeA = exports.b2ContactFeatureType.e_vertex;
manifold.pointCount = 1;
manifold.type = exports.b2ManifoldType.e_circles;
manifold.localNormal.SetZero();
manifold.localPoint.Copy(P_1);
manifold.points[0].id.Copy(id);
// manifold.points[0].id.key = 0;
// manifold.points[0].id.cf = cf;
manifold.points[0].localPoint.Copy(circleB.m_p);
return;
}
// Region B
if (u <= 0) {
var P_2 = B;
var d_2 = b2Vec2.SubVV(Q, P_2, b2CollideEdgeAndCircle_s_d);
var dd_2 = b2Vec2.DotVV(d_2, d_2);
if (dd_2 > radius * radius) {
return;
}
// Is there an edge connected to B?
if (edgeA.m_hasVertex3) {
var B2 = edgeA.m_vertex3;
var A2 = B;
var e2 = b2Vec2.SubVV(B2, A2, b2CollideEdgeAndCircle_s_e2);
var v2 = b2Vec2.DotVV(e2, b2Vec2.SubVV(Q, A2, b2Vec2.s_t0));
// Is the circle in Region AB of the next edge?
if (v2 > 0) {
return;
}
}
id.cf.indexA = 1;
id.cf.typeA = exports.b2ContactFeatureType.e_vertex;
manifold.pointCount = 1;
manifold.type = exports.b2ManifoldType.e_circles;
manifold.localNormal.SetZero();
manifold.localPoint.Copy(P_2);
manifold.points[0].id.Copy(id);
// manifold.points[0].id.key = 0;
// manifold.points[0].id.cf = cf;
manifold.points[0].localPoint.Copy(circleB.m_p);
return;
}
// Region AB
var den = b2Vec2.DotVV(e, e);
// DEBUG: b2Assert(den > 0);
var P = b2CollideEdgeAndCircle_s_P;
P.x = (1 / den) * (u * A.x + v * B.x);
P.y = (1 / den) * (u * A.y + v * B.y);
var d = b2Vec2.SubVV(Q, P, b2CollideEdgeAndCircle_s_d);
var dd = b2Vec2.DotVV(d, d);
if (dd > radius * radius) {
return;
}
var n = b2CollideEdgeAndCircle_s_n.Set(-e.y, e.x);
if (b2Vec2.DotVV(n, b2Vec2.SubVV(Q, A, b2Vec2.s_t0)) < 0) {
n.Set(-n.x, -n.y);
}
n.Normalize();
id.cf.indexA = 0;
id.cf.typeA = exports.b2ContactFeatureType.e_face;
manifold.pointCount = 1;
manifold.type = exports.b2ManifoldType.e_faceA;
manifold.localNormal.Copy(n);
manifold.localPoint.Copy(A);
manifold.points[0].id.Copy(id);
// manifold.points[0].id.key = 0;
// manifold.points[0].id.cf = cf;
manifold.points[0].localPoint.Copy(circleB.m_p);
}
var b2EPAxis = /** @class */ (function () {
function b2EPAxis() {
this.type = 0 /* e_unknown */;
this.index = 0;
this.separation = 0;
}
return b2EPAxis;
}());
var b2TempPolygon = /** @class */ (function () {
function b2TempPolygon() {
this.vertices = b2Vec2.MakeArray(b2_maxPolygonVertices);
this.normals = b2Vec2.MakeArray(b2_maxPolygonVertices);
this.count = 0;
}
return b2TempPolygon;
}());
var b2ReferenceFace = /** @class */ (function () {
function b2ReferenceFace() {
this.i1 = 0;
this.i2 = 0;
this.v1 = new b2Vec2();
this.v2 = new b2Vec2();
this.normal = new b2Vec2();
this.sideNormal1 = new b2Vec2();
this.sideOffset1 = 0;
this.sideNormal2 = new b2Vec2();
this.sideOffset2 = 0;
}
return b2ReferenceFace;
}());
var b2EPCollider = /** @class */ (function () {
function b2EPCollider() {
this.m_polygonB = new b2TempPolygon();
this.m_xf = new b2Transform();
this.m_centroidB = new b2Vec2();
this.m_v0 = new b2Vec2();
this.m_v1 = new b2Vec2();
this.m_v2 = new b2Vec2();
this.m_v3 = new b2Vec2();
this.m_normal0 = new b2Vec2();
this.m_normal1 = new b2Vec2();
this.m_normal2 = new b2Vec2();
this.m_normal = new b2Vec2();
this.m_type1 = 0 /* e_isolated */;
this.m_type2 = 0 /* e_isolated */;
this.m_lowerLimit = new b2Vec2();
this.m_upperLimit = new b2Vec2();
this.m_radius = 0;
this.m_front = false;
}
b2EPCollider.prototype.Collide = function (manifold, edgeA, xfA, polygonB, xfB) {
b2Transform.MulTXX(xfA, xfB, this.m_xf);
b2Transform.MulXV(this.m_xf, polygonB.m_centroid, this.m_centroidB);
this.m_v0.Copy(edgeA.m_vertex0);
this.m_v1.Copy(edgeA.m_vertex1);
this.m_v2.Copy(edgeA.m_vertex2);
this.m_v3.Copy(edgeA.m_vertex3);
var hasVertex0 = edgeA.m_hasVertex0;
var hasVertex3 = edgeA.m_hasVertex3;
var edge1 = b2Vec2.SubVV(this.m_v2, this.m_v1, b2EPCollider.s_edge1);
edge1.Normalize();
this.m_normal1.Set(edge1.y, -edge1.x);
var offset1 = b2Vec2.DotVV(this.m_normal1, b2Vec2.SubVV(this.m_centroidB, this.m_v1, b2Vec2.s_t0));
var offset0 = 0;
var offset2 = 0;
var convex1 = false;
var convex2 = false;
// Is there a preceding edge?
if (hasVertex0) {
var edge0 = b2Vec2.SubVV(this.m_v1, this.m_v0, b2EPCollider.s_edge0);
edge0.Normalize();
this.m_normal0.Set(edge0.y, -edge0.x);
convex1 = b2Vec2.CrossVV(edge0, edge1) >= 0;
offset0 = b2Vec2.DotVV(this.m_normal0, b2Vec2.SubVV(this.m_centroidB, this.m_v0, b2Vec2.s_t0));
}
// Is there a following edge?
if (hasVertex3) {
var edge2 = b2Vec2.SubVV(this.m_v3, this.m_v2, b2EPCollider.s_edge2);
edge2.Normalize();
this.m_normal2.Set(edge2.y, -edge2.x);
convex2 = b2Vec2.CrossVV(edge1, edge2) > 0;
offset2 = b2Vec2.DotVV(this.m_normal2, b2Vec2.SubVV(this.m_centroidB, this.m_v2, b2Vec2.s_t0));
}
// Determine front or back collision. Determine collision normal limits.
if (hasVertex0 && hasVertex3) {
if (convex1 && convex2) {
this.m_front = offset0 >= 0 || offset1 >= 0 || offset2 >= 0;
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal0);
this.m_upperLimit.Copy(this.m_normal2);
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal1).SelfNeg();
this.m_upperLimit.Copy(this.m_normal1).SelfNeg();
}
}
else if (convex1) {
this.m_front = offset0 >= 0 || (offset1 >= 0 && offset2 >= 0);
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal0);
this.m_upperLimit.Copy(this.m_normal1);
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal2).SelfNeg();
this.m_upperLimit.Copy(this.m_normal1).SelfNeg();
}
}
else if (convex2) {
this.m_front = offset2 >= 0 || (offset0 >= 0 && offset1 >= 0);
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal1);
this.m_upperLimit.Copy(this.m_normal2);
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal1).SelfNeg();
this.m_upperLimit.Copy(this.m_normal0).SelfNeg();
}
}
else {
this.m_front = offset0 >= 0 && offset1 >= 0 && offset2 >= 0;
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal1);
this.m_upperLimit.Copy(this.m_normal1);
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal2).SelfNeg();
this.m_upperLimit.Copy(this.m_normal0).SelfNeg();
}
}
}
else if (hasVertex0) {
if (convex1) {
this.m_front = offset0 >= 0 || offset1 >= 0;
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal0);
this.m_upperLimit.Copy(this.m_normal1).SelfNeg();
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal1);
this.m_upperLimit.Copy(this.m_normal1).SelfNeg();
}
}
else {
this.m_front = offset0 >= 0 && offset1 >= 0;
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal1);
this.m_upperLimit.Copy(this.m_normal1).SelfNeg();
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal1);
this.m_upperLimit.Copy(this.m_normal0).SelfNeg();
}
}
}
else if (hasVertex3) {
if (convex2) {
this.m_front = offset1 >= 0 || offset2 >= 0;
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal1).SelfNeg();
this.m_upperLimit.Copy(this.m_normal2);
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal1).SelfNeg();
this.m_upperLimit.Copy(this.m_normal1);
}
}
else {
this.m_front = offset1 >= 0 && offset2 >= 0;
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal1).SelfNeg();
this.m_upperLimit.Copy(this.m_normal1);
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal2).SelfNeg();
this.m_upperLimit.Copy(this.m_normal1);
}
}
}
else {
this.m_front = offset1 >= 0;
if (this.m_front) {
this.m_normal.Copy(this.m_normal1);
this.m_lowerLimit.Copy(this.m_normal1).SelfNeg();
this.m_upperLimit.Copy(this.m_normal1).SelfNeg();
}
else {
this.m_normal.Copy(this.m_normal1).SelfNeg();
this.m_lowerLimit.Copy(this.m_normal1);
this.m_upperLimit.Copy(this.m_normal1);
}
}
// Get polygonB in frameA
this.m_polygonB.count = polygonB.m_count;
for (var i = 0; i < polygonB.m_count; ++i) {
b2Transform.MulXV(this.m_xf, polygonB.m_vertices[i], this.m_polygonB.vertices[i]);
b2Rot.MulRV(this.m_xf.q, polygonB.m_normals[i], this.m_polygonB.normals[i]);
}
this.m_radius = polygonB.m_radius + edgeA.m_radius;
manifold.pointCount = 0;
var edgeAxis = this.ComputeEdgeSeparation(b2EPCollider.s_edgeAxis);
// If no valid normal can be found than this edge should not collide.
if (edgeAxis.type === 0 /* e_unknown */) {
return;
}
if (edgeAxis.separation > this.m_radius) {
return;
}
var polygonAxis = this.ComputePolygonSeparation(b2EPCollider.s_polygonAxis);
if (polygonAxis.type !== 0 /* e_unknown */ && polygonAxis.separation > this.m_radius) {
return;
}
// Use hysteresis for jitter reduction.
var k_relativeTol = 0.98;
var k_absoluteTol = 0.001;
var primaryAxis;
if (polygonAxis.type === 0 /* e_unknown */) {
primaryAxis = edgeAxis;
}
else if (polygonAxis.separation > k_relativeTol * edgeAxis.separation + k_absoluteTol) {
primaryAxis = polygonAxis;
}
else {
primaryAxis = edgeAxis;
}
var ie = b2EPCollider.s_ie;
var rf = b2EPCollider.s_rf;
if (primaryAxis.type === 1 /* e_edgeA */) {
manifold.type = exports.b2ManifoldType.e_faceA;
// Search for the polygon normal that is most anti-parallel to the edge normal.
var bestIndex = 0;
var bestValue = b2Vec2.DotVV(this.m_normal, this.m_polygonB.normals[0]);
for (var i = 1; i < this.m_polygonB.count; ++i) {
var value = b2Vec2.DotVV(this.m_normal, this.m_polygonB.normals[i]);
if (value < bestValue) {
bestValue = value;
bestIndex = i;
}
}
var i1 = bestIndex;
var i2 = (i1 + 1) % this.m_polygonB.count;
var ie0 = ie[0];
ie0.v.Copy(this.m_polygonB.vertices[i1]);
ie0.id.cf.indexA = 0;
ie0.id.cf.indexB = i1;
ie0.id.cf.typeA = exports.b2ContactFeatureType.e_face;
ie0.id.cf.typeB = exports.b2ContactFeatureType.e_vertex;
var ie1 = ie[1];
ie1.v.Copy(this.m_polygonB.vertices[i2]);
ie1.id.cf.indexA = 0;
ie1.id.cf.indexB = i2;
ie1.id.cf.typeA = exports.b2ContactFeatureType.e_face;
ie1.id.cf.typeB = exports.b2ContactFeatureType.e_vertex;
if (this.m_front) {
rf.i1 = 0;
rf.i2 = 1;
rf.v1.Copy(this.m_v1);
rf.v2.Copy(this.m_v2);
rf.normal.Copy(this.m_normal1);
}
else {
rf.i1 = 1;
rf.i2 = 0;
rf.v1.Copy(this.m_v2);
rf.v2.Copy(this.m_v1);
rf.normal.Copy(this.m_normal1).SelfNeg();
}
}
else {
manifold.type = exports.b2ManifoldType.e_faceB;
var ie0 = ie[0];
ie0.v.Copy(this.m_v1);
ie0.id.cf.indexA = 0;
ie0.id.cf.indexB = primaryAxis.index;
ie0.id.cf.typeA = exports.b2ContactFeatureType.e_vertex;
ie0.id.cf.typeB = exports.b2ContactFeatureType.e_face;
var ie1 = ie[1];
ie1.v.Copy(this.m_v2);
ie1.id.cf.indexA = 0;
ie1.id.cf.indexB = primaryAxis.index;
ie1.id.cf.typeA = exports.b2ContactFeatureType.e_vertex;
ie1.id.cf.typeB = exports.b2ContactFeatureType.e_face;
rf.i1 = primaryAxis.index;
rf.i2 = (rf.i1 + 1) % this.m_polygonB.count;
rf.v1.Copy(this.m_polygonB.vertices[rf.i1]);
rf.v2.Copy(this.m_polygonB.vertices[rf.i2]);
rf.normal.Copy(this.m_polygonB.normals[rf.i1]);
}
rf.sideNormal1.Set(rf.normal.y, -rf.normal.x);
rf.sideNormal2.Copy(rf.sideNormal1).SelfNeg();
rf.sideOffset1 = b2Vec2.DotVV(rf.sideNormal1, rf.v1);
rf.sideOffset2 = b2Vec2.DotVV(rf.sideNormal2, rf.v2);
// Clip incident edge against extruded edge1 side edges.
var clipPoints1 = b2EPCollider.s_clipPoints1;
var clipPoints2 = b2EPCollider.s_clipPoints2;
var np = 0;
// Clip to box side 1
np = b2ClipSegmentToLine(clipPoints1, ie, rf.sideNormal1, rf.sideOffset1, rf.i1);
if (np < b2_maxManifoldPoints) {
return;
}
// Clip to negative box side 1
np = b2ClipSegmentToLine(clipPoints2, clipPoints1, rf.sideNormal2, rf.sideOffset2, rf.i2);
if (np < b2_maxManifoldPoints) {
return;
}
// Now clipPoints2 contains the clipped points.
if (primaryAxis.type === 1 /* e_edgeA */) {
manifold.localNormal.Copy(rf.normal);
manifold.localPoint.Copy(rf.v1);
}
else {
manifold.localNormal.Copy(polygonB.m_normals[rf.i1]);
manifold.localPoint.Copy(polygonB.m_vertices[rf.i1]);
}
var pointCount = 0;
for (var i = 0; i < b2_maxManifoldPoints; ++i) {
var separation = void 0;
separation = b2Vec2.DotVV(rf.normal, b2Vec2.SubVV(clipPoints2[i].v, rf.v1, b2Vec2.s_t0));
if (separation <= this.m_radius) {
var cp = manifold.points[pointCount];
if (primaryAxis.type === 1 /* e_edgeA */) {
b2Transform.MulTXV(this.m_xf, clipPoints2[i].v, cp.localPoint);
cp.id = clipPoints2[i].id;
}
else {
cp.localPoint.Copy(clipPoints2[i].v);
cp.id.cf.typeA = clipPoints2[i].id.cf.typeB;
cp.id.cf.typeB = clipPoints2[i].id.cf.typeA;
cp.id.cf.indexA = clipPoints2[i].id.cf.indexB;
cp.id.cf.indexB = clipPoints2[i].id.cf.indexA;
}
++pointCount;
}
}
manifold.pointCount = pointCount;
};
b2EPCollider.prototype.ComputeEdgeSeparation = function (out) {
var axis = out;
axis.type = 1 /* e_edgeA */;
axis.index = this.m_front ? 0 : 1;
axis.separation = b2_maxFloat;
for (var i = 0; i < this.m_polygonB.count; ++i) {
var s = b2Vec2.DotVV(this.m_normal, b2Vec2.SubVV(this.m_polygonB.vertices[i], this.m_v1, b2Vec2.s_t0));
if (s < axis.separation) {
axis.separation = s;
}
}
return axis;
};
b2EPCollider.prototype.ComputePolygonSeparation = function (out) {
var axis = out;
axis.type = 0 /* e_unknown */;
axis.index = -1;
axis.separation = -b2_maxFloat;
var perp = b2EPCollider.s_perp.Set(-this.m_normal.y, this.m_normal.x);
for (var i = 0; i < this.m_polygonB.count; ++i) {
var n = b2Vec2.NegV(this.m_polygonB.normals[i], b2EPCollider.s_n);
var s1 = b2Vec2.DotVV(n, b2Vec2.SubVV(this.m_polygonB.vertices[i], this.m_v1, b2Vec2.s_t0));
var s2 = b2Vec2.DotVV(n, b2Vec2.SubVV(this.m_polygonB.vertices[i], this.m_v2, b2Vec2.s_t0));
var s = b2Min(s1, s2);
if (s > this.m_radius) {
// No collision
axis.type = 2 /* e_edgeB */;
axis.index = i;
axis.separation = s;
return axis;
}
// Adjacency
if (b2Vec2.DotVV(n, perp) >= 0) {
if (b2Vec2.DotVV(b2Vec2.SubVV(n, this.m_upperLimit, b2Vec2.s_t0), this.m_normal) < -b2_angularSlop) {
continue;
}
}
else {
if (b2Vec2.DotVV(b2Vec2.SubVV(n, this.m_lowerLimit, b2Vec2.s_t0), this.m_normal) < -b2_angularSlop) {
continue;
}
}
if (s > axis.separation) {
axis.type = 2 /* e_edgeB */;
axis.index = i;
axis.separation = s;
}
}
return axis;
};
b2EPCollider.s_edge1 = new b2Vec2();
b2EPCollider.s_edge0 = new b2Vec2();
b2EPCollider.s_edge2 = new b2Vec2();
b2EPCollider.s_ie = b2ClipVertex.MakeArray(2);
b2EPCollider.s_rf = new b2ReferenceFace();
b2EPCollider.s_clipPoints1 = b2ClipVertex.MakeArray(2);
b2EPCollider.s_clipPoints2 = b2ClipVertex.MakeArray(2);
b2EPCollider.s_edgeAxis = new b2EPAxis();
b2EPCollider.s_polygonAxis = new b2EPAxis();
b2EPCollider.s_n = new b2Vec2();
b2EPCollider.s_perp = new b2Vec2();
return b2EPCollider;
}());
var b2CollideEdgeAndPolygon_s_collider = new b2EPCollider();
function b2CollideEdgeAndPolygon(manifold, edgeA, xfA, polygonB, xfB) {
var collider = b2CollideEdgeAndPolygon_s_collider;
collider.Collide(manifold, edgeA, xfA, polygonB, xfB);
}
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// This holds the mass data computed for a shape.
var b2MassData = /** @class */ (function () {
function b2MassData() {
/// The mass of the shape, usually in kilograms.
this.mass = 0;
/// The position of the shape's centroid relative to the shape's origin.
this.center = new b2Vec2(0, 0);
/// The rotational inertia of the shape about the local origin.
this.I = 0;
}
return b2MassData;
}());
(function (b2ShapeType) {
b2ShapeType[b2ShapeType["e_unknown"] = -1] = "e_unknown";
b2ShapeType[b2ShapeType["e_circleShape"] = 0] = "e_circleShape";
b2ShapeType[b2ShapeType["e_edgeShape"] = 1] = "e_edgeShape";
b2ShapeType[b2ShapeType["e_polygonShape"] = 2] = "e_polygonShape";
b2ShapeType[b2ShapeType["e_chainShape"] = 3] = "e_chainShape";
b2ShapeType[b2ShapeType["e_shapeTypeCount"] = 4] = "e_shapeTypeCount";
})(exports.b2ShapeType || (exports.b2ShapeType = {}));
/// A shape is used for collision detection. You can create a shape however you like.
/// Shapes used for simulation in b2World are created automatically when a b2Fixture
/// is created. Shapes may encapsulate a one or more child shapes.
var b2Shape = /** @class */ (function () {
function b2Shape(type, radius) {
this.m_type = exports.b2ShapeType.e_unknown;
/// Radius of a shape. For polygonal shapes this must be b2_polygonRadius. There is no support for
/// making rounded polygons.
this.m_radius = 0;
this.m_type = type;
this.m_radius = radius;
}
b2Shape.prototype.Copy = function (other) {
// DEBUG: b2Assert(this.m_type === other.m_type);
this.m_radius = other.m_radius;
return this;
};
/// Get the type of this shape. You can use this to down cast to the concrete shape.
/// @return the shape type.
b2Shape.prototype.GetType = function () {
return this.m_type;
};
return b2Shape;
}());
/*! *****************************************************************************
Copyright (c) Microsoft Corporation. All rights reserved.
Licensed under the Apache License, Version 2.0 (the "License"); you may not use
this file except in compliance with the License. You may obtain a copy of the
License at http://www.apache.org/licenses/LICENSE-2.0
THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED
WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE,
MERCHANTABLITY OR NON-INFRINGEMENT.
See the Apache Version 2.0 License for specific language governing permissions
and limitations under the License.
***************************************************************************** */
/* global Reflect, Promise */
var extendStatics = Object.setPrototypeOf ||
({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
function __extends(d, b) {
extendStatics(d, b);
function __() { this.constructor = d; }
d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
}
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// A circle shape.
var b2CircleShape = /** @class */ (function (_super) {
__extends(b2CircleShape, _super);
function b2CircleShape(radius) {
if (radius === void 0) { radius = 0; }
var _this = _super.call(this, exports.b2ShapeType.e_circleShape, radius) || this;
_this.m_p = new b2Vec2();
return _this;
}
b2CircleShape.prototype.Set = function (position, radius) {
if (radius === void 0) { radius = this.m_radius; }
this.m_p.Copy(position);
this.m_radius = radius;
return this;
};
/// Implement b2Shape.
b2CircleShape.prototype.Clone = function () {
return new b2CircleShape().Copy(this);
};
b2CircleShape.prototype.Copy = function (other) {
_super.prototype.Copy.call(this, other);
// DEBUG: b2Assert(other instanceof b2CircleShape);
this.m_p.Copy(other.m_p);
return this;
};
/// @see b2Shape::GetChildCount
b2CircleShape.prototype.GetChildCount = function () {
return 1;
};
b2CircleShape.prototype.TestPoint = function (transform, p) {
var center = b2Transform.MulXV(transform, this.m_p, b2CircleShape.TestPoint_s_center);
var d = b2Vec2.SubVV(p, center, b2CircleShape.TestPoint_s_d);
return b2Vec2.DotVV(d, d) <= b2Sq(this.m_radius);
};
b2CircleShape.prototype.ComputeDistance = function (xf, p, normal, childIndex) {
var center = b2Transform.MulXV(xf, this.m_p, b2CircleShape.ComputeDistance_s_center);
b2Vec2.SubVV(p, center, normal);
return normal.Normalize() - this.m_radius;
};
b2CircleShape.prototype.RayCast = function (output, input, transform, childIndex) {
var position = b2Transform.MulXV(transform, this.m_p, b2CircleShape.RayCast_s_position);
var s = b2Vec2.SubVV(input.p1, position, b2CircleShape.RayCast_s_s);
var b = b2Vec2.DotVV(s, s) - b2Sq(this.m_radius);
// Solve quadratic equation.
var r = b2Vec2.SubVV(input.p2, input.p1, b2CircleShape.RayCast_s_r);
var c = b2Vec2.DotVV(s, r);
var rr = b2Vec2.DotVV(r, r);
var sigma = c * c - rr * b;
// Check for negative discriminant and short segment.
if (sigma < 0 || rr < b2_epsilon) {
return false;
}
// Find the point of intersection of the line with the circle.
var a = (-(c + b2Sqrt(sigma)));
// Is the intersection point on the segment?
if (0 <= a && a <= input.maxFraction * rr) {
a /= rr;
output.fraction = a;
b2Vec2.AddVMulSV(s, a, r, output.normal).SelfNormalize();
return true;
}
return false;
};
b2CircleShape.prototype.ComputeAABB = function (aabb, transform, childIndex) {
var p = b2Transform.MulXV(transform, this.m_p, b2CircleShape.ComputeAABB_s_p);
aabb.lowerBound.Set(p.x - this.m_radius, p.y - this.m_radius);
aabb.upperBound.Set(p.x + this.m_radius, p.y + this.m_radius);
};
/// @see b2Shape::ComputeMass
b2CircleShape.prototype.ComputeMass = function (massData, density) {
var radius_sq = b2Sq(this.m_radius);
massData.mass = density * b2_pi * radius_sq;
massData.center.Copy(this.m_p);
// inertia about the local origin
massData.I = massData.mass * (0.5 * radius_sq + b2Vec2.DotVV(this.m_p, this.m_p));
};
b2CircleShape.prototype.SetupDistanceProxy = function (proxy, index) {
proxy.m_vertices = proxy.m_buffer;
proxy.m_vertices[0].Copy(this.m_p);
proxy.m_count = 1;
proxy.m_radius = this.m_radius;
};
b2CircleShape.prototype.ComputeSubmergedArea = function (normal, offset, xf, c) {
var p = b2Transform.MulXV(xf, this.m_p, new b2Vec2());
var l = (-(b2Vec2.DotVV(normal, p) - offset));
if (l < (-this.m_radius) + b2_epsilon) {
// Completely dry
return 0;
}
if (l > this.m_radius) {
// Completely wet
c.Copy(p);
return b2_pi * this.m_radius * this.m_radius;
}
// Magic
var r2 = this.m_radius * this.m_radius;
var l2 = l * l;
var area = r2 * (b2Asin(l / this.m_radius) + b2_pi / 2) + l * b2Sqrt(r2 - l2);
var com = (-2 / 3 * b2Pow(r2 - l2, 1.5) / area);
c.x = p.x + normal.x * com;
c.y = p.y + normal.y * com;
return area;
};
b2CircleShape.prototype.Dump = function (log) {
log(" const shape: b2CircleShape = new b2CircleShape();\n");
log(" shape.m_radius = %.15f;\n", this.m_radius);
log(" shape.m_p.Set(%.15f, %.15f);\n", this.m_p.x, this.m_p.y);
};
/// Implement b2Shape.
b2CircleShape.TestPoint_s_center = new b2Vec2();
b2CircleShape.TestPoint_s_d = new b2Vec2();
// #if B2_ENABLE_PARTICLE
/// @see b2Shape::ComputeDistance
b2CircleShape.ComputeDistance_s_center = new b2Vec2();
// #endif
/// Implement b2Shape.
// Collision Detection in Interactive 3D Environments by Gino van den Bergen
// From Section 3.1.2
// x = s + a * r
// norm(x) = radius
b2CircleShape.RayCast_s_position = new b2Vec2();
b2CircleShape.RayCast_s_s = new b2Vec2();
b2CircleShape.RayCast_s_r = new b2Vec2();
/// @see b2Shape::ComputeAABB
b2CircleShape.ComputeAABB_s_p = new b2Vec2();
return b2CircleShape;
}(b2Shape));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// A convex polygon. It is assumed that the interior of the polygon is to
/// the left of each edge.
/// Polygons have a maximum number of vertices equal to b2_maxPolygonVertices.
/// In most cases you should not need many vertices for a convex polygon.
var b2PolygonShape = /** @class */ (function (_super) {
__extends(b2PolygonShape, _super);
function b2PolygonShape() {
var _this = _super.call(this, exports.b2ShapeType.e_polygonShape, b2_polygonRadius) || this;
_this.m_centroid = new b2Vec2(0, 0);
_this.m_vertices = []; // b2Vec2.MakeArray(b2_maxPolygonVertices);
_this.m_normals = []; // b2Vec2.MakeArray(b2_maxPolygonVertices);
_this.m_count = 0;
return _this;
}
/// Implement b2Shape.
b2PolygonShape.prototype.Clone = function () {
return new b2PolygonShape().Copy(this);
};
b2PolygonShape.prototype.Copy = function (other) {
_super.prototype.Copy.call(this, other);
// DEBUG: b2Assert(other instanceof b2PolygonShape);
this.m_centroid.Copy(other.m_centroid);
this.m_count = other.m_count;
this.m_vertices = b2Vec2.MakeArray(this.m_count);
this.m_normals = b2Vec2.MakeArray(this.m_count);
for (var i = 0; i < this.m_count; ++i) {
this.m_vertices[i].Copy(other.m_vertices[i]);
this.m_normals[i].Copy(other.m_normals[i]);
}
return this;
};
/// @see b2Shape::GetChildCount
b2PolygonShape.prototype.GetChildCount = function () {
return 1;
};
b2PolygonShape.prototype.Set = function (vertices, count, start) {
if (count === void 0) { count = vertices.length; }
if (start === void 0) { start = 0; }
// DEBUG: b2Assert(3 <= count && count <= b2_maxPolygonVertices);
if (count < 3) {
return this.SetAsBox(1, 1);
}
var n = b2Min(count, b2_maxPolygonVertices);
// Perform welding and copy vertices into local buffer.
var ps = b2PolygonShape.Set_s_ps;
var tempCount = 0;
for (var i = 0; i < n; ++i) {
var /*b2Vec2*/ v = vertices[start + i];
var /*bool*/ unique = true;
for (var /*int32*/ j = 0; j < tempCount; ++j) {
if (b2Vec2.DistanceSquaredVV(v, ps[j]) < ((0.5 * b2_linearSlop) * (0.5 * b2_linearSlop))) {
unique = false;
break;
}
}
if (unique) {
ps[tempCount++].Copy(v); // ps[tempCount++] = v;
}
}
n = tempCount;
if (n < 3) {
// Polygon is degenerate.
// DEBUG: b2Assert(false);
return this.SetAsBox(1.0, 1.0);
}
// Create the convex hull using the Gift wrapping algorithm
// http://en.wikipedia.org/wiki/Gift_wrapping_algorithm
// Find the right most point on the hull
var i0 = 0;
var x0 = ps[0].x;
for (var i = 1; i < n; ++i) {
var x = ps[i].x;
if (x > x0 || (x === x0 && ps[i].y < ps[i0].y)) {
i0 = i;
x0 = x;
}
}
var hull = b2PolygonShape.Set_s_hull;
var m = 0;
var ih = i0;
for (;;) {
// DEBUG: b2Assert(m < b2_maxPolygonVertices);
hull[m] = ih;
var ie = 0;
for (var j = 1; j < n; ++j) {
if (ie === ih) {
ie = j;
continue;
}
var r = b2Vec2.SubVV(ps[ie], ps[hull[m]], b2PolygonShape.Set_s_r);
var v = b2Vec2.SubVV(ps[j], ps[hull[m]], b2PolygonShape.Set_s_v);
var c = b2Vec2.CrossVV(r, v);
if (c < 0) {
ie = j;
}
// Collinearity check
if (c === 0 && v.LengthSquared() > r.LengthSquared()) {
ie = j;
}
}
++m;
ih = ie;
if (ie === i0) {
break;
}
}
this.m_count = m;
this.m_vertices = b2Vec2.MakeArray(this.m_count);
this.m_normals = b2Vec2.MakeArray(this.m_count);
// Copy vertices.
for (var i = 0; i < m; ++i) {
this.m_vertices[i].Copy(ps[hull[i]]);
}
// Compute normals. Ensure the edges have non-zero length.
for (var i = 0; i < m; ++i) {
var vertexi1 = this.m_vertices[i];
var vertexi2 = this.m_vertices[(i + 1) % m];
var edge = b2Vec2.SubVV(vertexi2, vertexi1, b2Vec2.s_t0); // edge uses s_t0
// DEBUG: b2Assert(edge.LengthSquared() > b2_epsilon_sq);
b2Vec2.CrossVOne(edge, this.m_normals[i]).SelfNormalize();
}
// Compute the polygon centroid.
b2PolygonShape.ComputeCentroid(this.m_vertices, m, this.m_centroid);
return this;
};
b2PolygonShape.prototype.SetAsArray = function (vertices, count) {
if (count === void 0) { count = vertices.length; }
return this.Set(vertices, count);
};
/// Build vertices to represent an axis-aligned box or an oriented box.
/// @param hx the half-width.
/// @param hy the half-height.
/// @param center the center of the box in local coordinates.
/// @param angle the rotation of the box in local coordinates.
b2PolygonShape.prototype.SetAsBox = function (hx, hy, center, angle) {
if (angle === void 0) { angle = 0; }
this.m_count = 4;
this.m_vertices = b2Vec2.MakeArray(this.m_count);
this.m_normals = b2Vec2.MakeArray(this.m_count);
this.m_vertices[0].Set((-hx), (-hy));
this.m_vertices[1].Set(hx, (-hy));
this.m_vertices[2].Set(hx, hy);
this.m_vertices[3].Set((-hx), hy);
this.m_normals[0].Set(0, (-1));
this.m_normals[1].Set(1, 0);
this.m_normals[2].Set(0, 1);
this.m_normals[3].Set((-1), 0);
this.m_centroid.SetZero();
if (center) {
this.m_centroid.Copy(center);
var xf = new b2Transform();
xf.SetPosition(center);
xf.SetRotationAngle(angle);
// Transform vertices and normals.
for (var i = 0; i < this.m_count; ++i) {
b2Transform.MulXV(xf, this.m_vertices[i], this.m_vertices[i]);
b2Rot.MulRV(xf.q, this.m_normals[i], this.m_normals[i]);
}
}
return this;
};
b2PolygonShape.prototype.TestPoint = function (xf, p) {
var pLocal = b2Transform.MulTXV(xf, p, b2PolygonShape.TestPoint_s_pLocal);
for (var i = 0; i < this.m_count; ++i) {
var dot = b2Vec2.DotVV(this.m_normals[i], b2Vec2.SubVV(pLocal, this.m_vertices[i], b2Vec2.s_t0));
if (dot > 0) {
return false;
}
}
return true;
};
b2PolygonShape.prototype.ComputeDistance = function (xf, p, normal, childIndex) {
var pLocal = b2Transform.MulTXV(xf, p, b2PolygonShape.ComputeDistance_s_pLocal);
var maxDistance = -b2_maxFloat;
var normalForMaxDistance = b2PolygonShape.ComputeDistance_s_normalForMaxDistance.Copy(pLocal);
for (var i = 0; i < this.m_count; ++i) {
var dot = b2Vec2.DotVV(this.m_normals[i], b2Vec2.SubVV(pLocal, this.m_vertices[i], b2Vec2.s_t0));
if (dot > maxDistance) {
maxDistance = dot;
normalForMaxDistance.Copy(this.m_normals[i]);
}
}
if (maxDistance > 0) {
var minDistance = b2PolygonShape.ComputeDistance_s_minDistance.Copy(normalForMaxDistance);
var minDistance2 = maxDistance * maxDistance;
for (var i = 0; i < this.m_count; ++i) {
var distance = b2Vec2.SubVV(pLocal, this.m_vertices[i], b2PolygonShape.ComputeDistance_s_distance);
var distance2 = distance.LengthSquared();
if (minDistance2 > distance2) {
minDistance.Copy(distance);
minDistance2 = distance2;
}
}
b2Rot.MulRV(xf.q, minDistance, normal);
normal.Normalize();
return Math.sqrt(minDistance2);
}
else {
b2Rot.MulRV(xf.q, normalForMaxDistance, normal);
return maxDistance;
}
};
b2PolygonShape.prototype.RayCast = function (output, input, xf, childIndex) {
// Put the ray into the polygon's frame of reference.
var p1 = b2Transform.MulTXV(xf, input.p1, b2PolygonShape.RayCast_s_p1);
var p2 = b2Transform.MulTXV(xf, input.p2, b2PolygonShape.RayCast_s_p2);
var d = b2Vec2.SubVV(p2, p1, b2PolygonShape.RayCast_s_d);
var lower = 0, upper = input.maxFraction;
var index = -1;
for (var i = 0; i < this.m_count; ++i) {
// p = p1 + a * d
// dot(normal, p - v) = 0
// dot(normal, p1 - v) + a * dot(normal, d) = 0
var numerator = b2Vec2.DotVV(this.m_normals[i], b2Vec2.SubVV(this.m_vertices[i], p1, b2Vec2.s_t0));
var denominator = b2Vec2.DotVV(this.m_normals[i], d);
if (denominator === 0) {
if (numerator < 0) {
return false;
}
}
else {
// Note: we want this predicate without division:
// lower < numerator / denominator, where denominator < 0
// Since denominator < 0, we have to flip the inequality:
// lower < numerator / denominator <==> denominator * lower > numerator.
if (denominator < 0 && numerator < lower * denominator) {
// Increase lower.
// The segment enters this half-space.
lower = numerator / denominator;
index = i;
}
else if (denominator > 0 && numerator < upper * denominator) {
// Decrease upper.
// The segment exits this half-space.
upper = numerator / denominator;
}
}
// The use of epsilon here causes the assert on lower to trip
// in some cases. Apparently the use of epsilon was to make edge
// shapes work, but now those are handled separately.
// if (upper < lower - b2_epsilon)
if (upper < lower) {
return false;
}
}
// DEBUG: b2Assert(0 <= lower && lower <= input.maxFraction);
if (index >= 0) {
output.fraction = lower;
b2Rot.MulRV(xf.q, this.m_normals[index], output.normal);
return true;
}
return false;
};
b2PolygonShape.prototype.ComputeAABB = function (aabb, xf, childIndex) {
var lower = b2Transform.MulXV(xf, this.m_vertices[0], aabb.lowerBound);
var upper = aabb.upperBound.Copy(lower);
for (var i = 0; i < this.m_count; ++i) {
var v = b2Transform.MulXV(xf, this.m_vertices[i], b2PolygonShape.ComputeAABB_s_v);
b2Vec2.MinV(v, lower, lower);
b2Vec2.MaxV(v, upper, upper);
}
var r = this.m_radius;
lower.SelfSubXY(r, r);
upper.SelfAddXY(r, r);
};
b2PolygonShape.prototype.ComputeMass = function (massData, density) {
// Polygon mass, centroid, and inertia.
// Let rho be the polygon density in mass per unit area.
// Then:
// mass = rho * int(dA)
// centroid.x = (1/mass) * rho * int(x * dA)
// centroid.y = (1/mass) * rho * int(y * dA)
// I = rho * int((x*x + y*y) * dA)
//
// We can compute these integrals by summing all the integrals
// for each triangle of the polygon. To evaluate the integral
// for a single triangle, we make a change of variables to
// the (u,v) coordinates of the triangle:
// x = x0 + e1x * u + e2x * v
// y = y0 + e1y * u + e2y * v
// where 0 <= u && 0 <= v && u + v <= 1.
//
// We integrate u from [0,1-v] and then v from [0,1].
// We also need to use the Jacobian of the transformation:
// D = cross(e1, e2)
//
// Simplification: triangle centroid = (1/3) * (p1 + p2 + p3)
//
// The rest of the derivation is handled by computer algebra.
// DEBUG: b2Assert(this.m_count >= 3);
var center = b2PolygonShape.ComputeMass_s_center.SetZero();
var area = 0;
var I = 0;
// s is the reference point for forming triangles.
// It's location doesn't change the result (except for rounding error).
var s = b2PolygonShape.ComputeMass_s_s.SetZero();
// This code would put the reference point inside the polygon.
for (var i = 0; i < this.m_count; ++i) {
s.SelfAdd(this.m_vertices[i]);
}
s.SelfMul(1 / this.m_count);
var k_inv3 = 1 / 3;
for (var i = 0; i < this.m_count; ++i) {
// Triangle vertices.
var e1 = b2Vec2.SubVV(this.m_vertices[i], s, b2PolygonShape.ComputeMass_s_e1);
var e2 = b2Vec2.SubVV(this.m_vertices[(i + 1) % this.m_count], s, b2PolygonShape.ComputeMass_s_e2);
var D = b2Vec2.CrossVV(e1, e2);
var triangleArea = 0.5 * D;
area += triangleArea;
// Area weighted centroid
center.SelfAdd(b2Vec2.MulSV(triangleArea * k_inv3, b2Vec2.AddVV(e1, e2, b2Vec2.s_t0), b2Vec2.s_t1));
var ex1 = e1.x;
var ey1 = e1.y;
var ex2 = e2.x;
var ey2 = e2.y;
var intx2 = ex1 * ex1 + ex2 * ex1 + ex2 * ex2;
var inty2 = ey1 * ey1 + ey2 * ey1 + ey2 * ey2;
I += (0.25 * k_inv3 * D) * (intx2 + inty2);
}
// Total mass
massData.mass = density * area;
// Center of mass
// DEBUG: b2Assert(area > b2_epsilon);
center.SelfMul(1 / area);
b2Vec2.AddVV(center, s, massData.center);
// Inertia tensor relative to the local origin (point s).
massData.I = density * I;
// Shift to center of mass then to original body origin.
massData.I += massData.mass * (b2Vec2.DotVV(massData.center, massData.center) - b2Vec2.DotVV(center, center));
};
b2PolygonShape.prototype.Validate = function () {
for (var i = 0; i < this.m_count; ++i) {
var i1 = i;
var i2 = (i + 1) % this.m_count;
var p = this.m_vertices[i1];
var e = b2Vec2.SubVV(this.m_vertices[i2], p, b2PolygonShape.Validate_s_e);
for (var j = 0; j < this.m_count; ++j) {
if (j === i1 || j === i2) {
continue;
}
var v = b2Vec2.SubVV(this.m_vertices[j], p, b2PolygonShape.Validate_s_v);
var c = b2Vec2.CrossVV(e, v);
if (c < 0) {
return false;
}
}
}
return true;
};
b2PolygonShape.prototype.SetupDistanceProxy = function (proxy, index) {
proxy.m_vertices = this.m_vertices;
proxy.m_count = this.m_count;
proxy.m_radius = this.m_radius;
};
b2PolygonShape.prototype.ComputeSubmergedArea = function (normal, offset, xf, c) {
// Transform plane into shape co-ordinates
var normalL = b2Rot.MulTRV(xf.q, normal, b2PolygonShape.ComputeSubmergedArea_s_normalL);
var offsetL = offset - b2Vec2.DotVV(normal, xf.p);
var depths = b2PolygonShape.ComputeSubmergedArea_s_depths;
var diveCount = 0;
var intoIndex = -1;
var outoIndex = -1;
var lastSubmerged = false;
for (var i_1 = 0; i_1 < this.m_count; ++i_1) {
depths[i_1] = b2Vec2.DotVV(normalL, this.m_vertices[i_1]) - offsetL;
var isSubmerged = depths[i_1] < (-b2_epsilon);
if (i_1 > 0) {
if (isSubmerged) {
if (!lastSubmerged) {
intoIndex = i_1 - 1;
diveCount++;
}
}
else {
if (lastSubmerged) {
outoIndex = i_1 - 1;
diveCount++;
}
}
}
lastSubmerged = isSubmerged;
}
switch (diveCount) {
case 0:
if (lastSubmerged) {
// Completely submerged
var md = b2PolygonShape.ComputeSubmergedArea_s_md;
this.ComputeMass(md, 1);
b2Transform.MulXV(xf, md.center, c);
return md.mass;
}
else {
// Completely dry
return 0;
}
case 1:
if (intoIndex === (-1)) {
intoIndex = this.m_count - 1;
}
else {
outoIndex = this.m_count - 1;
}
break;
}
var intoIndex2 = ((intoIndex + 1) % this.m_count);
var outoIndex2 = ((outoIndex + 1) % this.m_count);
var intoLamdda = (0 - depths[intoIndex]) / (depths[intoIndex2] - depths[intoIndex]);
var outoLamdda = (0 - depths[outoIndex]) / (depths[outoIndex2] - depths[outoIndex]);
var intoVec = b2PolygonShape.ComputeSubmergedArea_s_intoVec.Set(this.m_vertices[intoIndex].x * (1 - intoLamdda) + this.m_vertices[intoIndex2].x * intoLamdda, this.m_vertices[intoIndex].y * (1 - intoLamdda) + this.m_vertices[intoIndex2].y * intoLamdda);
var outoVec = b2PolygonShape.ComputeSubmergedArea_s_outoVec.Set(this.m_vertices[outoIndex].x * (1 - outoLamdda) + this.m_vertices[outoIndex2].x * outoLamdda, this.m_vertices[outoIndex].y * (1 - outoLamdda) + this.m_vertices[outoIndex2].y * outoLamdda);
// Initialize accumulator
var area = 0;
var center = b2PolygonShape.ComputeSubmergedArea_s_center.SetZero();
var p2 = this.m_vertices[intoIndex2];
var p3;
// An awkward loop from intoIndex2+1 to outIndex2
var i = intoIndex2;
while (i !== outoIndex2) {
i = (i + 1) % this.m_count;
if (i === outoIndex2) {
p3 = outoVec;
}
else {
p3 = this.m_vertices[i];
}
var triangleArea = 0.5 * ((p2.x - intoVec.x) * (p3.y - intoVec.y) - (p2.y - intoVec.y) * (p3.x - intoVec.x));
area += triangleArea;
// Area weighted centroid
center.x += triangleArea * (intoVec.x + p2.x + p3.x) / 3;
center.y += triangleArea * (intoVec.y + p2.y + p3.y) / 3;
p2 = p3;
}
// Normalize and transform centroid
center.SelfMul(1 / area);
b2Transform.MulXV(xf, center, c);
return area;
};
b2PolygonShape.prototype.Dump = function (log) {
log(" const shape: b2PolygonShape = new b2PolygonShape();\n");
log(" const vs: b2Vec2[] = b2Vec2.MakeArray(%d);\n", b2_maxPolygonVertices);
for (var i = 0; i < this.m_count; ++i) {
log(" vs[%d].Set(%.15f, %.15f);\n", i, this.m_vertices[i].x, this.m_vertices[i].y);
}
log(" shape.Set(vs, %d);\n", this.m_count);
};
b2PolygonShape.ComputeCentroid = function (vs, count, out) {
// DEBUG: b2Assert(count >= 3);
var c = out;
c.SetZero();
var area = 0;
// s is the reference point for forming triangles.
// It's location doesn't change the result (except for rounding error).
var pRef = b2PolygonShape.ComputeCentroid_s_pRef.SetZero();
/*
#if 0
// This code would put the reference point inside the polygon.
for (let i: number = 0; i < count; ++i) {
pRef.SelfAdd(vs[i]);
}
pRef.SelfMul(1 / count);
#endif
*/
var inv3 = 1 / 3;
for (var i = 0; i < count; ++i) {
// Triangle vertices.
var p1 = pRef;
var p2 = vs[i];
var p3 = vs[(i + 1) % count];
var e1 = b2Vec2.SubVV(p2, p1, b2PolygonShape.ComputeCentroid_s_e1);
var e2 = b2Vec2.SubVV(p3, p1, b2PolygonShape.ComputeCentroid_s_e2);
var D = b2Vec2.CrossVV(e1, e2);
var triangleArea = 0.5 * D;
area += triangleArea;
// Area weighted centroid
c.x += triangleArea * inv3 * (p1.x + p2.x + p3.x);
c.y += triangleArea * inv3 * (p1.y + p2.y + p3.y);
}
// Centroid
// DEBUG: b2Assert(area > b2_epsilon);
c.SelfMul(1 / area);
return c;
};
/// Create a convex hull from the given array of points.
/// The count must be in the range [3, b2_maxPolygonVertices].
/// @warning the points may be re-ordered, even if they form a convex polygon
/// @warning collinear points are handled but not removed. Collinear points
/// may lead to poor stacking behavior.
b2PolygonShape.Set_s_ps = b2Vec2.MakeArray(b2_maxPolygonVertices);
b2PolygonShape.Set_s_hull = b2MakeNumberArray(b2_maxPolygonVertices);
b2PolygonShape.Set_s_r = new b2Vec2();
b2PolygonShape.Set_s_v = new b2Vec2();
/// @see b2Shape::TestPoint
b2PolygonShape.TestPoint_s_pLocal = new b2Vec2();
// #if B2_ENABLE_PARTICLE
/// @see b2Shape::ComputeDistance
b2PolygonShape.ComputeDistance_s_pLocal = new b2Vec2();
b2PolygonShape.ComputeDistance_s_normalForMaxDistance = new b2Vec2();
b2PolygonShape.ComputeDistance_s_minDistance = new b2Vec2();
b2PolygonShape.ComputeDistance_s_distance = new b2Vec2();
// #endif
/// Implement b2Shape.
b2PolygonShape.RayCast_s_p1 = new b2Vec2();
b2PolygonShape.RayCast_s_p2 = new b2Vec2();
b2PolygonShape.RayCast_s_d = new b2Vec2();
/// @see b2Shape::ComputeAABB
b2PolygonShape.ComputeAABB_s_v = new b2Vec2();
/// @see b2Shape::ComputeMass
b2PolygonShape.ComputeMass_s_center = new b2Vec2();
b2PolygonShape.ComputeMass_s_s = new b2Vec2();
b2PolygonShape.ComputeMass_s_e1 = new b2Vec2();
b2PolygonShape.ComputeMass_s_e2 = new b2Vec2();
b2PolygonShape.Validate_s_e = new b2Vec2();
b2PolygonShape.Validate_s_v = new b2Vec2();
b2PolygonShape.ComputeSubmergedArea_s_normalL = new b2Vec2();
b2PolygonShape.ComputeSubmergedArea_s_depths = b2MakeNumberArray(b2_maxPolygonVertices);
b2PolygonShape.ComputeSubmergedArea_s_md = new b2MassData();
b2PolygonShape.ComputeSubmergedArea_s_intoVec = new b2Vec2();
b2PolygonShape.ComputeSubmergedArea_s_outoVec = new b2Vec2();
b2PolygonShape.ComputeSubmergedArea_s_center = new b2Vec2();
b2PolygonShape.ComputeCentroid_s_pRef = new b2Vec2();
b2PolygonShape.ComputeCentroid_s_e1 = new b2Vec2();
b2PolygonShape.ComputeCentroid_s_e2 = new b2Vec2();
return b2PolygonShape;
}(b2Shape));
/*
* Copyright (c) 2006-2010 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// A line segment (edge) shape. These can be connected in chains or loops
/// to other edge shapes. The connectivity information is used to ensure
/// correct contact normals.
var b2EdgeShape = /** @class */ (function (_super) {
__extends(b2EdgeShape, _super);
function b2EdgeShape() {
var _this = _super.call(this, exports.b2ShapeType.e_edgeShape, b2_polygonRadius) || this;
_this.m_vertex1 = new b2Vec2();
_this.m_vertex2 = new b2Vec2();
_this.m_vertex0 = new b2Vec2();
_this.m_vertex3 = new b2Vec2();
_this.m_hasVertex0 = false;
_this.m_hasVertex3 = false;
return _this;
}
/// Set this as an isolated edge.
b2EdgeShape.prototype.Set = function (v1, v2) {
this.m_vertex1.Copy(v1);
this.m_vertex2.Copy(v2);
this.m_hasVertex0 = false;
this.m_hasVertex3 = false;
return this;
};
/// Implement b2Shape.
b2EdgeShape.prototype.Clone = function () {
return new b2EdgeShape().Copy(this);
};
b2EdgeShape.prototype.Copy = function (other) {
_super.prototype.Copy.call(this, other);
// DEBUG: b2Assert(other instanceof b2EdgeShape);
this.m_vertex1.Copy(other.m_vertex1);
this.m_vertex2.Copy(other.m_vertex2);
this.m_vertex0.Copy(other.m_vertex0);
this.m_vertex3.Copy(other.m_vertex3);
this.m_hasVertex0 = other.m_hasVertex0;
this.m_hasVertex3 = other.m_hasVertex3;
return this;
};
/// @see b2Shape::GetChildCount
b2EdgeShape.prototype.GetChildCount = function () {
return 1;
};
/// @see b2Shape::TestPoint
b2EdgeShape.prototype.TestPoint = function (xf, p) {
return false;
};
b2EdgeShape.prototype.ComputeDistance = function (xf, p, normal, childIndex) {
var v1 = b2Transform.MulXV(xf, this.m_vertex1, b2EdgeShape.ComputeDistance_s_v1);
var v2 = b2Transform.MulXV(xf, this.m_vertex2, b2EdgeShape.ComputeDistance_s_v2);
var d = b2Vec2.SubVV(p, v1, b2EdgeShape.ComputeDistance_s_d);
var s = b2Vec2.SubVV(v2, v1, b2EdgeShape.ComputeDistance_s_s);
var ds = b2Vec2.DotVV(d, s);
if (ds > 0) {
var s2 = b2Vec2.DotVV(s, s);
if (ds > s2) {
b2Vec2.SubVV(p, v2, d);
}
else {
d.SelfMulSub(ds / s2, s);
}
}
normal.Copy(d);
return normal.Normalize();
};
b2EdgeShape.prototype.RayCast = function (output, input, xf, childIndex) {
// Put the ray into the edge's frame of reference.
var p1 = b2Transform.MulTXV(xf, input.p1, b2EdgeShape.RayCast_s_p1);
var p2 = b2Transform.MulTXV(xf, input.p2, b2EdgeShape.RayCast_s_p2);
var d = b2Vec2.SubVV(p2, p1, b2EdgeShape.RayCast_s_d);
var v1 = this.m_vertex1;
var v2 = this.m_vertex2;
var e = b2Vec2.SubVV(v2, v1, b2EdgeShape.RayCast_s_e);
var normal = output.normal.Set(e.y, -e.x).SelfNormalize();
// q = p1 + t * d
// dot(normal, q - v1) = 0
// dot(normal, p1 - v1) + t * dot(normal, d) = 0
var numerator = b2Vec2.DotVV(normal, b2Vec2.SubVV(v1, p1, b2Vec2.s_t0));
var denominator = b2Vec2.DotVV(normal, d);
if (denominator === 0) {
return false;
}
var t = numerator / denominator;
if (t < 0 || input.maxFraction < t) {
return false;
}
var q = b2Vec2.AddVMulSV(p1, t, d, b2EdgeShape.RayCast_s_q);
// q = v1 + s * r
// s = dot(q - v1, r) / dot(r, r)
var r = b2Vec2.SubVV(v2, v1, b2EdgeShape.RayCast_s_r);
var rr = b2Vec2.DotVV(r, r);
if (rr === 0) {
return false;
}
var s = b2Vec2.DotVV(b2Vec2.SubVV(q, v1, b2Vec2.s_t0), r) / rr;
if (s < 0 || 1 < s) {
return false;
}
output.fraction = t;
b2Rot.MulRV(xf.q, output.normal, output.normal);
if (numerator > 0) {
output.normal.SelfNeg();
}
return true;
};
b2EdgeShape.prototype.ComputeAABB = function (aabb, xf, childIndex) {
var v1 = b2Transform.MulXV(xf, this.m_vertex1, b2EdgeShape.ComputeAABB_s_v1);
var v2 = b2Transform.MulXV(xf, this.m_vertex2, b2EdgeShape.ComputeAABB_s_v2);
b2Vec2.MinV(v1, v2, aabb.lowerBound);
b2Vec2.MaxV(v1, v2, aabb.upperBound);
var r = this.m_radius;
aabb.lowerBound.SelfSubXY(r, r);
aabb.upperBound.SelfAddXY(r, r);
};
/// @see b2Shape::ComputeMass
b2EdgeShape.prototype.ComputeMass = function (massData, density) {
massData.mass = 0;
b2Vec2.MidVV(this.m_vertex1, this.m_vertex2, massData.center);
massData.I = 0;
};
b2EdgeShape.prototype.SetupDistanceProxy = function (proxy, index) {
proxy.m_vertices = proxy.m_buffer;
proxy.m_vertices[0].Copy(this.m_vertex1);
proxy.m_vertices[1].Copy(this.m_vertex2);
proxy.m_count = 2;
proxy.m_radius = this.m_radius;
};
b2EdgeShape.prototype.ComputeSubmergedArea = function (normal, offset, xf, c) {
c.SetZero();
return 0;
};
b2EdgeShape.prototype.Dump = function (log) {
log(" const shape: b2EdgeShape = new b2EdgeShape();\n");
log(" shape.m_radius = %.15f;\n", this.m_radius);
log(" shape.m_vertex0.Set(%.15f, %.15f);\n", this.m_vertex0.x, this.m_vertex0.y);
log(" shape.m_vertex1.Set(%.15f, %.15f);\n", this.m_vertex1.x, this.m_vertex1.y);
log(" shape.m_vertex2.Set(%.15f, %.15f);\n", this.m_vertex2.x, this.m_vertex2.y);
log(" shape.m_vertex3.Set(%.15f, %.15f);\n", this.m_vertex3.x, this.m_vertex3.y);
log(" shape.m_hasVertex0 = %s;\n", this.m_hasVertex0);
log(" shape.m_hasVertex3 = %s;\n", this.m_hasVertex3);
};
// #if B2_ENABLE_PARTICLE
/// @see b2Shape::ComputeDistance
b2EdgeShape.ComputeDistance_s_v1 = new b2Vec2();
b2EdgeShape.ComputeDistance_s_v2 = new b2Vec2();
b2EdgeShape.ComputeDistance_s_d = new b2Vec2();
b2EdgeShape.ComputeDistance_s_s = new b2Vec2();
// #endif
/// Implement b2Shape.
// p = p1 + t * d
// v = v1 + s * e
// p1 + t * d = v1 + s * e
// s * e - t * d = p1 - v1
b2EdgeShape.RayCast_s_p1 = new b2Vec2();
b2EdgeShape.RayCast_s_p2 = new b2Vec2();
b2EdgeShape.RayCast_s_d = new b2Vec2();
b2EdgeShape.RayCast_s_e = new b2Vec2();
b2EdgeShape.RayCast_s_q = new b2Vec2();
b2EdgeShape.RayCast_s_r = new b2Vec2();
/// @see b2Shape::ComputeAABB
b2EdgeShape.ComputeAABB_s_v1 = new b2Vec2();
b2EdgeShape.ComputeAABB_s_v2 = new b2Vec2();
return b2EdgeShape;
}(b2Shape));
/*
* Copyright (c) 2006-2010 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// A chain shape is a free form sequence of line segments.
/// The chain has two-sided collision, so you can use inside and outside collision.
/// Therefore, you may use any winding order.
/// Since there may be many vertices, they are allocated using b2Alloc.
/// Connectivity information is used to create smooth collisions.
/// WARNING: The chain will not collide properly if there are self-intersections.
var b2ChainShape = /** @class */ (function (_super) {
__extends(b2ChainShape, _super);
function b2ChainShape() {
var _this = _super.call(this, exports.b2ShapeType.e_chainShape, b2_polygonRadius) || this;
_this.m_vertices = [];
_this.m_count = 0;
_this.m_prevVertex = new b2Vec2();
_this.m_nextVertex = new b2Vec2();
_this.m_hasPrevVertex = false;
_this.m_hasNextVertex = false;
return _this;
}
/// Create a loop. This automatically adjusts connectivity.
/// @param vertices an array of vertices, these are copied
/// @param count the vertex count
b2ChainShape.prototype.CreateLoop = function (vertices, count, start) {
if (count === void 0) { count = vertices.length; }
if (start === void 0) { start = 0; }
// DEBUG: b2Assert(count >= 3);
if (count < 3) {
return this;
}
// DEBUG: for (let i: number = 1; i < count; ++i) {
// DEBUG: const v1 = vertices[start + i - 1];
// DEBUG: const v2 = vertices[start + i];
// DEBUG: // If the code crashes here, it means your vertices are too close together.
// DEBUG: b2Assert(b2Vec2.DistanceSquaredVV(v1, v2) > b2_linearSlop * b2_linearSlop);
// DEBUG: }
this.m_count = count + 1;
this.m_vertices = b2Vec2.MakeArray(this.m_count);
for (var i = 0; i < count; ++i) {
this.m_vertices[i].Copy(vertices[start + i]);
}
this.m_vertices[count].Copy(this.m_vertices[0]);
this.m_prevVertex.Copy(this.m_vertices[this.m_count - 2]);
this.m_nextVertex.Copy(this.m_vertices[1]);
this.m_hasPrevVertex = true;
this.m_hasNextVertex = true;
return this;
};
/// Create a chain with isolated end vertices.
/// @param vertices an array of vertices, these are copied
/// @param count the vertex count
b2ChainShape.prototype.CreateChain = function (vertices, count, start) {
// DEBUG: b2Assert(count >= 2);
// DEBUG: for (let i: number = 1; i < count; ++i) {
// DEBUG: const v1 = vertices[start + i - 1];
// DEBUG: const v2 = vertices[start + i];
// DEBUG: // If the code crashes here, it means your vertices are too close together.
// DEBUG: b2Assert(b2Vec2.DistanceSquaredVV(v1, v2) > b2_linearSlop * b2_linearSlop);
// DEBUG: }
if (count === void 0) { count = vertices.length; }
if (start === void 0) { start = 0; }
this.m_count = count;
this.m_vertices = b2Vec2.MakeArray(count);
for (var i = 0; i < count; ++i) {
this.m_vertices[i].Copy(vertices[start + i]);
}
this.m_hasPrevVertex = false;
this.m_hasNextVertex = false;
this.m_prevVertex.SetZero();
this.m_nextVertex.SetZero();
return this;
};
/// Establish connectivity to a vertex that precedes the first vertex.
/// Don't call this for loops.
b2ChainShape.prototype.SetPrevVertex = function (prevVertex) {
this.m_prevVertex.Copy(prevVertex);
this.m_hasPrevVertex = true;
return this;
};
/// Establish connectivity to a vertex that follows the last vertex.
/// Don't call this for loops.
b2ChainShape.prototype.SetNextVertex = function (nextVertex) {
this.m_nextVertex.Copy(nextVertex);
this.m_hasNextVertex = true;
return this;
};
/// Implement b2Shape. Vertices are cloned using b2Alloc.
b2ChainShape.prototype.Clone = function () {
return new b2ChainShape().Copy(this);
};
b2ChainShape.prototype.Copy = function (other) {
_super.prototype.Copy.call(this, other);
// DEBUG: b2Assert(other instanceof b2ChainShape);
this.CreateChain(other.m_vertices, other.m_count);
this.m_prevVertex.Copy(other.m_prevVertex);
this.m_nextVertex.Copy(other.m_nextVertex);
this.m_hasPrevVertex = other.m_hasPrevVertex;
this.m_hasNextVertex = other.m_hasNextVertex;
return this;
};
/// @see b2Shape::GetChildCount
b2ChainShape.prototype.GetChildCount = function () {
// edge count = vertex count - 1
return this.m_count - 1;
};
/// Get a child edge.
b2ChainShape.prototype.GetChildEdge = function (edge, index) {
// DEBUG: b2Assert(0 <= index && index < this.m_count - 1);
edge.m_type = exports.b2ShapeType.e_edgeShape;
edge.m_radius = this.m_radius;
edge.m_vertex1.Copy(this.m_vertices[index]);
edge.m_vertex2.Copy(this.m_vertices[index + 1]);
if (index > 0) {
edge.m_vertex0.Copy(this.m_vertices[index - 1]);
edge.m_hasVertex0 = true;
}
else {
edge.m_vertex0.Copy(this.m_prevVertex);
edge.m_hasVertex0 = this.m_hasPrevVertex;
}
if (index < this.m_count - 2) {
edge.m_vertex3.Copy(this.m_vertices[index + 2]);
edge.m_hasVertex3 = true;
}
else {
edge.m_vertex3.Copy(this.m_nextVertex);
edge.m_hasVertex3 = this.m_hasNextVertex;
}
};
/// This always return false.
/// @see b2Shape::TestPoint
b2ChainShape.prototype.TestPoint = function (xf, p) {
return false;
};
b2ChainShape.prototype.ComputeDistance = function (xf, p, normal, childIndex) {
var edge = b2ChainShape.ComputeDistance_s_edgeShape;
this.GetChildEdge(edge, childIndex);
return edge.ComputeDistance(xf, p, normal, 0);
};
b2ChainShape.prototype.RayCast = function (output, input, xf, childIndex) {
// DEBUG: b2Assert(childIndex < this.m_count);
var edgeShape = b2ChainShape.RayCast_s_edgeShape;
edgeShape.m_vertex1.Copy(this.m_vertices[childIndex]);
edgeShape.m_vertex2.Copy(this.m_vertices[(childIndex + 1) % this.m_count]);
return edgeShape.RayCast(output, input, xf, 0);
};
b2ChainShape.prototype.ComputeAABB = function (aabb, xf, childIndex) {
// DEBUG: b2Assert(childIndex < this.m_count);
var vertexi1 = this.m_vertices[childIndex];
var vertexi2 = this.m_vertices[(childIndex + 1) % this.m_count];
var v1 = b2Transform.MulXV(xf, vertexi1, b2ChainShape.ComputeAABB_s_v1);
var v2 = b2Transform.MulXV(xf, vertexi2, b2ChainShape.ComputeAABB_s_v2);
b2Vec2.MinV(v1, v2, aabb.lowerBound);
b2Vec2.MaxV(v1, v2, aabb.upperBound);
};
/// Chains have zero mass.
/// @see b2Shape::ComputeMass
b2ChainShape.prototype.ComputeMass = function (massData, density) {
massData.mass = 0;
massData.center.SetZero();
massData.I = 0;
};
b2ChainShape.prototype.SetupDistanceProxy = function (proxy, index) {
// DEBUG: b2Assert(0 <= index && index < this.m_count);
proxy.m_vertices = proxy.m_buffer;
proxy.m_vertices[0].Copy(this.m_vertices[index]);
if (index + 1 < this.m_count) {
proxy.m_vertices[1].Copy(this.m_vertices[index + 1]);
}
else {
proxy.m_vertices[1].Copy(this.m_vertices[0]);
}
proxy.m_count = 2;
proxy.m_radius = this.m_radius;
};
b2ChainShape.prototype.ComputeSubmergedArea = function (normal, offset, xf, c) {
c.SetZero();
return 0;
};
b2ChainShape.prototype.Dump = function (log) {
log(" const shape: b2ChainShape = new b2ChainShape();\n");
log(" const vs: b2Vec2[] = b2Vec2.MakeArray(%d);\n", b2_maxPolygonVertices);
for (var i = 0; i < this.m_count; ++i) {
log(" vs[%d].Set(%.15f, %.15f);\n", i, this.m_vertices[i].x, this.m_vertices[i].y);
}
log(" shape.CreateChain(vs, %d);\n", this.m_count);
log(" shape.m_prevVertex.Set(%.15f, %.15f);\n", this.m_prevVertex.x, this.m_prevVertex.y);
log(" shape.m_nextVertex.Set(%.15f, %.15f);\n", this.m_nextVertex.x, this.m_nextVertex.y);
log(" shape.m_hasPrevVertex = %s;\n", (this.m_hasPrevVertex) ? ("true") : ("false"));
log(" shape.m_hasNextVertex = %s;\n", (this.m_hasNextVertex) ? ("true") : ("false"));
};
// #if B2_ENABLE_PARTICLE
/// @see b2Shape::ComputeDistance
b2ChainShape.ComputeDistance_s_edgeShape = new b2EdgeShape();
// #endif
/// Implement b2Shape.
b2ChainShape.RayCast_s_edgeShape = new b2EdgeShape();
/// @see b2Shape::ComputeAABB
b2ChainShape.ComputeAABB_s_v1 = new b2Vec2();
b2ChainShape.ComputeAABB_s_v2 = new b2Vec2();
return b2ChainShape;
}(b2Shape));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// This holds contact filtering data.
var b2Filter = /** @class */ (function () {
function b2Filter() {
/// The collision category bits. Normally you would just set one bit.
this.categoryBits = 0x0001;
/// The collision mask bits. This states the categories that this
/// shape would accept for collision.
this.maskBits = 0xFFFF;
/// Collision groups allow a certain group of objects to never collide (negative)
/// or always collide (positive). Zero means no collision group. Non-zero group
/// filtering always wins against the mask bits.
this.groupIndex = 0;
}
b2Filter.prototype.Clone = function () {
return new b2Filter().Copy(this);
};
b2Filter.prototype.Copy = function (other) {
// DEBUG: b2Assert(this !== other);
this.categoryBits = other.categoryBits;
this.maskBits = other.maskBits;
this.groupIndex = other.groupIndex || 0;
return this;
};
b2Filter.DEFAULT = new b2Filter();
return b2Filter;
}());
/// A fixture definition is used to create a fixture. This class defines an
/// abstract fixture definition. You can reuse fixture definitions safely.
var b2FixtureDef = /** @class */ (function () {
function b2FixtureDef() {
/// Use this to store application specific fixture data.
this.userData = null;
/// The friction coefficient, usually in the range [0,1].
this.friction = 0.2;
/// The restitution (elasticity) usually in the range [0,1].
this.restitution = 0;
/// The density, usually in kg/m^2.
this.density = 0;
/// A sensor shape collects contact information but never generates a collision
/// response.
this.isSensor = false;
/// Contact filtering data.
this.filter = new b2Filter();
}
return b2FixtureDef;
}());
/// This proxy is used internally to connect fixtures to the broad-phase.
var b2FixtureProxy = /** @class */ (function () {
function b2FixtureProxy(fixture) {
this.aabb = new b2AABB();
this.childIndex = 0;
this.fixture = fixture;
}
return b2FixtureProxy;
}());
/// A fixture is used to attach a shape to a body for collision detection. A fixture
/// inherits its transform from its parent. Fixtures hold additional non-geometric data
/// such as friction, collision filters, etc.
/// Fixtures are created via b2Body::CreateFixture.
/// @warning you cannot reuse fixtures.
var b2Fixture = /** @class */ (function () {
function b2Fixture(def, body) {
this.m_density = 0;
this.m_next = null;
this.m_friction = 0;
this.m_restitution = 0;
this.m_proxies = [];
this.m_proxyCount = 0;
this.m_filter = new b2Filter();
this.m_isSensor = false;
this.m_userData = null;
this.m_body = body;
this.m_shape = def.shape.Clone();
}
/// Get the type of the child shape. You can use this to down cast to the concrete shape.
/// @return the shape type.
b2Fixture.prototype.GetType = function () {
return this.m_shape.GetType();
};
/// Get the child shape. You can modify the child shape, however you should not change the
/// number of vertices because this will crash some collision caching mechanisms.
/// Manipulating the shape may lead to non-physical behavior.
b2Fixture.prototype.GetShape = function () {
return this.m_shape;
};
/// Set if this fixture is a sensor.
b2Fixture.prototype.SetSensor = function (sensor) {
if (sensor !== this.m_isSensor) {
this.m_body.SetAwake(true);
this.m_isSensor = sensor;
}
};
/// Is this fixture a sensor (non-solid)?
/// @return the true if the shape is a sensor.
b2Fixture.prototype.IsSensor = function () {
return this.m_isSensor;
};
/// Set the contact filtering data. This will not update contacts until the next time
/// step when either parent body is active and awake.
/// This automatically calls Refilter.
b2Fixture.prototype.SetFilterData = function (filter) {
this.m_filter.Copy(filter);
this.Refilter();
};
/// Get the contact filtering data.
b2Fixture.prototype.GetFilterData = function () {
return this.m_filter;
};
/// Call this if you want to establish collision that was previously disabled by b2ContactFilter::ShouldCollide.
b2Fixture.prototype.Refilter = function () {
// Flag associated contacts for filtering.
var edge = this.m_body.GetContactList();
while (edge) {
var contact = edge.contact;
var fixtureA = contact.GetFixtureA();
var fixtureB = contact.GetFixtureB();
if (fixtureA === this || fixtureB === this) {
contact.FlagForFiltering();
}
edge = edge.next;
}
var world = this.m_body.GetWorld();
if (world === null) {
return;
}
// Touch each proxy so that new pairs may be created
var broadPhase = world.m_contactManager.m_broadPhase;
for (var i = 0; i < this.m_proxyCount; ++i) {
broadPhase.TouchProxy(this.m_proxies[i].treeNode);
}
};
/// Get the parent body of this fixture. This is NULL if the fixture is not attached.
/// @return the parent body.
b2Fixture.prototype.GetBody = function () {
return this.m_body;
};
/// Get the next fixture in the parent body's fixture list.
/// @return the next shape.
b2Fixture.prototype.GetNext = function () {
return this.m_next;
};
/// Get the user data that was assigned in the fixture definition. Use this to
/// store your application specific data.
b2Fixture.prototype.GetUserData = function () {
return this.m_userData;
};
/// Set the user data. Use this to store your application specific data.
b2Fixture.prototype.SetUserData = function (data) {
this.m_userData = data;
};
/// Test a point for containment in this fixture.
/// @param p a point in world coordinates.
b2Fixture.prototype.TestPoint = function (p) {
return this.m_shape.TestPoint(this.m_body.GetTransform(), p);
};
// #if B2_ENABLE_PARTICLE
b2Fixture.prototype.ComputeDistance = function (p, normal, childIndex) {
return this.m_shape.ComputeDistance(this.m_body.GetTransform(), p, normal, childIndex);
};
// #endif
/// Cast a ray against this shape.
/// @param output the ray-cast results.
/// @param input the ray-cast input parameters.
b2Fixture.prototype.RayCast = function (output, input, childIndex) {
return this.m_shape.RayCast(output, input, this.m_body.GetTransform(), childIndex);
};
/// Get the mass data for this fixture. The mass data is based on the density and
/// the shape. The rotational inertia is about the shape's origin. This operation
/// may be expensive.
b2Fixture.prototype.GetMassData = function (massData) {
if (massData === void 0) { massData = new b2MassData(); }
this.m_shape.ComputeMass(massData, this.m_density);
return massData;
};
/// Set the density of this fixture. This will _not_ automatically adjust the mass
/// of the body. You must call b2Body::ResetMassData to update the body's mass.
b2Fixture.prototype.SetDensity = function (density) {
this.m_density = density;
};
/// Get the density of this fixture.
b2Fixture.prototype.GetDensity = function () {
return this.m_density;
};
/// Get the coefficient of friction.
b2Fixture.prototype.GetFriction = function () {
return this.m_friction;
};
/// Set the coefficient of friction. This will _not_ change the friction of
/// existing contacts.
b2Fixture.prototype.SetFriction = function (friction) {
this.m_friction = friction;
};
/// Get the coefficient of restitution.
b2Fixture.prototype.GetRestitution = function () {
return this.m_restitution;
};
/// Set the coefficient of restitution. This will _not_ change the restitution of
/// existing contacts.
b2Fixture.prototype.SetRestitution = function (restitution) {
this.m_restitution = restitution;
};
/// Get the fixture's AABB. This AABB may be enlarge and/or stale.
/// If you need a more accurate AABB, compute it using the shape and
/// the body transform.
b2Fixture.prototype.GetAABB = function (childIndex) {
// DEBUG: b2Assert(0 <= childIndex && childIndex < this.m_proxyCount);
return this.m_proxies[childIndex].aabb;
};
/// Dump this fixture to the log file.
b2Fixture.prototype.Dump = function (log, bodyIndex) {
log(" const fd: b2FixtureDef = new b2FixtureDef();\n");
log(" fd.friction = %.15f;\n", this.m_friction);
log(" fd.restitution = %.15f;\n", this.m_restitution);
log(" fd.density = %.15f;\n", this.m_density);
log(" fd.isSensor = %s;\n", (this.m_isSensor) ? ("true") : ("false"));
log(" fd.filter.categoryBits = %d;\n", this.m_filter.categoryBits);
log(" fd.filter.maskBits = %d;\n", this.m_filter.maskBits);
log(" fd.filter.groupIndex = %d;\n", this.m_filter.groupIndex);
this.m_shape.Dump(log);
log("\n");
log(" fd.shape = shape;\n");
log("\n");
log(" bodies[%d].CreateFixture(fd);\n", bodyIndex);
};
// We need separation create/destroy functions from the constructor/destructor because
// the destructor cannot access the allocator (no destructor arguments allowed by C++).
b2Fixture.prototype.Create = function (def) {
var _this = this;
this.m_userData = def.userData;
this.m_friction = b2Maybe(def.friction, 0.2);
this.m_restitution = b2Maybe(def.restitution, 0);
// this.m_body = body;
this.m_next = null;
this.m_filter.Copy(b2Maybe(def.filter, b2Filter.DEFAULT));
this.m_isSensor = b2Maybe(def.isSensor, false);
// Reserve proxy space
// const childCount = m_shape->GetChildCount();
// m_proxies = (b2FixtureProxy*)allocator->Allocate(childCount * sizeof(b2FixtureProxy));
// for (int32 i = 0; i < childCount; ++i)
// {
// m_proxies[i].fixture = NULL;
// m_proxies[i].proxyId = b2BroadPhase::e_nullProxy;
// }
// this.m_proxies = b2FixtureProxy.MakeArray(this.m_shape.GetChildCount());
this.m_proxies = b2MakeArray(this.m_shape.GetChildCount(), function (i) { return new b2FixtureProxy(_this); });
this.m_proxyCount = 0;
this.m_density = b2Maybe(def.density, 0);
};
b2Fixture.prototype.Destroy = function () {
// The proxies must be destroyed before calling this.
// DEBUG: b2Assert(this.m_proxyCount === 0);
// Free the proxy array.
// int32 childCount = m_shape->GetChildCount();
// allocator->Free(m_proxies, childCount * sizeof(b2FixtureProxy));
// m_proxies = NULL;
// this.m_shape = null;
};
// These support body activation/deactivation.
b2Fixture.prototype.CreateProxies = function (xf) {
var broadPhase = this.m_body.m_world.m_contactManager.m_broadPhase;
// DEBUG: b2Assert(this.m_proxyCount === 0);
// Create proxies in the broad-phase.
this.m_proxyCount = this.m_shape.GetChildCount();
for (var i = 0; i < this.m_proxyCount; ++i) {
var proxy = this.m_proxies[i] = new b2FixtureProxy(this);
this.m_shape.ComputeAABB(proxy.aabb, xf, i);
proxy.treeNode = broadPhase.CreateProxy(proxy.aabb, proxy);
proxy.childIndex = i;
}
};
b2Fixture.prototype.DestroyProxies = function () {
var broadPhase = this.m_body.m_world.m_contactManager.m_broadPhase;
// Destroy proxies in the broad-phase.
for (var i = 0; i < this.m_proxyCount; ++i) {
var proxy = this.m_proxies[i];
delete proxy.treeNode.userData;
broadPhase.DestroyProxy(proxy.treeNode);
delete proxy.treeNode;
}
this.m_proxyCount = 0;
};
b2Fixture.prototype.TouchProxies = function () {
var broadPhase = this.m_body.m_world.m_contactManager.m_broadPhase;
var proxyCount = this.m_proxyCount;
for (var i = 0; i < proxyCount; ++i) {
broadPhase.TouchProxy(this.m_proxies[i].treeNode);
}
};
b2Fixture.prototype.Synchronize = function (transform1, transform2) {
if (this.m_proxyCount === 0) {
return;
}
var broadPhase = this.m_body.m_world.m_contactManager.m_broadPhase;
for (var i = 0; i < this.m_proxyCount; ++i) {
var proxy = this.m_proxies[i];
// Compute an AABB that covers the swept shape (may miss some rotation effect).
var aabb1 = b2Fixture.Synchronize_s_aabb1;
var aabb2 = b2Fixture.Synchronize_s_aabb2;
this.m_shape.ComputeAABB(aabb1, transform1, i);
this.m_shape.ComputeAABB(aabb2, transform2, i);
proxy.aabb.Combine2(aabb1, aabb2);
var displacement = b2Vec2.SubVV(transform2.p, transform1.p, b2Fixture.Synchronize_s_displacement);
broadPhase.MoveProxy(proxy.treeNode, proxy.aabb, displacement);
}
};
b2Fixture.Synchronize_s_aabb1 = new b2AABB();
b2Fixture.Synchronize_s_aabb2 = new b2AABB();
b2Fixture.Synchronize_s_displacement = new b2Vec2();
return b2Fixture;
}());
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
(function (b2BodyType) {
b2BodyType[b2BodyType["b2_unknown"] = -1] = "b2_unknown";
b2BodyType[b2BodyType["b2_staticBody"] = 0] = "b2_staticBody";
b2BodyType[b2BodyType["b2_kinematicBody"] = 1] = "b2_kinematicBody";
b2BodyType[b2BodyType["b2_dynamicBody"] = 2] = "b2_dynamicBody";
// TODO_ERIN
// b2_bulletBody = 3
})(exports.b2BodyType || (exports.b2BodyType = {}));
/// A body definition holds all the data needed to construct a rigid body.
/// You can safely re-use body definitions. Shapes are added to a body after construction.
var b2BodyDef = /** @class */ (function () {
function b2BodyDef() {
/// The body type: static, kinematic, or dynamic.
/// Note: if a dynamic body would have zero mass, the mass is set to one.
this.type = exports.b2BodyType.b2_staticBody;
/// The world position of the body. Avoid creating bodies at the origin
/// since this can lead to many overlapping shapes.
this.position = new b2Vec2(0, 0);
/// The world angle of the body in radians.
this.angle = 0;
/// The linear velocity of the body's origin in world co-ordinates.
this.linearVelocity = new b2Vec2(0, 0);
/// The angular velocity of the body.
this.angularVelocity = 0;
/// Linear damping is use to reduce the linear velocity. The damping parameter
/// can be larger than 1.0f but the damping effect becomes sensitive to the
/// time step when the damping parameter is large.
this.linearDamping = 0;
/// Angular damping is use to reduce the angular velocity. The damping parameter
/// can be larger than 1.0f but the damping effect becomes sensitive to the
/// time step when the damping parameter is large.
this.angularDamping = 0;
/// Set this flag to false if this body should never fall asleep. Note that
/// this increases CPU usage.
this.allowSleep = true;
/// Is this body initially awake or sleeping?
this.awake = true;
/// Should this body be prevented from rotating? Useful for characters.
this.fixedRotation = false;
/// Is this a fast moving body that should be prevented from tunneling through
/// other moving bodies? Note that all bodies are prevented from tunneling through
/// kinematic and static bodies. This setting is only considered on dynamic bodies.
/// @warning You should use this flag sparingly since it increases processing time.
this.bullet = false;
/// Does this body start out active?
this.active = true;
/// Use this to store application specific body data.
this.userData = null;
/// Scale the gravity applied to this body.
this.gravityScale = 1;
}
return b2BodyDef;
}());
/// A rigid body. These are created via b2World::CreateBody.
var b2Body = /** @class */ (function () {
// #endif
function b2Body(bd, world) {
this.m_type = exports.b2BodyType.b2_staticBody;
this.m_islandFlag = false;
this.m_awakeFlag = false;
this.m_autoSleepFlag = false;
this.m_bulletFlag = false;
this.m_fixedRotationFlag = false;
this.m_activeFlag = false;
this.m_toiFlag = false;
this.m_islandIndex = 0;
this.m_xf = new b2Transform(); // the body origin transform
// #if B2_ENABLE_PARTICLE
this.m_xf0 = new b2Transform();
// #endif
this.m_sweep = new b2Sweep(); // the swept motion for CCD
this.m_linearVelocity = new b2Vec2();
this.m_angularVelocity = 0;
this.m_force = new b2Vec2();
this.m_torque = 0;
this.m_prev = null;
this.m_next = null;
this.m_fixtureList = null;
this.m_fixtureCount = 0;
this.m_jointList = null;
this.m_contactList = null;
this.m_mass = 1;
this.m_invMass = 1;
// Rotational inertia about the center of mass.
this.m_I = 0;
this.m_invI = 0;
this.m_linearDamping = 0;
this.m_angularDamping = 0;
this.m_gravityScale = 1;
this.m_sleepTime = 0;
this.m_userData = null;
// #if B2_ENABLE_CONTROLLER
this.m_controllerList = null;
this.m_controllerCount = 0;
this.m_bulletFlag = b2Maybe(bd.bullet, false);
this.m_fixedRotationFlag = b2Maybe(bd.fixedRotation, false);
this.m_autoSleepFlag = b2Maybe(bd.allowSleep, true);
this.m_awakeFlag = b2Maybe(bd.awake, true);
this.m_activeFlag = b2Maybe(bd.active, true);
this.m_world = world;
this.m_xf.p.Copy(b2Maybe(bd.position, b2Vec2.ZERO));
// DEBUG: b2Assert(this.m_xf.p.IsValid());
this.m_xf.q.SetAngle(b2Maybe(bd.angle, 0));
// DEBUG: b2Assert(b2IsValid(this.m_xf.q.GetAngle()));
// #if B2_ENABLE_PARTICLE
this.m_xf0.Copy(this.m_xf);
// #endif
this.m_sweep.localCenter.SetZero();
this.m_sweep.c0.Copy(this.m_xf.p);
this.m_sweep.c.Copy(this.m_xf.p);
this.m_sweep.a0 = this.m_sweep.a = this.m_xf.q.GetAngle();
this.m_sweep.alpha0 = 0;
this.m_linearVelocity.Copy(b2Maybe(bd.linearVelocity, b2Vec2.ZERO));
// DEBUG: b2Assert(this.m_linearVelocity.IsValid());
this.m_angularVelocity = b2Maybe(bd.angularVelocity, 0);
// DEBUG: b2Assert(b2IsValid(this.m_angularVelocity));
this.m_linearDamping = b2Maybe(bd.linearDamping, 0);
this.m_angularDamping = b2Maybe(bd.angularDamping, 0);
this.m_gravityScale = b2Maybe(bd.gravityScale, 1);
// DEBUG: b2Assert(b2IsValid(this.m_gravityScale) && this.m_gravityScale >= 0);
// DEBUG: b2Assert(b2IsValid(this.m_angularDamping) && this.m_angularDamping >= 0);
// DEBUG: b2Assert(b2IsValid(this.m_linearDamping) && this.m_linearDamping >= 0);
this.m_force.SetZero();
this.m_torque = 0;
this.m_sleepTime = 0;
this.m_type = b2Maybe(bd.type, exports.b2BodyType.b2_staticBody);
if (bd.type === exports.b2BodyType.b2_dynamicBody) {
this.m_mass = 1;
this.m_invMass = 1;
}
else {
this.m_mass = 0;
this.m_invMass = 0;
}
this.m_I = 0;
this.m_invI = 0;
this.m_userData = bd.userData;
this.m_fixtureList = null;
this.m_fixtureCount = 0;
// #if B2_ENABLE_CONTROLLER
this.m_controllerList = null;
this.m_controllerCount = 0;
// #endif
}
b2Body.prototype.CreateFixture = function (a, b) {
if (b === void 0) { b = 0; }
if (a instanceof b2Shape) {
return this.CreateFixtureShapeDensity(a, b);
}
else {
return this.CreateFixtureDef(a);
}
};
/// Creates a fixture and attach it to this body. Use this function if you need
/// to set some fixture parameters, like friction. Otherwise you can create the
/// fixture directly from a shape.
/// If the density is non-zero, this function automatically updates the mass of the body.
/// Contacts are not created until the next time step.
/// @param def the fixture definition.
/// @warning This function is locked during callbacks.
b2Body.prototype.CreateFixtureDef = function (def) {
if (this.m_world.IsLocked()) {
throw new Error();
}
var fixture = new b2Fixture(def, this);
fixture.Create(def);
if (this.m_activeFlag) {
fixture.CreateProxies(this.m_xf);
}
fixture.m_next = this.m_fixtureList;
this.m_fixtureList = fixture;
++this.m_fixtureCount;
// fixture.m_body = this;
// Adjust mass properties if needed.
if (fixture.m_density > 0) {
this.ResetMassData();
}
// Let the world know we have a new fixture. This will cause new contacts
// to be created at the beginning of the next time step.
this.m_world.m_newFixture = true;
return fixture;
};
b2Body.prototype.CreateFixtureShapeDensity = function (shape, density) {
if (density === void 0) { density = 0; }
var def = b2Body.CreateFixtureShapeDensity_s_def;
def.shape = shape;
def.density = density;
return this.CreateFixtureDef(def);
};
/// Destroy a fixture. This removes the fixture from the broad-phase and
/// destroys all contacts associated with this fixture. This will
/// automatically adjust the mass of the body if the body is dynamic and the
/// fixture has positive density.
/// All fixtures attached to a body are implicitly destroyed when the body is destroyed.
/// @param fixture the fixture to be removed.
/// @warning This function is locked during callbacks.
b2Body.prototype.DestroyFixture = function (fixture) {
if (this.m_world.IsLocked()) {
throw new Error();
}
// DEBUG: b2Assert(fixture.m_body === this);
// Remove the fixture from this body's singly linked list.
// DEBUG: b2Assert(this.m_fixtureCount > 0);
var node = this.m_fixtureList;
var ppF = null;
// DEBUG: let found: boolean = false;
while (node !== null) {
if (node === fixture) {
if (ppF) {
ppF.m_next = fixture.m_next;
}
else {
this.m_fixtureList = fixture.m_next;
}
// DEBUG: found = true;
break;
}
ppF = node;
node = node.m_next;
}
// You tried to remove a shape that is not attached to this body.
// DEBUG: b2Assert(found);
// Destroy any contacts associated with the fixture.
var edge = this.m_contactList;
while (edge) {
var c = edge.contact;
edge = edge.next;
var fixtureA = c.GetFixtureA();
var fixtureB = c.GetFixtureB();
if (fixture === fixtureA || fixture === fixtureB) {
// This destroys the contact and removes it from
// this body's contact list.
this.m_world.m_contactManager.Destroy(c);
}
}
if (this.m_activeFlag) {
fixture.DestroyProxies();
}
// fixture.m_body = null;
fixture.m_next = null;
fixture.Destroy();
--this.m_fixtureCount;
// Reset the mass data.
this.ResetMassData();
};
/// Set the position of the body's origin and rotation.
/// This breaks any contacts and wakes the other bodies.
/// Manipulating a body's transform may cause non-physical behavior.
/// @param position the world position of the body's local origin.
/// @param angle the world rotation in radians.
b2Body.prototype.SetTransformVec = function (position, angle) {
this.SetTransformXY(position.x, position.y, angle);
};
b2Body.prototype.SetTransformXY = function (x, y, angle) {
if (this.m_world.IsLocked()) {
throw new Error();
}
this.m_xf.q.SetAngle(angle);
this.m_xf.p.Set(x, y);
// #if B2_ENABLE_PARTICLE
this.m_xf0.Copy(this.m_xf);
// #endif
b2Transform.MulXV(this.m_xf, this.m_sweep.localCenter, this.m_sweep.c);
this.m_sweep.a = angle;
this.m_sweep.c0.Copy(this.m_sweep.c);
this.m_sweep.a0 = angle;
for (var f = this.m_fixtureList; f; f = f.m_next) {
f.Synchronize(this.m_xf, this.m_xf);
}
this.m_world.m_contactManager.FindNewContacts();
};
b2Body.prototype.SetTransform = function (xf) {
this.SetTransformVec(xf.p, xf.GetAngle());
};
/// Get the body transform for the body's origin.
/// @return the world transform of the body's origin.
b2Body.prototype.GetTransform = function () {
return this.m_xf;
};
/// Get the world body origin position.
/// @return the world position of the body's origin.
b2Body.prototype.GetPosition = function () {
return this.m_xf.p;
};
b2Body.prototype.SetPosition = function (position) {
this.SetTransformVec(position, this.GetAngle());
};
b2Body.prototype.SetPositionXY = function (x, y) {
this.SetTransformXY(x, y, this.GetAngle());
};
/// Get the angle in radians.
/// @return the current world rotation angle in radians.
b2Body.prototype.GetAngle = function () {
return this.m_sweep.a;
};
b2Body.prototype.SetAngle = function (angle) {
this.SetTransformVec(this.GetPosition(), angle);
};
/// Get the world position of the center of mass.
b2Body.prototype.GetWorldCenter = function () {
return this.m_sweep.c;
};
/// Get the local position of the center of mass.
b2Body.prototype.GetLocalCenter = function () {
return this.m_sweep.localCenter;
};
/// Set the linear velocity of the center of mass.
/// @param v the new linear velocity of the center of mass.
b2Body.prototype.SetLinearVelocity = function (v) {
if (this.m_type === exports.b2BodyType.b2_staticBody) {
return;
}
if (b2Vec2.DotVV(v, v) > 0) {
this.SetAwake(true);
}
this.m_linearVelocity.Copy(v);
};
/// Get the linear velocity of the center of mass.
/// @return the linear velocity of the center of mass.
b2Body.prototype.GetLinearVelocity = function () {
return this.m_linearVelocity;
};
/// Set the angular velocity.
/// @param omega the new angular velocity in radians/second.
b2Body.prototype.SetAngularVelocity = function (w) {
if (this.m_type === exports.b2BodyType.b2_staticBody) {
return;
}
if (w * w > 0) {
this.SetAwake(true);
}
this.m_angularVelocity = w;
};
/// Get the angular velocity.
/// @return the angular velocity in radians/second.
b2Body.prototype.GetAngularVelocity = function () {
return this.m_angularVelocity;
};
b2Body.prototype.GetDefinition = function (bd) {
bd.type = this.GetType();
bd.allowSleep = this.m_autoSleepFlag;
bd.angle = this.GetAngle();
bd.angularDamping = this.m_angularDamping;
bd.gravityScale = this.m_gravityScale;
bd.angularVelocity = this.m_angularVelocity;
bd.fixedRotation = this.m_fixedRotationFlag;
bd.bullet = this.m_bulletFlag;
bd.awake = this.m_awakeFlag;
bd.linearDamping = this.m_linearDamping;
bd.linearVelocity.Copy(this.GetLinearVelocity());
bd.position.Copy(this.GetPosition());
bd.userData = this.GetUserData();
return bd;
};
/// Apply a force at a world point. If the force is not
/// applied at the center of mass, it will generate a torque and
/// affect the angular velocity. This wakes up the body.
/// @param force the world force vector, usually in Newtons (N).
/// @param point the world position of the point of application.
/// @param wake also wake up the body
b2Body.prototype.ApplyForce = function (force, point, wake) {
if (wake === void 0) { wake = true; }
if (this.m_type !== exports.b2BodyType.b2_dynamicBody) {
return;
}
if (wake && !this.m_awakeFlag) {
this.SetAwake(true);
}
// Don't accumulate a force if the body is sleeping.
if (this.m_awakeFlag) {
this.m_force.x += force.x;
this.m_force.y += force.y;
this.m_torque += ((point.x - this.m_sweep.c.x) * force.y - (point.y - this.m_sweep.c.y) * force.x);
}
};
/// Apply a force to the center of mass. This wakes up the body.
/// @param force the world force vector, usually in Newtons (N).
/// @param wake also wake up the body
b2Body.prototype.ApplyForceToCenter = function (force, wake) {
if (wake === void 0) { wake = true; }
if (this.m_type !== exports.b2BodyType.b2_dynamicBody) {
return;
}
if (wake && !this.m_awakeFlag) {
this.SetAwake(true);
}
// Don't accumulate a force if the body is sleeping.
if (this.m_awakeFlag) {
this.m_force.x += force.x;
this.m_force.y += force.y;
}
};
/// Apply a torque. This affects the angular velocity
/// without affecting the linear velocity of the center of mass.
/// @param torque about the z-axis (out of the screen), usually in N-m.
/// @param wake also wake up the body
b2Body.prototype.ApplyTorque = function (torque, wake) {
if (wake === void 0) { wake = true; }
if (this.m_type !== exports.b2BodyType.b2_dynamicBody) {
return;
}
if (wake && !this.m_awakeFlag) {
this.SetAwake(true);
}
// Don't accumulate a force if the body is sleeping.
if (this.m_awakeFlag) {
this.m_torque += torque;
}
};
/// Apply an impulse at a point. This immediately modifies the velocity.
/// It also modifies the angular velocity if the point of application
/// is not at the center of mass. This wakes up the body.
/// @param impulse the world impulse vector, usually in N-seconds or kg-m/s.
/// @param point the world position of the point of application.
/// @param wake also wake up the body
b2Body.prototype.ApplyLinearImpulse = function (impulse, point, wake) {
if (wake === void 0) { wake = true; }
if (this.m_type !== exports.b2BodyType.b2_dynamicBody) {
return;
}
if (wake && !this.m_awakeFlag) {
this.SetAwake(true);
}
// Don't accumulate a force if the body is sleeping.
if (this.m_awakeFlag) {
this.m_linearVelocity.x += this.m_invMass * impulse.x;
this.m_linearVelocity.y += this.m_invMass * impulse.y;
this.m_angularVelocity += this.m_invI * ((point.x - this.m_sweep.c.x) * impulse.y - (point.y - this.m_sweep.c.y) * impulse.x);
}
};
/// Apply an impulse at the center of gravity. This immediately modifies the velocity.
/// @param impulse the world impulse vector, usually in N-seconds or kg-m/s.
/// @param wake also wake up the body
b2Body.prototype.ApplyLinearImpulseToCenter = function (impulse, wake) {
if (wake === void 0) { wake = true; }
if (this.m_type !== exports.b2BodyType.b2_dynamicBody) {
return;
}
if (wake && !this.m_awakeFlag) {
this.SetAwake(true);
}
// Don't accumulate a force if the body is sleeping.
if (this.m_awakeFlag) {
this.m_linearVelocity.x += this.m_invMass * impulse.x;
this.m_linearVelocity.y += this.m_invMass * impulse.y;
}
};
/// Apply an angular impulse.
/// @param impulse the angular impulse in units of kg*m*m/s
/// @param wake also wake up the body
b2Body.prototype.ApplyAngularImpulse = function (impulse, wake) {
if (wake === void 0) { wake = true; }
if (this.m_type !== exports.b2BodyType.b2_dynamicBody) {
return;
}
if (wake && !this.m_awakeFlag) {
this.SetAwake(true);
}
// Don't accumulate a force if the body is sleeping.
if (this.m_awakeFlag) {
this.m_angularVelocity += this.m_invI * impulse;
}
};
/// Get the total mass of the body.
/// @return the mass, usually in kilograms (kg).
b2Body.prototype.GetMass = function () {
return this.m_mass;
};
/// Get the rotational inertia of the body about the local origin.
/// @return the rotational inertia, usually in kg-m^2.
b2Body.prototype.GetInertia = function () {
return this.m_I + this.m_mass * b2Vec2.DotVV(this.m_sweep.localCenter, this.m_sweep.localCenter);
};
/// Get the mass data of the body.
/// @return a struct containing the mass, inertia and center of the body.
b2Body.prototype.GetMassData = function (data) {
data.mass = this.m_mass;
data.I = this.m_I + this.m_mass * b2Vec2.DotVV(this.m_sweep.localCenter, this.m_sweep.localCenter);
data.center.Copy(this.m_sweep.localCenter);
return data;
};
b2Body.prototype.SetMassData = function (massData) {
if (this.m_world.IsLocked()) {
throw new Error();
}
if (this.m_type !== exports.b2BodyType.b2_dynamicBody) {
return;
}
this.m_invMass = 0;
this.m_I = 0;
this.m_invI = 0;
this.m_mass = massData.mass;
if (this.m_mass <= 0) {
this.m_mass = 1;
}
this.m_invMass = 1 / this.m_mass;
if (massData.I > 0 && !this.m_fixedRotationFlag) {
this.m_I = massData.I - this.m_mass * b2Vec2.DotVV(massData.center, massData.center);
// DEBUG: b2Assert(this.m_I > 0);
this.m_invI = 1 / this.m_I;
}
// Move center of mass.
var oldCenter = b2Body.SetMassData_s_oldCenter.Copy(this.m_sweep.c);
this.m_sweep.localCenter.Copy(massData.center);
b2Transform.MulXV(this.m_xf, this.m_sweep.localCenter, this.m_sweep.c);
this.m_sweep.c0.Copy(this.m_sweep.c);
// Update center of mass velocity.
b2Vec2.AddVCrossSV(this.m_linearVelocity, this.m_angularVelocity, b2Vec2.SubVV(this.m_sweep.c, oldCenter, b2Vec2.s_t0), this.m_linearVelocity);
};
b2Body.prototype.ResetMassData = function () {
// Compute mass data from shapes. Each shape has its own density.
this.m_mass = 0;
this.m_invMass = 0;
this.m_I = 0;
this.m_invI = 0;
this.m_sweep.localCenter.SetZero();
// Static and kinematic bodies have zero mass.
if (this.m_type === exports.b2BodyType.b2_staticBody || this.m_type === exports.b2BodyType.b2_kinematicBody) {
this.m_sweep.c0.Copy(this.m_xf.p);
this.m_sweep.c.Copy(this.m_xf.p);
this.m_sweep.a0 = this.m_sweep.a;
return;
}
// DEBUG: b2Assert(this.m_type === b2BodyType.b2_dynamicBody);
// Accumulate mass over all fixtures.
var localCenter = b2Body.ResetMassData_s_localCenter.SetZero();
for (var f = this.m_fixtureList; f; f = f.m_next) {
if (f.m_density === 0) {
continue;
}
var massData = f.GetMassData(b2Body.ResetMassData_s_massData);
this.m_mass += massData.mass;
localCenter.x += massData.center.x * massData.mass;
localCenter.y += massData.center.y * massData.mass;
this.m_I += massData.I;
}
// Compute center of mass.
if (this.m_mass > 0) {
this.m_invMass = 1 / this.m_mass;
localCenter.x *= this.m_invMass;
localCenter.y *= this.m_invMass;
}
else {
// Force all dynamic bodies to have a positive mass.
this.m_mass = 1;
this.m_invMass = 1;
}
if (this.m_I > 0 && !this.m_fixedRotationFlag) {
// Center the inertia about the center of mass.
this.m_I -= this.m_mass * b2Vec2.DotVV(localCenter, localCenter);
// DEBUG: b2Assert(this.m_I > 0);
this.m_invI = 1 / this.m_I;
}
else {
this.m_I = 0;
this.m_invI = 0;
}
// Move center of mass.
var oldCenter = b2Body.ResetMassData_s_oldCenter.Copy(this.m_sweep.c);
this.m_sweep.localCenter.Copy(localCenter);
b2Transform.MulXV(this.m_xf, this.m_sweep.localCenter, this.m_sweep.c);
this.m_sweep.c0.Copy(this.m_sweep.c);
// Update center of mass velocity.
b2Vec2.AddVCrossSV(this.m_linearVelocity, this.m_angularVelocity, b2Vec2.SubVV(this.m_sweep.c, oldCenter, b2Vec2.s_t0), this.m_linearVelocity);
};
/// Get the world coordinates of a point given the local coordinates.
/// @param localPoint a point on the body measured relative the the body's origin.
/// @return the same point expressed in world coordinates.
b2Body.prototype.GetWorldPoint = function (localPoint, out) {
return b2Transform.MulXV(this.m_xf, localPoint, out);
};
/// Get the world coordinates of a vector given the local coordinates.
/// @param localVector a vector fixed in the body.
/// @return the same vector expressed in world coordinates.
b2Body.prototype.GetWorldVector = function (localVector, out) {
return b2Rot.MulRV(this.m_xf.q, localVector, out);
};
/// Gets a local point relative to the body's origin given a world point.
/// @param a point in world coordinates.
/// @return the corresponding local point relative to the body's origin.
b2Body.prototype.GetLocalPoint = function (worldPoint, out) {
return b2Transform.MulTXV(this.m_xf, worldPoint, out);
};
/// Gets a local vector given a world vector.
/// @param a vector in world coordinates.
/// @return the corresponding local vector.
b2Body.prototype.GetLocalVector = function (worldVector, out) {
return b2Rot.MulTRV(this.m_xf.q, worldVector, out);
};
/// Get the world linear velocity of a world point attached to this body.
/// @param a point in world coordinates.
/// @return the world velocity of a point.
b2Body.prototype.GetLinearVelocityFromWorldPoint = function (worldPoint, out) {
return b2Vec2.AddVCrossSV(this.m_linearVelocity, this.m_angularVelocity, b2Vec2.SubVV(worldPoint, this.m_sweep.c, b2Vec2.s_t0), out);
};
/// Get the world velocity of a local point.
/// @param a point in local coordinates.
/// @return the world velocity of a point.
b2Body.prototype.GetLinearVelocityFromLocalPoint = function (localPoint, out) {
return this.GetLinearVelocityFromWorldPoint(this.GetWorldPoint(localPoint, out), out);
};
/// Get the linear damping of the body.
b2Body.prototype.GetLinearDamping = function () {
return this.m_linearDamping;
};
/// Set the linear damping of the body.
b2Body.prototype.SetLinearDamping = function (linearDamping) {
this.m_linearDamping = linearDamping;
};
/// Get the angular damping of the body.
b2Body.prototype.GetAngularDamping = function () {
return this.m_angularDamping;
};
/// Set the angular damping of the body.
b2Body.prototype.SetAngularDamping = function (angularDamping) {
this.m_angularDamping = angularDamping;
};
/// Get the gravity scale of the body.
b2Body.prototype.GetGravityScale = function () {
return this.m_gravityScale;
};
/// Set the gravity scale of the body.
b2Body.prototype.SetGravityScale = function (scale) {
this.m_gravityScale = scale;
};
/// Set the type of this body. This may alter the mass and velocity.
b2Body.prototype.SetType = function (type) {
if (this.m_world.IsLocked()) {
throw new Error();
}
if (this.m_type === type) {
return;
}
this.m_type = type;
this.ResetMassData();
if (this.m_type === exports.b2BodyType.b2_staticBody) {
this.m_linearVelocity.SetZero();
this.m_angularVelocity = 0;
this.m_sweep.a0 = this.m_sweep.a;
this.m_sweep.c0.Copy(this.m_sweep.c);
this.SynchronizeFixtures();
}
this.SetAwake(true);
this.m_force.SetZero();
this.m_torque = 0;
// Delete the attached contacts.
var ce = this.m_contactList;
while (ce) {
var ce0 = ce;
ce = ce.next;
this.m_world.m_contactManager.Destroy(ce0.contact);
}
this.m_contactList = null;
// Touch the proxies so that new contacts will be created (when appropriate)
for (var f = this.m_fixtureList; f; f = f.m_next) {
f.TouchProxies();
}
};
/// Get the type of this body.
b2Body.prototype.GetType = function () {
return this.m_type;
};
/// Should this body be treated like a bullet for continuous collision detection?
b2Body.prototype.SetBullet = function (flag) {
this.m_bulletFlag = flag;
};
/// Is this body treated like a bullet for continuous collision detection?
b2Body.prototype.IsBullet = function () {
return this.m_bulletFlag;
};
/// You can disable sleeping on this body. If you disable sleeping, the
/// body will be woken.
b2Body.prototype.SetSleepingAllowed = function (flag) {
this.m_autoSleepFlag = flag;
if (!flag) {
this.SetAwake(true);
}
};
/// Is this body allowed to sleep
b2Body.prototype.IsSleepingAllowed = function () {
return this.m_autoSleepFlag;
};
/// Set the sleep state of the body. A sleeping body has very
/// low CPU cost.
/// @param flag set to true to wake the body, false to put it to sleep.
b2Body.prototype.SetAwake = function (flag) {
if (flag) {
this.m_awakeFlag = true;
this.m_sleepTime = 0;
}
else {
this.m_awakeFlag = false;
this.m_sleepTime = 0;
this.m_linearVelocity.SetZero();
this.m_angularVelocity = 0;
this.m_force.SetZero();
this.m_torque = 0;
}
};
/// Get the sleeping state of this body.
/// @return true if the body is sleeping.
b2Body.prototype.IsAwake = function () {
return this.m_awakeFlag;
};
/// Set the active state of the body. An inactive body is not
/// simulated and cannot be collided with or woken up.
/// If you pass a flag of true, all fixtures will be added to the
/// broad-phase.
/// If you pass a flag of false, all fixtures will be removed from
/// the broad-phase and all contacts will be destroyed.
/// Fixtures and joints are otherwise unaffected. You may continue
/// to create/destroy fixtures and joints on inactive bodies.
/// Fixtures on an inactive body are implicitly inactive and will
/// not participate in collisions, ray-casts, or queries.
/// Joints connected to an inactive body are implicitly inactive.
/// An inactive body is still owned by a b2World object and remains
/// in the body list.
b2Body.prototype.SetActive = function (flag) {
if (this.m_world.IsLocked()) {
throw new Error();
}
if (flag === this.IsActive()) {
return;
}
this.m_activeFlag = flag;
if (flag) {
// Create all proxies.
for (var f = this.m_fixtureList; f; f = f.m_next) {
f.CreateProxies(this.m_xf);
}
// Contacts are created the next time step.
}
else {
// Destroy all proxies.
for (var f = this.m_fixtureList; f; f = f.m_next) {
f.DestroyProxies();
}
// Destroy the attached contacts.
var ce = this.m_contactList;
while (ce) {
var ce0 = ce;
ce = ce.next;
this.m_world.m_contactManager.Destroy(ce0.contact);
}
this.m_contactList = null;
}
};
/// Get the active state of the body.
b2Body.prototype.IsActive = function () {
return this.m_activeFlag;
};
/// Set this body to have fixed rotation. This causes the mass
/// to be reset.
b2Body.prototype.SetFixedRotation = function (flag) {
if (this.m_fixedRotationFlag === flag) {
return;
}
this.m_fixedRotationFlag = flag;
this.m_angularVelocity = 0;
this.ResetMassData();
};
/// Does this body have fixed rotation?
b2Body.prototype.IsFixedRotation = function () {
return this.m_fixedRotationFlag;
};
/// Get the list of all fixtures attached to this body.
b2Body.prototype.GetFixtureList = function () {
return this.m_fixtureList;
};
/// Get the list of all joints attached to this body.
b2Body.prototype.GetJointList = function () {
return this.m_jointList;
};
/// Get the list of all contacts attached to this body.
/// @warning this list changes during the time step and you may
/// miss some collisions if you don't use b2ContactListener.
b2Body.prototype.GetContactList = function () {
return this.m_contactList;
};
/// Get the next body in the world's body list.
b2Body.prototype.GetNext = function () {
return this.m_next;
};
/// Get the user data pointer that was provided in the body definition.
b2Body.prototype.GetUserData = function () {
return this.m_userData;
};
/// Set the user data. Use this to store your application specific data.
b2Body.prototype.SetUserData = function (data) {
this.m_userData = data;
};
/// Get the parent world of this body.
b2Body.prototype.GetWorld = function () {
return this.m_world;
};
/// Dump this body to a log file
b2Body.prototype.Dump = function (log) {
var bodyIndex = this.m_islandIndex;
log("{\n");
log(" const bd: b2BodyDef = new b2BodyDef();\n");
var type_str = "";
switch (this.m_type) {
case exports.b2BodyType.b2_staticBody:
type_str = "b2BodyType.b2_staticBody";
break;
case exports.b2BodyType.b2_kinematicBody:
type_str = "b2BodyType.b2_kinematicBody";
break;
case exports.b2BodyType.b2_dynamicBody:
type_str = "b2BodyType.b2_dynamicBody";
break;
default:
// DEBUG: b2Assert(false);
break;
}
log(" bd.type = %s;\n", type_str);
log(" bd.position.Set(%.15f, %.15f);\n", this.m_xf.p.x, this.m_xf.p.y);
log(" bd.angle = %.15f;\n", this.m_sweep.a);
log(" bd.linearVelocity.Set(%.15f, %.15f);\n", this.m_linearVelocity.x, this.m_linearVelocity.y);
log(" bd.angularVelocity = %.15f;\n", this.m_angularVelocity);
log(" bd.linearDamping = %.15f;\n", this.m_linearDamping);
log(" bd.angularDamping = %.15f;\n", this.m_angularDamping);
log(" bd.allowSleep = %s;\n", (this.m_autoSleepFlag) ? ("true") : ("false"));
log(" bd.awake = %s;\n", (this.m_awakeFlag) ? ("true") : ("false"));
log(" bd.fixedRotation = %s;\n", (this.m_fixedRotationFlag) ? ("true") : ("false"));
log(" bd.bullet = %s;\n", (this.m_bulletFlag) ? ("true") : ("false"));
log(" bd.active = %s;\n", (this.m_activeFlag) ? ("true") : ("false"));
log(" bd.gravityScale = %.15f;\n", this.m_gravityScale);
log("\n");
log(" bodies[%d] = this.m_world.CreateBody(bd);\n", this.m_islandIndex);
log("\n");
for (var f = this.m_fixtureList; f; f = f.m_next) {
log(" {\n");
f.Dump(log, bodyIndex);
log(" }\n");
}
log("}\n");
};
b2Body.prototype.SynchronizeFixtures = function () {
var xf1 = b2Body.SynchronizeFixtures_s_xf1;
xf1.q.SetAngle(this.m_sweep.a0);
b2Rot.MulRV(xf1.q, this.m_sweep.localCenter, xf1.p);
b2Vec2.SubVV(this.m_sweep.c0, xf1.p, xf1.p);
for (var f = this.m_fixtureList; f; f = f.m_next) {
f.Synchronize(xf1, this.m_xf);
}
};
b2Body.prototype.SynchronizeTransform = function () {
this.m_xf.q.SetAngle(this.m_sweep.a);
b2Rot.MulRV(this.m_xf.q, this.m_sweep.localCenter, this.m_xf.p);
b2Vec2.SubVV(this.m_sweep.c, this.m_xf.p, this.m_xf.p);
};
// This is used to prevent connected bodies from colliding.
// It may lie, depending on the collideConnected flag.
b2Body.prototype.ShouldCollide = function (other) {
// At least one body should be dynamic or kinematic.
if (this.m_type === exports.b2BodyType.b2_staticBody && other.m_type === exports.b2BodyType.b2_staticBody) {
return false;
}
return this.ShouldCollideConnected(other);
};
b2Body.prototype.ShouldCollideConnected = function (other) {
// Does a joint prevent collision?
for (var jn = this.m_jointList; jn; jn = jn.next) {
if (jn.other === other) {
if (!jn.joint.m_collideConnected) {
return false;
}
}
}
return true;
};
b2Body.prototype.Advance = function (alpha) {
// Advance to the new safe time. This doesn't sync the broad-phase.
this.m_sweep.Advance(alpha);
this.m_sweep.c.Copy(this.m_sweep.c0);
this.m_sweep.a = this.m_sweep.a0;
this.m_xf.q.SetAngle(this.m_sweep.a);
b2Rot.MulRV(this.m_xf.q, this.m_sweep.localCenter, this.m_xf.p);
b2Vec2.SubVV(this.m_sweep.c, this.m_xf.p, this.m_xf.p);
};
// #if B2_ENABLE_CONTROLLER
b2Body.prototype.GetControllerList = function () {
return this.m_controllerList;
};
b2Body.prototype.GetControllerCount = function () {
return this.m_controllerCount;
};
/// Creates a fixture from a shape and attach it to this body.
/// This is a convenience function. Use b2FixtureDef if you need to set parameters
/// like friction, restitution, user data, or filtering.
/// If the density is non-zero, this function automatically updates the mass of the body.
/// @param shape the shape to be cloned.
/// @param density the shape density (set to zero for static bodies).
/// @warning This function is locked during callbacks.
b2Body.CreateFixtureShapeDensity_s_def = new b2FixtureDef();
/// Set the mass properties to override the mass properties of the fixtures.
/// Note that this changes the center of mass position.
/// Note that creating or destroying fixtures can also alter the mass.
/// This function has no effect if the body isn't dynamic.
/// @param massData the mass properties.
b2Body.SetMassData_s_oldCenter = new b2Vec2();
/// This resets the mass properties to the sum of the mass properties of the fixtures.
/// This normally does not need to be called unless you called SetMassData to override
/// the mass and you later want to reset the mass.
b2Body.ResetMassData_s_localCenter = new b2Vec2();
b2Body.ResetMassData_s_oldCenter = new b2Vec2();
b2Body.ResetMassData_s_massData = new b2MassData();
b2Body.SynchronizeFixtures_s_xf1 = new b2Transform();
return b2Body;
}());
/*
* Copyright (c) 2006-2007 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
(function (b2JointType) {
b2JointType[b2JointType["e_unknownJoint"] = 0] = "e_unknownJoint";
b2JointType[b2JointType["e_revoluteJoint"] = 1] = "e_revoluteJoint";
b2JointType[b2JointType["e_prismaticJoint"] = 2] = "e_prismaticJoint";
b2JointType[b2JointType["e_distanceJoint"] = 3] = "e_distanceJoint";
b2JointType[b2JointType["e_pulleyJoint"] = 4] = "e_pulleyJoint";
b2JointType[b2JointType["e_mouseJoint"] = 5] = "e_mouseJoint";
b2JointType[b2JointType["e_gearJoint"] = 6] = "e_gearJoint";
b2JointType[b2JointType["e_wheelJoint"] = 7] = "e_wheelJoint";
b2JointType[b2JointType["e_weldJoint"] = 8] = "e_weldJoint";
b2JointType[b2JointType["e_frictionJoint"] = 9] = "e_frictionJoint";
b2JointType[b2JointType["e_ropeJoint"] = 10] = "e_ropeJoint";
b2JointType[b2JointType["e_motorJoint"] = 11] = "e_motorJoint";
b2JointType[b2JointType["e_areaJoint"] = 12] = "e_areaJoint";
})(exports.b2JointType || (exports.b2JointType = {}));
(function (b2LimitState) {
b2LimitState[b2LimitState["e_inactiveLimit"] = 0] = "e_inactiveLimit";
b2LimitState[b2LimitState["e_atLowerLimit"] = 1] = "e_atLowerLimit";
b2LimitState[b2LimitState["e_atUpperLimit"] = 2] = "e_atUpperLimit";
b2LimitState[b2LimitState["e_equalLimits"] = 3] = "e_equalLimits";
})(exports.b2LimitState || (exports.b2LimitState = {}));
var b2Jacobian = /** @class */ (function () {
function b2Jacobian() {
this.linear = new b2Vec2();
this.angularA = 0;
this.angularB = 0;
}
b2Jacobian.prototype.SetZero = function () {
this.linear.SetZero();
this.angularA = 0;
this.angularB = 0;
return this;
};
b2Jacobian.prototype.Set = function (x, a1, a2) {
this.linear.Copy(x);
this.angularA = a1;
this.angularB = a2;
return this;
};
return b2Jacobian;
}());
/// A joint edge is used to connect bodies and joints together
/// in a joint graph where each body is a node and each joint
/// is an edge. A joint edge belongs to a doubly linked list
/// maintained in each attached body. Each joint has two joint
/// nodes, one for each attached body.
var b2JointEdge = /** @class */ (function () {
function b2JointEdge(joint, other) {
this.prev = null; ///< the previous joint edge in the body's joint list
this.next = null; ///< the next joint edge in the body's joint list
this.joint = joint;
this.other = other;
}
return b2JointEdge;
}());
/// Joint definitions are used to construct joints.
var b2JointDef = /** @class */ (function () {
function b2JointDef(type) {
/// The joint type is set automatically for concrete joint types.
this.type = exports.b2JointType.e_unknownJoint;
/// Use this to attach application specific data to your joints.
this.userData = null;
/// Set this flag to true if the attached bodies should collide.
this.collideConnected = false;
this.type = type;
}
return b2JointDef;
}());
/// The base joint class. Joints are used to constraint two bodies together in
/// various fashions. Some joints also feature limits and motors.
var b2Joint = /** @class */ (function () {
function b2Joint(def) {
// DEBUG: b2Assert(def.bodyA !== def.bodyB);
this.m_type = exports.b2JointType.e_unknownJoint;
this.m_prev = null;
this.m_next = null;
this.m_index = 0;
this.m_islandFlag = false;
this.m_collideConnected = false;
this.m_userData = null;
this.m_type = def.type;
this.m_edgeA = new b2JointEdge(this, def.bodyB);
this.m_edgeB = new b2JointEdge(this, def.bodyA);
this.m_bodyA = def.bodyA;
this.m_bodyB = def.bodyB;
this.m_collideConnected = b2Maybe(def.collideConnected, false);
this.m_userData = def.userData;
}
/// Get the type of the concrete joint.
b2Joint.prototype.GetType = function () {
return this.m_type;
};
/// Get the first body attached to this joint.
b2Joint.prototype.GetBodyA = function () {
return this.m_bodyA;
};
/// Get the second body attached to this joint.
b2Joint.prototype.GetBodyB = function () {
return this.m_bodyB;
};
/// Get the next joint the world joint list.
b2Joint.prototype.GetNext = function () {
return this.m_next;
};
/// Get the user data pointer.
b2Joint.prototype.GetUserData = function () {
return this.m_userData;
};
/// Set the user data pointer.
b2Joint.prototype.SetUserData = function (data) {
this.m_userData = data;
};
/// Short-cut function to determine if either body is inactive.
b2Joint.prototype.IsActive = function () {
return this.m_bodyA.IsActive() && this.m_bodyB.IsActive();
};
/// Get collide connected.
/// Note: modifying the collide connect flag won't work correctly because
/// the flag is only checked when fixture AABBs begin to overlap.
b2Joint.prototype.GetCollideConnected = function () {
return this.m_collideConnected;
};
/// Dump this joint to the log file.
b2Joint.prototype.Dump = function (log) {
log("// Dump is not supported for this joint type.\n");
};
/// Shift the origin for any points stored in world coordinates.
b2Joint.prototype.ShiftOrigin = function (newOrigin) {
};
return b2Joint;
}());
/*
* Copyright (c) 2006-2007 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Distance joint definition. This requires defining an
/// anchor point on both bodies and the non-zero length of the
/// distance joint. The definition uses local anchor points
/// so that the initial configuration can violate the constraint
/// slightly. This helps when saving and loading a game.
/// @warning Do not use a zero or short length.
var b2DistanceJointDef = /** @class */ (function (_super) {
__extends(b2DistanceJointDef, _super);
function b2DistanceJointDef() {
var _this = _super.call(this, exports.b2JointType.e_distanceJoint) || this;
_this.localAnchorA = new b2Vec2();
_this.localAnchorB = new b2Vec2();
_this.length = 1;
_this.frequencyHz = 0;
_this.dampingRatio = 0;
return _this;
}
b2DistanceJointDef.prototype.Initialize = function (b1, b2, anchor1, anchor2) {
this.bodyA = b1;
this.bodyB = b2;
this.bodyA.GetLocalPoint(anchor1, this.localAnchorA);
this.bodyB.GetLocalPoint(anchor2, this.localAnchorB);
this.length = b2Vec2.DistanceVV(anchor1, anchor2);
this.frequencyHz = 0;
this.dampingRatio = 0;
};
return b2DistanceJointDef;
}(b2JointDef));
var b2DistanceJoint = /** @class */ (function (_super) {
__extends(b2DistanceJoint, _super);
function b2DistanceJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_frequencyHz = 0;
_this.m_dampingRatio = 0;
_this.m_bias = 0;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_gamma = 0;
_this.m_impulse = 0;
_this.m_length = 0;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_u = new b2Vec2();
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_mass = 0;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_frequencyHz = b2Maybe(def.frequencyHz, 0);
_this.m_dampingRatio = b2Maybe(def.dampingRatio, 0);
_this.m_localAnchorA.Copy(def.localAnchorA);
_this.m_localAnchorB.Copy(def.localAnchorB);
_this.m_length = def.length;
return _this;
}
b2DistanceJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2DistanceJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2DistanceJoint.prototype.GetReactionForce = function (inv_dt, out) {
out.x = inv_dt * this.m_impulse * this.m_u.x;
out.y = inv_dt * this.m_impulse * this.m_u.y;
return out;
};
b2DistanceJoint.prototype.GetReactionTorque = function (inv_dt) {
return 0;
};
b2DistanceJoint.prototype.GetLocalAnchorA = function () { return this.m_localAnchorA; };
b2DistanceJoint.prototype.GetLocalAnchorB = function () { return this.m_localAnchorB; };
b2DistanceJoint.prototype.SetLength = function (length) {
this.m_length = length;
};
b2DistanceJoint.prototype.Length = function () {
return this.m_length;
};
b2DistanceJoint.prototype.SetFrequency = function (hz) {
this.m_frequencyHz = hz;
};
b2DistanceJoint.prototype.GetFrequency = function () {
return this.m_frequencyHz;
};
b2DistanceJoint.prototype.SetDampingRatio = function (ratio) {
this.m_dampingRatio = ratio;
};
b2DistanceJoint.prototype.GetDampingRatio = function () {
return this.m_dampingRatio;
};
b2DistanceJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2DistanceJointDef = new b2DistanceJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.length = %.15f;\n", this.m_length);
log(" jd.frequencyHz = %.15f;\n", this.m_frequencyHz);
log(" jd.dampingRatio = %.15f;\n", this.m_dampingRatio);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2DistanceJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// const qA: b2Rot = new b2Rot(aA), qB: b2Rot = new b2Rot(aB);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// m_rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// m_rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// m_u = cB + m_rB - cA - m_rA;
this.m_u.x = cB.x + this.m_rB.x - cA.x - this.m_rA.x;
this.m_u.y = cB.y + this.m_rB.y - cA.y - this.m_rA.y;
// Handle singularity.
var length = this.m_u.Length();
if (length > b2_linearSlop) {
this.m_u.SelfMul(1 / length);
}
else {
this.m_u.SetZero();
}
// float32 crAu = b2Cross(m_rA, m_u);
var crAu = b2Vec2.CrossVV(this.m_rA, this.m_u);
// float32 crBu = b2Cross(m_rB, m_u);
var crBu = b2Vec2.CrossVV(this.m_rB, this.m_u);
// float32 invMass = m_invMassA + m_invIA * crAu * crAu + m_invMassB + m_invIB * crBu * crBu;
var invMass = this.m_invMassA + this.m_invIA * crAu * crAu + this.m_invMassB + this.m_invIB * crBu * crBu;
// Compute the effective mass matrix.
this.m_mass = invMass !== 0 ? 1 / invMass : 0;
if (this.m_frequencyHz > 0) {
var C = length - this.m_length;
// Frequency
var omega = 2 * b2_pi * this.m_frequencyHz;
// Damping coefficient
var d = 2 * this.m_mass * this.m_dampingRatio * omega;
// Spring stiffness
var k = this.m_mass * omega * omega;
// magic formulas
var h = data.step.dt;
this.m_gamma = h * (d + h * k);
this.m_gamma = this.m_gamma !== 0 ? 1 / this.m_gamma : 0;
this.m_bias = C * h * k * this.m_gamma;
invMass += this.m_gamma;
this.m_mass = invMass !== 0 ? 1 / invMass : 0;
}
else {
this.m_gamma = 0;
this.m_bias = 0;
}
if (data.step.warmStarting) {
// Scale the impulse to support a variable time step.
this.m_impulse *= data.step.dtRatio;
// b2Vec2 P = m_impulse * m_u;
var P = b2Vec2.MulSV(this.m_impulse, this.m_u, b2DistanceJoint.InitVelocityConstraints_s_P);
// vA -= m_invMassA * P;
vA.SelfMulSub(this.m_invMassA, P);
// wA -= m_invIA * b2Cross(m_rA, P);
wA -= this.m_invIA * b2Vec2.CrossVV(this.m_rA, P);
// vB += m_invMassB * P;
vB.SelfMulAdd(this.m_invMassB, P);
// wB += m_invIB * b2Cross(m_rB, P);
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, P);
}
else {
this.m_impulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2DistanceJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// b2Vec2 vpA = vA + b2Cross(wA, m_rA);
var vpA = b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2DistanceJoint.SolveVelocityConstraints_s_vpA);
// b2Vec2 vpB = vB + b2Cross(wB, m_rB);
var vpB = b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2DistanceJoint.SolveVelocityConstraints_s_vpB);
// float32 Cdot = b2Dot(m_u, vpB - vpA);
var Cdot = b2Vec2.DotVV(this.m_u, b2Vec2.SubVV(vpB, vpA, b2Vec2.s_t0));
var impulse = (-this.m_mass * (Cdot + this.m_bias + this.m_gamma * this.m_impulse));
this.m_impulse += impulse;
// b2Vec2 P = impulse * m_u;
var P = b2Vec2.MulSV(impulse, this.m_u, b2DistanceJoint.SolveVelocityConstraints_s_P);
// vA -= m_invMassA * P;
vA.SelfMulSub(this.m_invMassA, P);
// wA -= m_invIA * b2Cross(m_rA, P);
wA -= this.m_invIA * b2Vec2.CrossVV(this.m_rA, P);
// vB += m_invMassB * P;
vB.SelfMulAdd(this.m_invMassB, P);
// wB += m_invIB * b2Cross(m_rB, P);
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, P);
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2DistanceJoint.prototype.SolvePositionConstraints = function (data) {
if (this.m_frequencyHz > 0) {
// There is no position correction for soft distance constraints.
return true;
}
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
// const qA: b2Rot = new b2Rot(aA), qB: b2Rot = new b2Rot(aB);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA); // use m_rA
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB); // use m_rB
// b2Vec2 u = cB + rB - cA - rA;
var u = this.m_u; // use m_u
u.x = cB.x + rB.x - cA.x - rA.x;
u.y = cB.y + rB.y - cA.y - rA.y;
// float32 length = u.Normalize();
var length = this.m_u.Normalize();
// float32 C = length - m_length;
var C = length - this.m_length;
C = b2Clamp(C, (-b2_maxLinearCorrection), b2_maxLinearCorrection);
var impulse = (-this.m_mass * C);
// b2Vec2 P = impulse * u;
var P = b2Vec2.MulSV(impulse, u, b2DistanceJoint.SolvePositionConstraints_s_P);
// cA -= m_invMassA * P;
cA.SelfMulSub(this.m_invMassA, P);
// aA -= m_invIA * b2Cross(rA, P);
aA -= this.m_invIA * b2Vec2.CrossVV(rA, P);
// cB += m_invMassB * P;
cB.SelfMulAdd(this.m_invMassB, P);
// aB += m_invIB * b2Cross(rB, P);
aB += this.m_invIB * b2Vec2.CrossVV(rB, P);
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
return b2Abs(C) < b2_linearSlop;
};
b2DistanceJoint.InitVelocityConstraints_s_P = new b2Vec2();
b2DistanceJoint.SolveVelocityConstraints_s_vpA = new b2Vec2();
b2DistanceJoint.SolveVelocityConstraints_s_vpB = new b2Vec2();
b2DistanceJoint.SolveVelocityConstraints_s_P = new b2Vec2();
b2DistanceJoint.SolvePositionConstraints_s_P = new b2Vec2();
return b2DistanceJoint;
}(b2Joint));
var b2AreaJointDef = /** @class */ (function (_super) {
__extends(b2AreaJointDef, _super);
function b2AreaJointDef() {
var _this = _super.call(this, exports.b2JointType.e_areaJoint) || this;
_this.bodies = [];
_this.frequencyHz = 0;
_this.dampingRatio = 0;
return _this;
}
b2AreaJointDef.prototype.AddBody = function (body) {
this.bodies.push(body);
if (this.bodies.length === 1) {
this.bodyA = body;
}
else if (this.bodies.length === 2) {
this.bodyB = body;
}
};
return b2AreaJointDef;
}(b2JointDef));
var b2AreaJoint = /** @class */ (function (_super) {
__extends(b2AreaJoint, _super);
function b2AreaJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_frequencyHz = 0;
_this.m_dampingRatio = 0;
// Solver shared
_this.m_impulse = 0;
_this.m_targetArea = 0;
// DEBUG: b2Assert(def.bodies.length >= 3, "You cannot create an area joint with less than three bodies.");
_this.m_bodies = def.bodies;
_this.m_frequencyHz = b2Maybe(def.frequencyHz, 0);
_this.m_dampingRatio = b2Maybe(def.dampingRatio, 0);
_this.m_targetLengths = b2MakeNumberArray(def.bodies.length);
_this.m_normals = b2Vec2.MakeArray(def.bodies.length);
_this.m_joints = []; // b2MakeNullArray(def.bodies.length);
_this.m_deltas = b2Vec2.MakeArray(def.bodies.length);
_this.m_delta = new b2Vec2();
var djd = new b2DistanceJointDef();
djd.frequencyHz = _this.m_frequencyHz;
djd.dampingRatio = _this.m_dampingRatio;
_this.m_targetArea = 0;
for (var i = 0; i < _this.m_bodies.length; ++i) {
var body = _this.m_bodies[i];
var next = _this.m_bodies[(i + 1) % _this.m_bodies.length];
var body_c = body.GetWorldCenter();
var next_c = next.GetWorldCenter();
_this.m_targetLengths[i] = b2Vec2.DistanceVV(body_c, next_c);
_this.m_targetArea += b2Vec2.CrossVV(body_c, next_c);
djd.Initialize(body, next, body_c, next_c);
_this.m_joints[i] = body.GetWorld().CreateJoint(djd);
}
_this.m_targetArea *= 0.5;
return _this;
}
b2AreaJoint.prototype.GetAnchorA = function (out) {
return out;
};
b2AreaJoint.prototype.GetAnchorB = function (out) {
return out;
};
b2AreaJoint.prototype.GetReactionForce = function (inv_dt, out) {
return out;
};
b2AreaJoint.prototype.GetReactionTorque = function (inv_dt) {
return 0;
};
b2AreaJoint.prototype.SetFrequency = function (hz) {
this.m_frequencyHz = hz;
for (var i = 0; i < this.m_joints.length; ++i) {
this.m_joints[i].SetFrequency(hz);
}
};
b2AreaJoint.prototype.GetFrequency = function () {
return this.m_frequencyHz;
};
b2AreaJoint.prototype.SetDampingRatio = function (ratio) {
this.m_dampingRatio = ratio;
for (var i = 0; i < this.m_joints.length; ++i) {
this.m_joints[i].SetDampingRatio(ratio);
}
};
b2AreaJoint.prototype.GetDampingRatio = function () {
return this.m_dampingRatio;
};
b2AreaJoint.prototype.Dump = function (log) {
log("Area joint dumping is not supported.\n");
};
b2AreaJoint.prototype.InitVelocityConstraints = function (data) {
for (var i = 0; i < this.m_bodies.length; ++i) {
var prev = this.m_bodies[(i + this.m_bodies.length - 1) % this.m_bodies.length];
var next = this.m_bodies[(i + 1) % this.m_bodies.length];
var prev_c = data.positions[prev.m_islandIndex].c;
var next_c = data.positions[next.m_islandIndex].c;
var delta = this.m_deltas[i];
b2Vec2.SubVV(next_c, prev_c, delta);
}
if (data.step.warmStarting) {
this.m_impulse *= data.step.dtRatio;
for (var i = 0; i < this.m_bodies.length; ++i) {
var body = this.m_bodies[i];
var body_v = data.velocities[body.m_islandIndex].v;
var delta = this.m_deltas[i];
body_v.x += body.m_invMass * delta.y * 0.5 * this.m_impulse;
body_v.y += body.m_invMass * -delta.x * 0.5 * this.m_impulse;
}
}
else {
this.m_impulse = 0;
}
};
b2AreaJoint.prototype.SolveVelocityConstraints = function (data) {
var dotMassSum = 0;
var crossMassSum = 0;
for (var i = 0; i < this.m_bodies.length; ++i) {
var body = this.m_bodies[i];
var body_v = data.velocities[body.m_islandIndex].v;
var delta = this.m_deltas[i];
dotMassSum += delta.LengthSquared() / body.GetMass();
crossMassSum += b2Vec2.CrossVV(body_v, delta);
}
var lambda = -2 * crossMassSum / dotMassSum;
// lambda = b2Clamp(lambda, -b2_maxLinearCorrection, b2_maxLinearCorrection);
this.m_impulse += lambda;
for (var i = 0; i < this.m_bodies.length; ++i) {
var body = this.m_bodies[i];
var body_v = data.velocities[body.m_islandIndex].v;
var delta = this.m_deltas[i];
body_v.x += body.m_invMass * delta.y * 0.5 * lambda;
body_v.y += body.m_invMass * -delta.x * 0.5 * lambda;
}
};
b2AreaJoint.prototype.SolvePositionConstraints = function (data) {
var perimeter = 0;
var area = 0;
for (var i = 0; i < this.m_bodies.length; ++i) {
var body = this.m_bodies[i];
var next = this.m_bodies[(i + 1) % this.m_bodies.length];
var body_c = data.positions[body.m_islandIndex].c;
var next_c = data.positions[next.m_islandIndex].c;
var delta = b2Vec2.SubVV(next_c, body_c, this.m_delta);
var dist = delta.Length();
if (dist < b2_epsilon) {
dist = 1;
}
this.m_normals[i].x = delta.y / dist;
this.m_normals[i].y = -delta.x / dist;
perimeter += dist;
area += b2Vec2.CrossVV(body_c, next_c);
}
area *= 0.5;
var deltaArea = this.m_targetArea - area;
var toExtrude = 0.5 * deltaArea / perimeter;
var done = true;
for (var i = 0; i < this.m_bodies.length; ++i) {
var body = this.m_bodies[i];
var body_c = data.positions[body.m_islandIndex].c;
var next_i = (i + 1) % this.m_bodies.length;
var delta = b2Vec2.AddVV(this.m_normals[i], this.m_normals[next_i], this.m_delta);
delta.SelfMul(toExtrude);
var norm_sq = delta.LengthSquared();
if (norm_sq > b2Sq(b2_maxLinearCorrection)) {
delta.SelfMul(b2_maxLinearCorrection / b2Sqrt(norm_sq));
}
if (norm_sq > b2Sq(b2_linearSlop)) {
done = false;
}
body_c.x += delta.x;
body_c.y += delta.y;
}
return done;
};
return b2AreaJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2007 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Friction joint definition.
var b2FrictionJointDef = /** @class */ (function (_super) {
__extends(b2FrictionJointDef, _super);
function b2FrictionJointDef() {
var _this = _super.call(this, exports.b2JointType.e_frictionJoint) || this;
_this.localAnchorA = new b2Vec2();
_this.localAnchorB = new b2Vec2();
_this.maxForce = 0;
_this.maxTorque = 0;
return _this;
}
b2FrictionJointDef.prototype.Initialize = function (bA, bB, anchor) {
this.bodyA = bA;
this.bodyB = bB;
this.bodyA.GetLocalPoint(anchor, this.localAnchorA);
this.bodyB.GetLocalPoint(anchor, this.localAnchorB);
};
return b2FrictionJointDef;
}(b2JointDef));
var b2FrictionJoint = /** @class */ (function (_super) {
__extends(b2FrictionJoint, _super);
function b2FrictionJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
// Solver shared
_this.m_linearImpulse = new b2Vec2();
_this.m_angularImpulse = 0;
_this.m_maxForce = 0;
_this.m_maxTorque = 0;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_linearMass = new b2Mat22();
_this.m_angularMass = 0;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_K = new b2Mat22();
_this.m_localAnchorA.Copy(def.localAnchorA);
_this.m_localAnchorB.Copy(def.localAnchorB);
_this.m_linearImpulse.SetZero();
_this.m_maxForce = b2Maybe(def.maxForce, 0);
_this.m_maxTorque = b2Maybe(def.maxTorque, 0);
_this.m_linearMass.SetZero();
return _this;
}
b2FrictionJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
// const cA: b2Vec2 = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
// const cB: b2Vec2 = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// const qA: b2Rot = new b2Rot(aA), qB: b2Rot = new b2Rot(aB);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// Compute the effective mass matrix.
// m_rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// m_rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// J = [-I -r1_skew I r2_skew]
// [ 0 -1 0 1]
// r_skew = [-ry; rx]
// Matlab
// K = [ mA+r1y^2*iA+mB+r2y^2*iB, -r1y*iA*r1x-r2y*iB*r2x, -r1y*iA-r2y*iB]
// [ -r1y*iA*r1x-r2y*iB*r2x, mA+r1x^2*iA+mB+r2x^2*iB, r1x*iA+r2x*iB]
// [ -r1y*iA-r2y*iB, r1x*iA+r2x*iB, iA+iB]
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var K = this.m_K; // new b2Mat22();
K.ex.x = mA + mB + iA * rA.y * rA.y + iB * rB.y * rB.y;
K.ex.y = -iA * rA.x * rA.y - iB * rB.x * rB.y;
K.ey.x = K.ex.y;
K.ey.y = mA + mB + iA * rA.x * rA.x + iB * rB.x * rB.x;
K.GetInverse(this.m_linearMass);
this.m_angularMass = iA + iB;
if (this.m_angularMass > 0) {
this.m_angularMass = 1 / this.m_angularMass;
}
if (data.step.warmStarting) {
// Scale impulses to support a variable time step.
// m_linearImpulse *= data.step.dtRatio;
this.m_linearImpulse.SelfMul(data.step.dtRatio);
this.m_angularImpulse *= data.step.dtRatio;
// const P: b2Vec2(m_linearImpulse.x, m_linearImpulse.y);
var P = this.m_linearImpulse;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
// wA -= iA * (b2Cross(m_rA, P) + m_angularImpulse);
wA -= iA * (b2Vec2.CrossVV(this.m_rA, P) + this.m_angularImpulse);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
// wB += iB * (b2Cross(m_rB, P) + m_angularImpulse);
wB += iB * (b2Vec2.CrossVV(this.m_rB, P) + this.m_angularImpulse);
}
else {
this.m_linearImpulse.SetZero();
this.m_angularImpulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2FrictionJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var h = data.step.dt;
// Solve angular friction
{
var Cdot = wB - wA;
var impulse = (-this.m_angularMass * Cdot);
var oldImpulse = this.m_angularImpulse;
var maxImpulse = h * this.m_maxTorque;
this.m_angularImpulse = b2Clamp(this.m_angularImpulse + impulse, (-maxImpulse), maxImpulse);
impulse = this.m_angularImpulse - oldImpulse;
wA -= iA * impulse;
wB += iB * impulse;
}
// Solve linear friction
{
// b2Vec2 Cdot = vB + b2Cross(wB, m_rB) - vA - b2Cross(wA, m_rA);
var Cdot_v2 = b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2Vec2.s_t1), b2FrictionJoint.SolveVelocityConstraints_s_Cdot_v2);
// b2Vec2 impulse = -b2Mul(m_linearMass, Cdot);
var impulseV = b2Mat22.MulMV(this.m_linearMass, Cdot_v2, b2FrictionJoint.SolveVelocityConstraints_s_impulseV).SelfNeg();
// b2Vec2 oldImpulse = m_linearImpulse;
var oldImpulseV = b2FrictionJoint.SolveVelocityConstraints_s_oldImpulseV.Copy(this.m_linearImpulse);
// m_linearImpulse += impulse;
this.m_linearImpulse.SelfAdd(impulseV);
var maxImpulse = h * this.m_maxForce;
if (this.m_linearImpulse.LengthSquared() > maxImpulse * maxImpulse) {
this.m_linearImpulse.Normalize();
this.m_linearImpulse.SelfMul(maxImpulse);
}
// impulse = m_linearImpulse - oldImpulse;
b2Vec2.SubVV(this.m_linearImpulse, oldImpulseV, impulseV);
// vA -= mA * impulse;
vA.SelfMulSub(mA, impulseV);
// wA -= iA * b2Cross(m_rA, impulse);
wA -= iA * b2Vec2.CrossVV(this.m_rA, impulseV);
// vB += mB * impulse;
vB.SelfMulAdd(mB, impulseV);
// wB += iB * b2Cross(m_rB, impulse);
wB += iB * b2Vec2.CrossVV(this.m_rB, impulseV);
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2FrictionJoint.prototype.SolvePositionConstraints = function (data) {
return true;
};
b2FrictionJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2FrictionJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2FrictionJoint.prototype.GetReactionForce = function (inv_dt, out) {
out.x = inv_dt * this.m_linearImpulse.x;
out.y = inv_dt * this.m_linearImpulse.y;
return out;
};
b2FrictionJoint.prototype.GetReactionTorque = function (inv_dt) {
return inv_dt * this.m_angularImpulse;
};
b2FrictionJoint.prototype.GetLocalAnchorA = function () { return this.m_localAnchorA; };
b2FrictionJoint.prototype.GetLocalAnchorB = function () { return this.m_localAnchorB; };
b2FrictionJoint.prototype.SetMaxForce = function (force) {
this.m_maxForce = force;
};
b2FrictionJoint.prototype.GetMaxForce = function () {
return this.m_maxForce;
};
b2FrictionJoint.prototype.SetMaxTorque = function (torque) {
this.m_maxTorque = torque;
};
b2FrictionJoint.prototype.GetMaxTorque = function () {
return this.m_maxTorque;
};
b2FrictionJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2FrictionJointDef = new b2FrictionJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.maxForce = %.15f;\n", this.m_maxForce);
log(" jd.maxTorque = %.15f;\n", this.m_maxTorque);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2FrictionJoint.SolveVelocityConstraints_s_Cdot_v2 = new b2Vec2();
b2FrictionJoint.SolveVelocityConstraints_s_impulseV = new b2Vec2();
b2FrictionJoint.SolveVelocityConstraints_s_oldImpulseV = new b2Vec2();
return b2FrictionJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Gear joint definition. This definition requires two existing
/// revolute or prismatic joints (any combination will work).
var b2GearJointDef = /** @class */ (function (_super) {
__extends(b2GearJointDef, _super);
function b2GearJointDef() {
var _this = _super.call(this, exports.b2JointType.e_gearJoint) || this;
_this.ratio = 1;
return _this;
}
return b2GearJointDef;
}(b2JointDef));
var b2GearJoint = /** @class */ (function (_super) {
__extends(b2GearJoint, _super);
function b2GearJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_typeA = exports.b2JointType.e_unknownJoint;
_this.m_typeB = exports.b2JointType.e_unknownJoint;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_localAnchorC = new b2Vec2();
_this.m_localAnchorD = new b2Vec2();
_this.m_localAxisC = new b2Vec2();
_this.m_localAxisD = new b2Vec2();
_this.m_referenceAngleA = 0;
_this.m_referenceAngleB = 0;
_this.m_constant = 0;
_this.m_ratio = 0;
_this.m_impulse = 0;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_indexC = 0;
_this.m_indexD = 0;
_this.m_lcA = new b2Vec2();
_this.m_lcB = new b2Vec2();
_this.m_lcC = new b2Vec2();
_this.m_lcD = new b2Vec2();
_this.m_mA = 0;
_this.m_mB = 0;
_this.m_mC = 0;
_this.m_mD = 0;
_this.m_iA = 0;
_this.m_iB = 0;
_this.m_iC = 0;
_this.m_iD = 0;
_this.m_JvAC = new b2Vec2();
_this.m_JvBD = new b2Vec2();
_this.m_JwA = 0;
_this.m_JwB = 0;
_this.m_JwC = 0;
_this.m_JwD = 0;
_this.m_mass = 0;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_qC = new b2Rot();
_this.m_qD = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_lalcC = new b2Vec2();
_this.m_lalcD = new b2Vec2();
_this.m_joint1 = def.joint1;
_this.m_joint2 = def.joint2;
_this.m_typeA = _this.m_joint1.GetType();
_this.m_typeB = _this.m_joint2.GetType();
// DEBUG: b2Assert(this.m_typeA === b2JointType.e_revoluteJoint || this.m_typeA === b2JointType.e_prismaticJoint);
// DEBUG: b2Assert(this.m_typeB === b2JointType.e_revoluteJoint || this.m_typeB === b2JointType.e_prismaticJoint);
var coordinateA, coordinateB;
// TODO_ERIN there might be some problem with the joint edges in b2Joint.
_this.m_bodyC = _this.m_joint1.GetBodyA();
_this.m_bodyA = _this.m_joint1.GetBodyB();
// Get geometry of joint1
var xfA = _this.m_bodyA.m_xf;
var aA = _this.m_bodyA.m_sweep.a;
var xfC = _this.m_bodyC.m_xf;
var aC = _this.m_bodyC.m_sweep.a;
if (_this.m_typeA === exports.b2JointType.e_revoluteJoint) {
var revolute = def.joint1;
_this.m_localAnchorC.Copy(revolute.m_localAnchorA);
_this.m_localAnchorA.Copy(revolute.m_localAnchorB);
_this.m_referenceAngleA = revolute.m_referenceAngle;
_this.m_localAxisC.SetZero();
coordinateA = aA - aC - _this.m_referenceAngleA;
}
else {
var prismatic = def.joint1;
_this.m_localAnchorC.Copy(prismatic.m_localAnchorA);
_this.m_localAnchorA.Copy(prismatic.m_localAnchorB);
_this.m_referenceAngleA = prismatic.m_referenceAngle;
_this.m_localAxisC.Copy(prismatic.m_localXAxisA);
// b2Vec2 pC = m_localAnchorC;
var pC = _this.m_localAnchorC;
// b2Vec2 pA = b2MulT(xfC.q, b2Mul(xfA.q, m_localAnchorA) + (xfA.p - xfC.p));
var pA = b2Rot.MulTRV(xfC.q, b2Vec2.AddVV(b2Rot.MulRV(xfA.q, _this.m_localAnchorA, b2Vec2.s_t0), b2Vec2.SubVV(xfA.p, xfC.p, b2Vec2.s_t1), b2Vec2.s_t0), b2Vec2.s_t0); // pA uses s_t0
// coordinateA = b2Dot(pA - pC, m_localAxisC);
coordinateA = b2Vec2.DotVV(b2Vec2.SubVV(pA, pC, b2Vec2.s_t0), _this.m_localAxisC);
}
_this.m_bodyD = _this.m_joint2.GetBodyA();
_this.m_bodyB = _this.m_joint2.GetBodyB();
// Get geometry of joint2
var xfB = _this.m_bodyB.m_xf;
var aB = _this.m_bodyB.m_sweep.a;
var xfD = _this.m_bodyD.m_xf;
var aD = _this.m_bodyD.m_sweep.a;
if (_this.m_typeB === exports.b2JointType.e_revoluteJoint) {
var revolute = def.joint2;
_this.m_localAnchorD.Copy(revolute.m_localAnchorA);
_this.m_localAnchorB.Copy(revolute.m_localAnchorB);
_this.m_referenceAngleB = revolute.m_referenceAngle;
_this.m_localAxisD.SetZero();
coordinateB = aB - aD - _this.m_referenceAngleB;
}
else {
var prismatic = def.joint2;
_this.m_localAnchorD.Copy(prismatic.m_localAnchorA);
_this.m_localAnchorB.Copy(prismatic.m_localAnchorB);
_this.m_referenceAngleB = prismatic.m_referenceAngle;
_this.m_localAxisD.Copy(prismatic.m_localXAxisA);
// b2Vec2 pD = m_localAnchorD;
var pD = _this.m_localAnchorD;
// b2Vec2 pB = b2MulT(xfD.q, b2Mul(xfB.q, m_localAnchorB) + (xfB.p - xfD.p));
var pB = b2Rot.MulTRV(xfD.q, b2Vec2.AddVV(b2Rot.MulRV(xfB.q, _this.m_localAnchorB, b2Vec2.s_t0), b2Vec2.SubVV(xfB.p, xfD.p, b2Vec2.s_t1), b2Vec2.s_t0), b2Vec2.s_t0); // pB uses s_t0
// coordinateB = b2Dot(pB - pD, m_localAxisD);
coordinateB = b2Vec2.DotVV(b2Vec2.SubVV(pB, pD, b2Vec2.s_t0), _this.m_localAxisD);
}
_this.m_ratio = b2Maybe(def.ratio, 1);
_this.m_constant = coordinateA + _this.m_ratio * coordinateB;
_this.m_impulse = 0;
return _this;
}
b2GearJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_indexC = this.m_bodyC.m_islandIndex;
this.m_indexD = this.m_bodyD.m_islandIndex;
this.m_lcA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_lcB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_lcC.Copy(this.m_bodyC.m_sweep.localCenter);
this.m_lcD.Copy(this.m_bodyD.m_sweep.localCenter);
this.m_mA = this.m_bodyA.m_invMass;
this.m_mB = this.m_bodyB.m_invMass;
this.m_mC = this.m_bodyC.m_invMass;
this.m_mD = this.m_bodyD.m_invMass;
this.m_iA = this.m_bodyA.m_invI;
this.m_iB = this.m_bodyB.m_invI;
this.m_iC = this.m_bodyC.m_invI;
this.m_iD = this.m_bodyD.m_invI;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var aC = data.positions[this.m_indexC].a;
var vC = data.velocities[this.m_indexC].v;
var wC = data.velocities[this.m_indexC].w;
var aD = data.positions[this.m_indexD].a;
var vD = data.velocities[this.m_indexD].v;
var wD = data.velocities[this.m_indexD].w;
// b2Rot qA(aA), qB(aB), qC(aC), qD(aD);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB), qC = this.m_qC.SetAngle(aC), qD = this.m_qD.SetAngle(aD);
this.m_mass = 0;
if (this.m_typeA === exports.b2JointType.e_revoluteJoint) {
this.m_JvAC.SetZero();
this.m_JwA = 1;
this.m_JwC = 1;
this.m_mass += this.m_iA + this.m_iC;
}
else {
// b2Vec2 u = b2Mul(qC, m_localAxisC);
var u = b2Rot.MulRV(qC, this.m_localAxisC, b2GearJoint.InitVelocityConstraints_s_u);
// b2Vec2 rC = b2Mul(qC, m_localAnchorC - m_lcC);
b2Vec2.SubVV(this.m_localAnchorC, this.m_lcC, this.m_lalcC);
var rC = b2Rot.MulRV(qC, this.m_lalcC, b2GearJoint.InitVelocityConstraints_s_rC);
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_lcA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_lcA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, b2GearJoint.InitVelocityConstraints_s_rA);
// m_JvAC = u;
this.m_JvAC.Copy(u);
// m_JwC = b2Cross(rC, u);
this.m_JwC = b2Vec2.CrossVV(rC, u);
// m_JwA = b2Cross(rA, u);
this.m_JwA = b2Vec2.CrossVV(rA, u);
this.m_mass += this.m_mC + this.m_mA + this.m_iC * this.m_JwC * this.m_JwC + this.m_iA * this.m_JwA * this.m_JwA;
}
if (this.m_typeB === exports.b2JointType.e_revoluteJoint) {
this.m_JvBD.SetZero();
this.m_JwB = this.m_ratio;
this.m_JwD = this.m_ratio;
this.m_mass += this.m_ratio * this.m_ratio * (this.m_iB + this.m_iD);
}
else {
// b2Vec2 u = b2Mul(qD, m_localAxisD);
var u = b2Rot.MulRV(qD, this.m_localAxisD, b2GearJoint.InitVelocityConstraints_s_u);
// b2Vec2 rD = b2Mul(qD, m_localAnchorD - m_lcD);
b2Vec2.SubVV(this.m_localAnchorD, this.m_lcD, this.m_lalcD);
var rD = b2Rot.MulRV(qD, this.m_lalcD, b2GearJoint.InitVelocityConstraints_s_rD);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_lcB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_lcB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, b2GearJoint.InitVelocityConstraints_s_rB);
// m_JvBD = m_ratio * u;
b2Vec2.MulSV(this.m_ratio, u, this.m_JvBD);
// m_JwD = m_ratio * b2Cross(rD, u);
this.m_JwD = this.m_ratio * b2Vec2.CrossVV(rD, u);
// m_JwB = m_ratio * b2Cross(rB, u);
this.m_JwB = this.m_ratio * b2Vec2.CrossVV(rB, u);
this.m_mass += this.m_ratio * this.m_ratio * (this.m_mD + this.m_mB) + this.m_iD * this.m_JwD * this.m_JwD + this.m_iB * this.m_JwB * this.m_JwB;
}
// Compute effective mass.
this.m_mass = this.m_mass > 0 ? 1 / this.m_mass : 0;
if (data.step.warmStarting) {
// vA += (m_mA * m_impulse) * m_JvAC;
vA.SelfMulAdd(this.m_mA * this.m_impulse, this.m_JvAC);
wA += this.m_iA * this.m_impulse * this.m_JwA;
// vB += (m_mB * m_impulse) * m_JvBD;
vB.SelfMulAdd(this.m_mB * this.m_impulse, this.m_JvBD);
wB += this.m_iB * this.m_impulse * this.m_JwB;
// vC -= (m_mC * m_impulse) * m_JvAC;
vC.SelfMulSub(this.m_mC * this.m_impulse, this.m_JvAC);
wC -= this.m_iC * this.m_impulse * this.m_JwC;
// vD -= (m_mD * m_impulse) * m_JvBD;
vD.SelfMulSub(this.m_mD * this.m_impulse, this.m_JvBD);
wD -= this.m_iD * this.m_impulse * this.m_JwD;
}
else {
this.m_impulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
// data.velocities[this.m_indexC].v = vC;
data.velocities[this.m_indexC].w = wC;
// data.velocities[this.m_indexD].v = vD;
data.velocities[this.m_indexD].w = wD;
};
b2GearJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var vC = data.velocities[this.m_indexC].v;
var wC = data.velocities[this.m_indexC].w;
var vD = data.velocities[this.m_indexD].v;
var wD = data.velocities[this.m_indexD].w;
// float32 Cdot = b2Dot(m_JvAC, vA - vC) + b2Dot(m_JvBD, vB - vD);
var Cdot = b2Vec2.DotVV(this.m_JvAC, b2Vec2.SubVV(vA, vC, b2Vec2.s_t0)) +
b2Vec2.DotVV(this.m_JvBD, b2Vec2.SubVV(vB, vD, b2Vec2.s_t0));
Cdot += (this.m_JwA * wA - this.m_JwC * wC) + (this.m_JwB * wB - this.m_JwD * wD);
var impulse = -this.m_mass * Cdot;
this.m_impulse += impulse;
// vA += (m_mA * impulse) * m_JvAC;
vA.SelfMulAdd((this.m_mA * impulse), this.m_JvAC);
wA += this.m_iA * impulse * this.m_JwA;
// vB += (m_mB * impulse) * m_JvBD;
vB.SelfMulAdd((this.m_mB * impulse), this.m_JvBD);
wB += this.m_iB * impulse * this.m_JwB;
// vC -= (m_mC * impulse) * m_JvAC;
vC.SelfMulSub((this.m_mC * impulse), this.m_JvAC);
wC -= this.m_iC * impulse * this.m_JwC;
// vD -= (m_mD * impulse) * m_JvBD;
vD.SelfMulSub((this.m_mD * impulse), this.m_JvBD);
wD -= this.m_iD * impulse * this.m_JwD;
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
// data.velocities[this.m_indexC].v = vC;
data.velocities[this.m_indexC].w = wC;
// data.velocities[this.m_indexD].v = vD;
data.velocities[this.m_indexD].w = wD;
};
b2GearJoint.prototype.SolvePositionConstraints = function (data) {
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var cC = data.positions[this.m_indexC].c;
var aC = data.positions[this.m_indexC].a;
var cD = data.positions[this.m_indexD].c;
var aD = data.positions[this.m_indexD].a;
// b2Rot qA(aA), qB(aB), qC(aC), qD(aD);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB), qC = this.m_qC.SetAngle(aC), qD = this.m_qD.SetAngle(aD);
var linearError = 0;
var coordinateA, coordinateB;
var JvAC = this.m_JvAC, JvBD = this.m_JvBD;
var JwA, JwB, JwC, JwD;
var mass = 0;
if (this.m_typeA === exports.b2JointType.e_revoluteJoint) {
JvAC.SetZero();
JwA = 1;
JwC = 1;
mass += this.m_iA + this.m_iC;
coordinateA = aA - aC - this.m_referenceAngleA;
}
else {
// b2Vec2 u = b2Mul(qC, m_localAxisC);
var u = b2Rot.MulRV(qC, this.m_localAxisC, b2GearJoint.SolvePositionConstraints_s_u);
// b2Vec2 rC = b2Mul(qC, m_localAnchorC - m_lcC);
var rC = b2Rot.MulRV(qC, this.m_lalcC, b2GearJoint.SolvePositionConstraints_s_rC);
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_lcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, b2GearJoint.SolvePositionConstraints_s_rA);
// JvAC = u;
JvAC.Copy(u);
// JwC = b2Cross(rC, u);
JwC = b2Vec2.CrossVV(rC, u);
// JwA = b2Cross(rA, u);
JwA = b2Vec2.CrossVV(rA, u);
mass += this.m_mC + this.m_mA + this.m_iC * JwC * JwC + this.m_iA * JwA * JwA;
// b2Vec2 pC = m_localAnchorC - m_lcC;
var pC = this.m_lalcC;
// b2Vec2 pA = b2MulT(qC, rA + (cA - cC));
var pA = b2Rot.MulTRV(qC, b2Vec2.AddVV(rA, b2Vec2.SubVV(cA, cC, b2Vec2.s_t0), b2Vec2.s_t0), b2Vec2.s_t0); // pA uses s_t0
// coordinateA = b2Dot(pA - pC, m_localAxisC);
coordinateA = b2Vec2.DotVV(b2Vec2.SubVV(pA, pC, b2Vec2.s_t0), this.m_localAxisC);
}
if (this.m_typeB === exports.b2JointType.e_revoluteJoint) {
JvBD.SetZero();
JwB = this.m_ratio;
JwD = this.m_ratio;
mass += this.m_ratio * this.m_ratio * (this.m_iB + this.m_iD);
coordinateB = aB - aD - this.m_referenceAngleB;
}
else {
// b2Vec2 u = b2Mul(qD, m_localAxisD);
var u = b2Rot.MulRV(qD, this.m_localAxisD, b2GearJoint.SolvePositionConstraints_s_u);
// b2Vec2 rD = b2Mul(qD, m_localAnchorD - m_lcD);
var rD = b2Rot.MulRV(qD, this.m_lalcD, b2GearJoint.SolvePositionConstraints_s_rD);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_lcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, b2GearJoint.SolvePositionConstraints_s_rB);
// JvBD = m_ratio * u;
b2Vec2.MulSV(this.m_ratio, u, JvBD);
// JwD = m_ratio * b2Cross(rD, u);
JwD = this.m_ratio * b2Vec2.CrossVV(rD, u);
// JwB = m_ratio * b2Cross(rB, u);
JwB = this.m_ratio * b2Vec2.CrossVV(rB, u);
mass += this.m_ratio * this.m_ratio * (this.m_mD + this.m_mB) + this.m_iD * JwD * JwD + this.m_iB * JwB * JwB;
// b2Vec2 pD = m_localAnchorD - m_lcD;
var pD = this.m_lalcD;
// b2Vec2 pB = b2MulT(qD, rB + (cB - cD));
var pB = b2Rot.MulTRV(qD, b2Vec2.AddVV(rB, b2Vec2.SubVV(cB, cD, b2Vec2.s_t0), b2Vec2.s_t0), b2Vec2.s_t0); // pB uses s_t0
// coordinateB = b2Dot(pB - pD, m_localAxisD);
coordinateB = b2Vec2.DotVV(b2Vec2.SubVV(pB, pD, b2Vec2.s_t0), this.m_localAxisD);
}
var C = (coordinateA + this.m_ratio * coordinateB) - this.m_constant;
var impulse = 0;
if (mass > 0) {
impulse = -C / mass;
}
// cA += m_mA * impulse * JvAC;
cA.SelfMulAdd(this.m_mA * impulse, JvAC);
aA += this.m_iA * impulse * JwA;
// cB += m_mB * impulse * JvBD;
cB.SelfMulAdd(this.m_mB * impulse, JvBD);
aB += this.m_iB * impulse * JwB;
// cC -= m_mC * impulse * JvAC;
cC.SelfMulSub(this.m_mC * impulse, JvAC);
aC -= this.m_iC * impulse * JwC;
// cD -= m_mD * impulse * JvBD;
cD.SelfMulSub(this.m_mD * impulse, JvBD);
aD -= this.m_iD * impulse * JwD;
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
// data.positions[this.m_indexC].c = cC;
data.positions[this.m_indexC].a = aC;
// data.positions[this.m_indexD].c = cD;
data.positions[this.m_indexD].a = aD;
// TODO_ERIN not implemented
return linearError < b2_linearSlop;
};
b2GearJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2GearJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2GearJoint.prototype.GetReactionForce = function (inv_dt, out) {
// b2Vec2 P = m_impulse * m_JvAC;
// return inv_dt * P;
return b2Vec2.MulSV(inv_dt * this.m_impulse, this.m_JvAC, out);
};
b2GearJoint.prototype.GetReactionTorque = function (inv_dt) {
// float32 L = m_impulse * m_JwA;
// return inv_dt * L;
return inv_dt * this.m_impulse * this.m_JwA;
};
b2GearJoint.prototype.GetJoint1 = function () { return this.m_joint1; };
b2GearJoint.prototype.GetJoint2 = function () { return this.m_joint2; };
b2GearJoint.prototype.GetRatio = function () {
return this.m_ratio;
};
b2GearJoint.prototype.SetRatio = function (ratio) {
// DEBUG: b2Assert(b2IsValid(ratio));
this.m_ratio = ratio;
};
b2GearJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
var index1 = this.m_joint1.m_index;
var index2 = this.m_joint2.m_index;
log(" const jd: b2GearJointDef = new b2GearJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.joint1 = joints[%d];\n", index1);
log(" jd.joint2 = joints[%d];\n", index2);
log(" jd.ratio = %.15f;\n", this.m_ratio);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2GearJoint.InitVelocityConstraints_s_u = new b2Vec2();
b2GearJoint.InitVelocityConstraints_s_rA = new b2Vec2();
b2GearJoint.InitVelocityConstraints_s_rB = new b2Vec2();
b2GearJoint.InitVelocityConstraints_s_rC = new b2Vec2();
b2GearJoint.InitVelocityConstraints_s_rD = new b2Vec2();
b2GearJoint.SolvePositionConstraints_s_u = new b2Vec2();
b2GearJoint.SolvePositionConstraints_s_rA = new b2Vec2();
b2GearJoint.SolvePositionConstraints_s_rB = new b2Vec2();
b2GearJoint.SolvePositionConstraints_s_rC = new b2Vec2();
b2GearJoint.SolvePositionConstraints_s_rD = new b2Vec2();
return b2GearJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2012 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2MotorJointDef = /** @class */ (function (_super) {
__extends(b2MotorJointDef, _super);
function b2MotorJointDef() {
var _this = _super.call(this, exports.b2JointType.e_motorJoint) || this;
_this.linearOffset = new b2Vec2(0, 0);
_this.angularOffset = 0;
_this.maxForce = 1;
_this.maxTorque = 1;
_this.correctionFactor = 0.3;
return _this;
}
b2MotorJointDef.prototype.Initialize = function (bA, bB) {
this.bodyA = bA;
this.bodyB = bB;
// b2Vec2 xB = bodyB->GetPosition();
// linearOffset = bodyA->GetLocalPoint(xB);
this.bodyA.GetLocalPoint(this.bodyB.GetPosition(), this.linearOffset);
var angleA = this.bodyA.GetAngle();
var angleB = this.bodyB.GetAngle();
this.angularOffset = angleB - angleA;
};
return b2MotorJointDef;
}(b2JointDef));
var b2MotorJoint = /** @class */ (function (_super) {
__extends(b2MotorJoint, _super);
function b2MotorJoint(def) {
var _this = _super.call(this, def) || this;
// Solver shared
_this.m_linearOffset = new b2Vec2();
_this.m_angularOffset = 0;
_this.m_linearImpulse = new b2Vec2();
_this.m_angularImpulse = 0;
_this.m_maxForce = 0;
_this.m_maxTorque = 0;
_this.m_correctionFactor = 0.3;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_linearError = new b2Vec2();
_this.m_angularError = 0;
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_linearMass = new b2Mat22();
_this.m_angularMass = 0;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_K = new b2Mat22();
_this.m_linearOffset.Copy(b2Maybe(def.linearOffset, b2Vec2.ZERO));
_this.m_linearImpulse.SetZero();
_this.m_maxForce = b2Maybe(def.maxForce, 0);
_this.m_maxTorque = b2Maybe(def.maxTorque, 0);
_this.m_correctionFactor = b2Maybe(def.correctionFactor, 0.3);
return _this;
}
b2MotorJoint.prototype.GetAnchorA = function (out) {
var pos = this.m_bodyA.GetPosition();
out.x = pos.x;
out.y = pos.y;
return out;
};
b2MotorJoint.prototype.GetAnchorB = function (out) {
var pos = this.m_bodyB.GetPosition();
out.x = pos.x;
out.y = pos.y;
return out;
};
b2MotorJoint.prototype.GetReactionForce = function (inv_dt, out) {
// return inv_dt * m_linearImpulse;
return b2Vec2.MulSV(inv_dt, this.m_linearImpulse, out);
};
b2MotorJoint.prototype.GetReactionTorque = function (inv_dt) {
return inv_dt * this.m_angularImpulse;
};
b2MotorJoint.prototype.SetLinearOffset = function (linearOffset) {
if (!b2Vec2.IsEqualToV(linearOffset, this.m_linearOffset)) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_linearOffset.Copy(linearOffset);
}
};
b2MotorJoint.prototype.GetLinearOffset = function () {
return this.m_linearOffset;
};
b2MotorJoint.prototype.SetAngularOffset = function (angularOffset) {
if (angularOffset !== this.m_angularOffset) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_angularOffset = angularOffset;
}
};
b2MotorJoint.prototype.GetAngularOffset = function () {
return this.m_angularOffset;
};
b2MotorJoint.prototype.SetMaxForce = function (force) {
// DEBUG: b2Assert(b2IsValid(force) && force >= 0);
this.m_maxForce = force;
};
b2MotorJoint.prototype.GetMaxForce = function () {
return this.m_maxForce;
};
b2MotorJoint.prototype.SetMaxTorque = function (torque) {
// DEBUG: b2Assert(b2IsValid(torque) && torque >= 0);
this.m_maxTorque = torque;
};
b2MotorJoint.prototype.GetMaxTorque = function () {
return this.m_maxTorque;
};
b2MotorJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// Compute the effective mass matrix.
// this.m_rA = b2Mul(qA, m_linearOffset - this.m_localCenterA);
var rA = b2Rot.MulRV(qA, b2Vec2.SubVV(this.m_linearOffset, this.m_localCenterA, b2Vec2.s_t0), this.m_rA);
// this.m_rB = b2Mul(qB, -this.m_localCenterB);
var rB = b2Rot.MulRV(qB, b2Vec2.NegV(this.m_localCenterB, b2Vec2.s_t0), this.m_rB);
// J = [-I -r1_skew I r2_skew]
// r_skew = [-ry; rx]
// Matlab
// K = [ mA+r1y^2*iA+mB+r2y^2*iB, -r1y*iA*r1x-r2y*iB*r2x, -r1y*iA-r2y*iB]
// [ -r1y*iA*r1x-r2y*iB*r2x, mA+r1x^2*iA+mB+r2x^2*iB, r1x*iA+r2x*iB]
// [ -r1y*iA-r2y*iB, r1x*iA+r2x*iB, iA+iB]
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
// Upper 2 by 2 of K for point to point
var K = this.m_K;
K.ex.x = mA + mB + iA * rA.y * rA.y + iB * rB.y * rB.y;
K.ex.y = -iA * rA.x * rA.y - iB * rB.x * rB.y;
K.ey.x = K.ex.y;
K.ey.y = mA + mB + iA * rA.x * rA.x + iB * rB.x * rB.x;
// this.m_linearMass = K.GetInverse();
K.GetInverse(this.m_linearMass);
this.m_angularMass = iA + iB;
if (this.m_angularMass > 0) {
this.m_angularMass = 1 / this.m_angularMass;
}
// this.m_linearError = cB + rB - cA - rA;
b2Vec2.SubVV(b2Vec2.AddVV(cB, rB, b2Vec2.s_t0), b2Vec2.AddVV(cA, rA, b2Vec2.s_t1), this.m_linearError);
this.m_angularError = aB - aA - this.m_angularOffset;
if (data.step.warmStarting) {
// Scale impulses to support a variable time step.
// this.m_linearImpulse *= data.step.dtRatio;
this.m_linearImpulse.SelfMul(data.step.dtRatio);
this.m_angularImpulse *= data.step.dtRatio;
// b2Vec2 P(this.m_linearImpulse.x, this.m_linearImpulse.y);
var P = this.m_linearImpulse;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * (b2Vec2.CrossVV(rA, P) + this.m_angularImpulse);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * (b2Vec2.CrossVV(rB, P) + this.m_angularImpulse);
}
else {
this.m_linearImpulse.SetZero();
this.m_angularImpulse = 0;
}
// data.velocities[this.m_indexA].v = vA; // vA is a reference
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB; // vB is a reference
data.velocities[this.m_indexB].w = wB;
};
b2MotorJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var h = data.step.dt;
var inv_h = data.step.inv_dt;
// Solve angular friction
{
var Cdot = wB - wA + inv_h * this.m_correctionFactor * this.m_angularError;
var impulse = -this.m_angularMass * Cdot;
var oldImpulse = this.m_angularImpulse;
var maxImpulse = h * this.m_maxTorque;
this.m_angularImpulse = b2Clamp(this.m_angularImpulse + impulse, -maxImpulse, maxImpulse);
impulse = this.m_angularImpulse - oldImpulse;
wA -= iA * impulse;
wB += iB * impulse;
}
// Solve linear friction
{
var rA = this.m_rA;
var rB = this.m_rB;
// b2Vec2 Cdot = vB + b2Vec2.CrossSV(wB, rB) - vA - b2Vec2.CrossSV(wA, rA) + inv_h * this.m_correctionFactor * this.m_linearError;
var Cdot_v2 = b2Vec2.AddVV(b2Vec2.SubVV(b2Vec2.AddVV(vB, b2Vec2.CrossSV(wB, rB, b2Vec2.s_t0), b2Vec2.s_t0), b2Vec2.AddVV(vA, b2Vec2.CrossSV(wA, rA, b2Vec2.s_t1), b2Vec2.s_t1), b2Vec2.s_t2), b2Vec2.MulSV(inv_h * this.m_correctionFactor, this.m_linearError, b2Vec2.s_t3), b2MotorJoint.SolveVelocityConstraints_s_Cdot_v2);
// b2Vec2 impulse = -b2Mul(this.m_linearMass, Cdot);
var impulse_v2 = b2Mat22.MulMV(this.m_linearMass, Cdot_v2, b2MotorJoint.SolveVelocityConstraints_s_impulse_v2).SelfNeg();
// b2Vec2 oldImpulse = this.m_linearImpulse;
var oldImpulse_v2 = b2MotorJoint.SolveVelocityConstraints_s_oldImpulse_v2.Copy(this.m_linearImpulse);
// this.m_linearImpulse += impulse;
this.m_linearImpulse.SelfAdd(impulse_v2);
var maxImpulse = h * this.m_maxForce;
if (this.m_linearImpulse.LengthSquared() > maxImpulse * maxImpulse) {
this.m_linearImpulse.Normalize();
// this.m_linearImpulse *= maxImpulse;
this.m_linearImpulse.SelfMul(maxImpulse);
}
// impulse = this.m_linearImpulse - oldImpulse;
b2Vec2.SubVV(this.m_linearImpulse, oldImpulse_v2, impulse_v2);
// vA -= mA * impulse;
vA.SelfMulSub(mA, impulse_v2);
// wA -= iA * b2Vec2.CrossVV(rA, impulse);
wA -= iA * b2Vec2.CrossVV(rA, impulse_v2);
// vB += mB * impulse;
vB.SelfMulAdd(mB, impulse_v2);
// wB += iB * b2Vec2.CrossVV(rB, impulse);
wB += iB * b2Vec2.CrossVV(rB, impulse_v2);
}
// data.velocities[this.m_indexA].v = vA; // vA is a reference
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB; // vB is a reference
data.velocities[this.m_indexB].w = wB;
};
b2MotorJoint.prototype.SolvePositionConstraints = function (data) {
return true;
};
b2MotorJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2MotorJointDef = new b2MotorJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.linearOffset.Set(%.15f, %.15f);\n", this.m_linearOffset.x, this.m_linearOffset.y);
log(" jd.angularOffset = %.15f;\n", this.m_angularOffset);
log(" jd.maxForce = %.15f;\n", this.m_maxForce);
log(" jd.maxTorque = %.15f;\n", this.m_maxTorque);
log(" jd.correctionFactor = %.15f;\n", this.m_correctionFactor);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2MotorJoint.SolveVelocityConstraints_s_Cdot_v2 = new b2Vec2();
b2MotorJoint.SolveVelocityConstraints_s_impulse_v2 = new b2Vec2();
b2MotorJoint.SolveVelocityConstraints_s_oldImpulse_v2 = new b2Vec2();
return b2MotorJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2007 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Mouse joint definition. This requires a world target point,
/// tuning parameters, and the time step.
var b2MouseJointDef = /** @class */ (function (_super) {
__extends(b2MouseJointDef, _super);
function b2MouseJointDef() {
var _this = _super.call(this, exports.b2JointType.e_mouseJoint) || this;
_this.target = new b2Vec2();
_this.maxForce = 0;
_this.frequencyHz = 5;
_this.dampingRatio = 0.7;
return _this;
}
return b2MouseJointDef;
}(b2JointDef));
var b2MouseJoint = /** @class */ (function (_super) {
__extends(b2MouseJoint, _super);
function b2MouseJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_localAnchorB = new b2Vec2();
_this.m_targetA = new b2Vec2();
_this.m_frequencyHz = 0;
_this.m_dampingRatio = 0;
_this.m_beta = 0;
// Solver shared
_this.m_impulse = new b2Vec2();
_this.m_maxForce = 0;
_this.m_gamma = 0;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_rB = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassB = 0;
_this.m_invIB = 0;
_this.m_mass = new b2Mat22();
_this.m_C = new b2Vec2();
_this.m_qB = new b2Rot();
_this.m_lalcB = new b2Vec2();
_this.m_K = new b2Mat22();
_this.m_targetA.Copy(b2Maybe(def.target, b2Vec2.ZERO));
// DEBUG: b2Assert(this.m_targetA.IsValid());
b2Transform.MulTXV(_this.m_bodyB.GetTransform(), _this.m_targetA, _this.m_localAnchorB);
_this.m_maxForce = b2Maybe(def.maxForce, 0);
// DEBUG: b2Assert(b2IsValid(this.m_maxForce) && this.m_maxForce >= 0);
_this.m_impulse.SetZero();
_this.m_frequencyHz = b2Maybe(def.frequencyHz, 0);
// DEBUG: b2Assert(b2IsValid(this.m_frequencyHz) && this.m_frequencyHz >= 0);
_this.m_dampingRatio = b2Maybe(def.dampingRatio, 0);
// DEBUG: b2Assert(b2IsValid(this.m_dampingRatio) && this.m_dampingRatio >= 0);
_this.m_beta = 0;
_this.m_gamma = 0;
return _this;
}
b2MouseJoint.prototype.SetTarget = function (target) {
if (!this.m_bodyB.IsAwake()) {
this.m_bodyB.SetAwake(true);
}
this.m_targetA.Copy(target);
};
b2MouseJoint.prototype.GetTarget = function () {
return this.m_targetA;
};
b2MouseJoint.prototype.SetMaxForce = function (maxForce) {
this.m_maxForce = maxForce;
};
b2MouseJoint.prototype.GetMaxForce = function () {
return this.m_maxForce;
};
b2MouseJoint.prototype.SetFrequency = function (hz) {
this.m_frequencyHz = hz;
};
b2MouseJoint.prototype.GetFrequency = function () {
return this.m_frequencyHz;
};
b2MouseJoint.prototype.SetDampingRatio = function (ratio) {
this.m_dampingRatio = ratio;
};
b2MouseJoint.prototype.GetDampingRatio = function () {
return this.m_dampingRatio;
};
b2MouseJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIB = this.m_bodyB.m_invI;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var qB = this.m_qB.SetAngle(aB);
var mass = this.m_bodyB.GetMass();
// Frequency
var omega = 2 * b2_pi * this.m_frequencyHz;
// Damping coefficient
var d = 2 * mass * this.m_dampingRatio * omega;
// Spring stiffness
var k = mass * (omega * omega);
// magic formulas
// gamma has units of inverse mass.
// beta has units of inverse time.
var h = data.step.dt;
// DEBUG: b2Assert(d + h * k > b2_epsilon);
this.m_gamma = h * (d + h * k);
if (this.m_gamma !== 0) {
this.m_gamma = 1 / this.m_gamma;
}
this.m_beta = h * k * this.m_gamma;
// Compute the effective mass matrix.
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// K = [(1/m1 + 1/m2) * eye(2) - skew(r1) * invI1 * skew(r1) - skew(r2) * invI2 * skew(r2)]
// = [1/m1+1/m2 0 ] + invI1 * [r1.y*r1.y -r1.x*r1.y] + invI2 * [r1.y*r1.y -r1.x*r1.y]
// [ 0 1/m1+1/m2] [-r1.x*r1.y r1.x*r1.x] [-r1.x*r1.y r1.x*r1.x]
var K = this.m_K;
K.ex.x = this.m_invMassB + this.m_invIB * this.m_rB.y * this.m_rB.y + this.m_gamma;
K.ex.y = -this.m_invIB * this.m_rB.x * this.m_rB.y;
K.ey.x = K.ex.y;
K.ey.y = this.m_invMassB + this.m_invIB * this.m_rB.x * this.m_rB.x + this.m_gamma;
K.GetInverse(this.m_mass);
// m_C = cB + m_rB - m_targetA;
this.m_C.x = cB.x + this.m_rB.x - this.m_targetA.x;
this.m_C.y = cB.y + this.m_rB.y - this.m_targetA.y;
// m_C *= m_beta;
this.m_C.SelfMul(this.m_beta);
// Cheat with some damping
wB *= 0.98;
if (data.step.warmStarting) {
this.m_impulse.SelfMul(data.step.dtRatio);
// vB += m_invMassB * m_impulse;
vB.x += this.m_invMassB * this.m_impulse.x;
vB.y += this.m_invMassB * this.m_impulse.y;
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, this.m_impulse);
}
else {
this.m_impulse.SetZero();
}
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2MouseJoint.prototype.SolveVelocityConstraints = function (data) {
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// Cdot = v + cross(w, r)
// b2Vec2 Cdot = vB + b2Cross(wB, m_rB);
var Cdot = b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2MouseJoint.SolveVelocityConstraints_s_Cdot);
// b2Vec2 impulse = b2Mul(m_mass, -(Cdot + m_C + m_gamma * m_impulse));
var impulse = b2Mat22.MulMV(this.m_mass, b2Vec2.AddVV(Cdot, b2Vec2.AddVV(this.m_C, b2Vec2.MulSV(this.m_gamma, this.m_impulse, b2Vec2.s_t0), b2Vec2.s_t0), b2Vec2.s_t0).SelfNeg(), b2MouseJoint.SolveVelocityConstraints_s_impulse);
// b2Vec2 oldImpulse = m_impulse;
var oldImpulse = b2MouseJoint.SolveVelocityConstraints_s_oldImpulse.Copy(this.m_impulse);
// m_impulse += impulse;
this.m_impulse.SelfAdd(impulse);
var maxImpulse = data.step.dt * this.m_maxForce;
if (this.m_impulse.LengthSquared() > maxImpulse * maxImpulse) {
this.m_impulse.SelfMul(maxImpulse / this.m_impulse.Length());
}
// impulse = m_impulse - oldImpulse;
b2Vec2.SubVV(this.m_impulse, oldImpulse, impulse);
// vB += m_invMassB * impulse;
vB.SelfMulAdd(this.m_invMassB, impulse);
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, impulse);
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2MouseJoint.prototype.SolvePositionConstraints = function (data) {
return true;
};
b2MouseJoint.prototype.GetAnchorA = function (out) {
out.x = this.m_targetA.x;
out.y = this.m_targetA.y;
return out;
};
b2MouseJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2MouseJoint.prototype.GetReactionForce = function (inv_dt, out) {
return b2Vec2.MulSV(inv_dt, this.m_impulse, out);
};
b2MouseJoint.prototype.GetReactionTorque = function (inv_dt) {
return 0;
};
b2MouseJoint.prototype.Dump = function (log) {
log("Mouse joint dumping is not supported.\n");
};
b2MouseJoint.prototype.ShiftOrigin = function (newOrigin) {
this.m_targetA.SelfSub(newOrigin);
};
b2MouseJoint.SolveVelocityConstraints_s_Cdot = new b2Vec2();
b2MouseJoint.SolveVelocityConstraints_s_impulse = new b2Vec2();
b2MouseJoint.SolveVelocityConstraints_s_oldImpulse = new b2Vec2();
return b2MouseJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Prismatic joint definition. This requires defining a line of
/// motion using an axis and an anchor point. The definition uses local
/// anchor points and a local axis so that the initial configuration
/// can violate the constraint slightly. The joint translation is zero
/// when the local anchor points coincide in world space. Using local
/// anchors and a local axis helps when saving and loading a game.
var b2PrismaticJointDef = /** @class */ (function (_super) {
__extends(b2PrismaticJointDef, _super);
function b2PrismaticJointDef() {
var _this = _super.call(this, exports.b2JointType.e_prismaticJoint) || this;
_this.localAnchorA = new b2Vec2();
_this.localAnchorB = new b2Vec2();
_this.localAxisA = new b2Vec2(1, 0);
_this.referenceAngle = 0;
_this.enableLimit = false;
_this.lowerTranslation = 0;
_this.upperTranslation = 0;
_this.enableMotor = false;
_this.maxMotorForce = 0;
_this.motorSpeed = 0;
return _this;
}
b2PrismaticJointDef.prototype.Initialize = function (bA, bB, anchor, axis) {
this.bodyA = bA;
this.bodyB = bB;
this.bodyA.GetLocalPoint(anchor, this.localAnchorA);
this.bodyB.GetLocalPoint(anchor, this.localAnchorB);
this.bodyA.GetLocalVector(axis, this.localAxisA);
this.referenceAngle = this.bodyB.GetAngle() - this.bodyA.GetAngle();
};
return b2PrismaticJointDef;
}(b2JointDef));
var b2PrismaticJoint = /** @class */ (function (_super) {
__extends(b2PrismaticJoint, _super);
function b2PrismaticJoint(def) {
var _this = _super.call(this, def) || this;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_localXAxisA = new b2Vec2();
_this.m_localYAxisA = new b2Vec2();
_this.m_referenceAngle = 0;
_this.m_impulse = new b2Vec3(0, 0, 0);
_this.m_motorImpulse = 0;
_this.m_lowerTranslation = 0;
_this.m_upperTranslation = 0;
_this.m_maxMotorForce = 0;
_this.m_motorSpeed = 0;
_this.m_enableLimit = false;
_this.m_enableMotor = false;
_this.m_limitState = exports.b2LimitState.e_inactiveLimit;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_axis = new b2Vec2(0, 0);
_this.m_perp = new b2Vec2(0, 0);
_this.m_s1 = 0;
_this.m_s2 = 0;
_this.m_a1 = 0;
_this.m_a2 = 0;
_this.m_K = new b2Mat33();
_this.m_K3 = new b2Mat33();
_this.m_K2 = new b2Mat22();
_this.m_motorMass = 0;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localAnchorA.Copy(b2Maybe(def.localAnchorA, b2Vec2.ZERO));
_this.m_localAnchorB.Copy(b2Maybe(def.localAnchorB, b2Vec2.ZERO));
_this.m_localXAxisA.Copy(b2Maybe(def.localAxisA, new b2Vec2(1, 0))).SelfNormalize();
b2Vec2.CrossOneV(_this.m_localXAxisA, _this.m_localYAxisA);
_this.m_referenceAngle = b2Maybe(def.referenceAngle, 0);
_this.m_lowerTranslation = b2Maybe(def.lowerTranslation, 0);
_this.m_upperTranslation = b2Maybe(def.upperTranslation, 0);
_this.m_maxMotorForce = b2Maybe(def.maxMotorForce, 0);
_this.m_motorSpeed = b2Maybe(def.motorSpeed, 0);
_this.m_enableLimit = b2Maybe(def.enableLimit, false);
_this.m_enableMotor = b2Maybe(def.enableMotor, false);
return _this;
}
b2PrismaticJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// Compute the effective masses.
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// b2Vec2 d = (cB - cA) + rB - rA;
var d = b2Vec2.AddVV(b2Vec2.SubVV(cB, cA, b2Vec2.s_t0), b2Vec2.SubVV(rB, rA, b2Vec2.s_t1), b2PrismaticJoint.InitVelocityConstraints_s_d);
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
// Compute motor Jacobian and effective mass.
{
// m_axis = b2Mul(qA, m_localXAxisA);
b2Rot.MulRV(qA, this.m_localXAxisA, this.m_axis);
// m_a1 = b2Cross(d + rA, m_axis);
this.m_a1 = b2Vec2.CrossVV(b2Vec2.AddVV(d, rA, b2Vec2.s_t0), this.m_axis);
// m_a2 = b2Cross(rB, m_axis);
this.m_a2 = b2Vec2.CrossVV(rB, this.m_axis);
this.m_motorMass = mA + mB + iA * this.m_a1 * this.m_a1 + iB * this.m_a2 * this.m_a2;
if (this.m_motorMass > 0) {
this.m_motorMass = 1 / this.m_motorMass;
}
}
// Prismatic constraint.
{
// m_perp = b2Mul(qA, m_localYAxisA);
b2Rot.MulRV(qA, this.m_localYAxisA, this.m_perp);
// m_s1 = b2Cross(d + rA, m_perp);
this.m_s1 = b2Vec2.CrossVV(b2Vec2.AddVV(d, rA, b2Vec2.s_t0), this.m_perp);
// m_s2 = b2Cross(rB, m_perp);
this.m_s2 = b2Vec2.CrossVV(rB, this.m_perp);
// float32 k11 = mA + mB + iA * m_s1 * m_s1 + iB * m_s2 * m_s2;
this.m_K.ex.x = mA + mB + iA * this.m_s1 * this.m_s1 + iB * this.m_s2 * this.m_s2;
// float32 k12 = iA * m_s1 + iB * m_s2;
this.m_K.ex.y = iA * this.m_s1 + iB * this.m_s2;
// float32 k13 = iA * m_s1 * m_a1 + iB * m_s2 * m_a2;
this.m_K.ex.z = iA * this.m_s1 * this.m_a1 + iB * this.m_s2 * this.m_a2;
this.m_K.ey.x = this.m_K.ex.y;
// float32 k22 = iA + iB;
this.m_K.ey.y = iA + iB;
if (this.m_K.ey.y === 0) {
// For bodies with fixed rotation.
this.m_K.ey.y = 1;
}
// float32 k23 = iA * m_a1 + iB * m_a2;
this.m_K.ey.z = iA * this.m_a1 + iB * this.m_a2;
this.m_K.ez.x = this.m_K.ex.z;
this.m_K.ez.y = this.m_K.ey.z;
// float32 k33 = mA + mB + iA * m_a1 * m_a1 + iB * m_a2 * m_a2;
this.m_K.ez.z = mA + mB + iA * this.m_a1 * this.m_a1 + iB * this.m_a2 * this.m_a2;
// m_K.ex.Set(k11, k12, k13);
// m_K.ey.Set(k12, k22, k23);
// m_K.ez.Set(k13, k23, k33);
}
// Compute motor and limit terms.
if (this.m_enableLimit) {
// float32 jointTranslation = b2Dot(m_axis, d);
var jointTranslation = b2Vec2.DotVV(this.m_axis, d);
if (b2Abs(this.m_upperTranslation - this.m_lowerTranslation) < 2 * b2_linearSlop) {
this.m_limitState = exports.b2LimitState.e_equalLimits;
}
else if (jointTranslation <= this.m_lowerTranslation) {
if (this.m_limitState !== exports.b2LimitState.e_atLowerLimit) {
this.m_limitState = exports.b2LimitState.e_atLowerLimit;
this.m_impulse.z = 0;
}
}
else if (jointTranslation >= this.m_upperTranslation) {
if (this.m_limitState !== exports.b2LimitState.e_atUpperLimit) {
this.m_limitState = exports.b2LimitState.e_atUpperLimit;
this.m_impulse.z = 0;
}
}
else {
this.m_limitState = exports.b2LimitState.e_inactiveLimit;
this.m_impulse.z = 0;
}
}
else {
this.m_limitState = exports.b2LimitState.e_inactiveLimit;
this.m_impulse.z = 0;
}
if (!this.m_enableMotor) {
this.m_motorImpulse = 0;
}
if (data.step.warmStarting) {
// Account for variable time step.
// m_impulse *= data.step.dtRatio;
this.m_impulse.SelfMul(data.step.dtRatio);
this.m_motorImpulse *= data.step.dtRatio;
// b2Vec2 P = m_impulse.x * m_perp + (m_motorImpulse + m_impulse.z) * m_axis;
var P = b2Vec2.AddVV(b2Vec2.MulSV(this.m_impulse.x, this.m_perp, b2Vec2.s_t0), b2Vec2.MulSV((this.m_motorImpulse + this.m_impulse.z), this.m_axis, b2Vec2.s_t1), b2PrismaticJoint.InitVelocityConstraints_s_P);
// float32 LA = m_impulse.x * m_s1 + m_impulse.y + (m_motorImpulse + m_impulse.z) * m_a1;
var LA = this.m_impulse.x * this.m_s1 + this.m_impulse.y + (this.m_motorImpulse + this.m_impulse.z) * this.m_a1;
// float32 LB = m_impulse.x * m_s2 + m_impulse.y + (m_motorImpulse + m_impulse.z) * m_a2;
var LB = this.m_impulse.x * this.m_s2 + this.m_impulse.y + (this.m_motorImpulse + this.m_impulse.z) * this.m_a2;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * LA;
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * LB;
}
else {
this.m_impulse.SetZero();
this.m_motorImpulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2PrismaticJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
// Solve linear motor constraint.
if (this.m_enableMotor && this.m_limitState !== exports.b2LimitState.e_equalLimits) {
// float32 Cdot = b2Dot(m_axis, vB - vA) + m_a2 * wB - m_a1 * wA;
var Cdot = b2Vec2.DotVV(this.m_axis, b2Vec2.SubVV(vB, vA, b2Vec2.s_t0)) + this.m_a2 * wB - this.m_a1 * wA;
var impulse = this.m_motorMass * (this.m_motorSpeed - Cdot);
var oldImpulse = this.m_motorImpulse;
var maxImpulse = data.step.dt * this.m_maxMotorForce;
this.m_motorImpulse = b2Clamp(this.m_motorImpulse + impulse, (-maxImpulse), maxImpulse);
impulse = this.m_motorImpulse - oldImpulse;
// b2Vec2 P = impulse * m_axis;
var P = b2Vec2.MulSV(impulse, this.m_axis, b2PrismaticJoint.SolveVelocityConstraints_s_P);
var LA = impulse * this.m_a1;
var LB = impulse * this.m_a2;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * LA;
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * LB;
}
// b2Vec2 Cdot1;
// Cdot1.x = b2Dot(m_perp, vB - vA) + m_s2 * wB - m_s1 * wA;
var Cdot1_x = b2Vec2.DotVV(this.m_perp, b2Vec2.SubVV(vB, vA, b2Vec2.s_t0)) + this.m_s2 * wB - this.m_s1 * wA;
// Cdot1.y = wB - wA;
var Cdot1_y = wB - wA;
if (this.m_enableLimit && this.m_limitState !== exports.b2LimitState.e_inactiveLimit) {
// Solve prismatic and limit constraint in block form.
// float32 Cdot2;
// Cdot2 = b2Dot(m_axis, vB - vA) + m_a2 * wB - m_a1 * wA;
var Cdot2 = b2Vec2.DotVV(this.m_axis, b2Vec2.SubVV(vB, vA, b2Vec2.s_t0)) + this.m_a2 * wB - this.m_a1 * wA;
// b2Vec3 Cdot(Cdot1.x, Cdot1.y, Cdot2);
// b2Vec3 f1 = m_impulse;
var f1 = b2PrismaticJoint.SolveVelocityConstraints_s_f1.Copy(this.m_impulse);
// b2Vec3 df = m_K.Solve33(-Cdot);
var df3 = this.m_K.Solve33((-Cdot1_x), (-Cdot1_y), (-Cdot2), b2PrismaticJoint.SolveVelocityConstraints_s_df3);
// m_impulse += df;
this.m_impulse.SelfAdd(df3);
if (this.m_limitState === exports.b2LimitState.e_atLowerLimit) {
this.m_impulse.z = b2Max(this.m_impulse.z, 0);
}
else if (this.m_limitState === exports.b2LimitState.e_atUpperLimit) {
this.m_impulse.z = b2Min(this.m_impulse.z, 0);
}
// f2(1:2) = invK(1:2,1:2) * (-Cdot(1:2) - K(1:2,3) * (f2(3) - f1(3))) + f1(1:2)
// b2Vec2 b = -Cdot1 - (m_impulse.z - f1.z) * b2Vec2(m_K.ez.x, m_K.ez.y);
var b_x = (-Cdot1_x) - (this.m_impulse.z - f1.z) * this.m_K.ez.x;
var b_y = (-Cdot1_y) - (this.m_impulse.z - f1.z) * this.m_K.ez.y;
// b2Vec2 f2r = m_K.Solve22(b) + b2Vec2(f1.x, f1.y);
var f2r = this.m_K.Solve22(b_x, b_y, b2PrismaticJoint.SolveVelocityConstraints_s_f2r);
f2r.x += f1.x;
f2r.y += f1.y;
// m_impulse.x = f2r.x;
this.m_impulse.x = f2r.x;
// m_impulse.y = f2r.y;
this.m_impulse.y = f2r.y;
// df = m_impulse - f1;
df3.x = this.m_impulse.x - f1.x;
df3.y = this.m_impulse.y - f1.y;
df3.z = this.m_impulse.z - f1.z;
// b2Vec2 P = df.x * m_perp + df.z * m_axis;
var P = b2Vec2.AddVV(b2Vec2.MulSV(df3.x, this.m_perp, b2Vec2.s_t0), b2Vec2.MulSV(df3.z, this.m_axis, b2Vec2.s_t1), b2PrismaticJoint.SolveVelocityConstraints_s_P);
// float32 LA = df.x * m_s1 + df.y + df.z * m_a1;
var LA = df3.x * this.m_s1 + df3.y + df3.z * this.m_a1;
// float32 LB = df.x * m_s2 + df.y + df.z * m_a2;
var LB = df3.x * this.m_s2 + df3.y + df3.z * this.m_a2;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * LA;
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * LB;
}
else {
// Limit is inactive, just solve the prismatic constraint in block form.
// b2Vec2 df = m_K.Solve22(-Cdot1);
var df2 = this.m_K.Solve22((-Cdot1_x), (-Cdot1_y), b2PrismaticJoint.SolveVelocityConstraints_s_df2);
this.m_impulse.x += df2.x;
this.m_impulse.y += df2.y;
// b2Vec2 P = df.x * m_perp;
var P = b2Vec2.MulSV(df2.x, this.m_perp, b2PrismaticJoint.SolveVelocityConstraints_s_P);
// float32 LA = df.x * m_s1 + df.y;
var LA = df2.x * this.m_s1 + df2.y;
// float32 LB = df.x * m_s2 + df.y;
var LB = df2.x * this.m_s2 + df2.y;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * LA;
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * LB;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2PrismaticJoint.prototype.SolvePositionConstraints = function (data) {
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// b2Vec2 d = cB + rB - cA - rA;
var d = b2Vec2.SubVV(b2Vec2.AddVV(cB, rB, b2Vec2.s_t0), b2Vec2.AddVV(cA, rA, b2Vec2.s_t1), b2PrismaticJoint.SolvePositionConstraints_s_d);
// b2Vec2 axis = b2Mul(qA, m_localXAxisA);
var axis = b2Rot.MulRV(qA, this.m_localXAxisA, this.m_axis);
// float32 a1 = b2Cross(d + rA, axis);
var a1 = b2Vec2.CrossVV(b2Vec2.AddVV(d, rA, b2Vec2.s_t0), axis);
// float32 a2 = b2Cross(rB, axis);
var a2 = b2Vec2.CrossVV(rB, axis);
// b2Vec2 perp = b2Mul(qA, m_localYAxisA);
var perp = b2Rot.MulRV(qA, this.m_localYAxisA, this.m_perp);
// float32 s1 = b2Cross(d + rA, perp);
var s1 = b2Vec2.CrossVV(b2Vec2.AddVV(d, rA, b2Vec2.s_t0), perp);
// float32 s2 = b2Cross(rB, perp);
var s2 = b2Vec2.CrossVV(rB, perp);
// b2Vec3 impulse;
var impulse = b2PrismaticJoint.SolvePositionConstraints_s_impulse;
// b2Vec2 C1;
// C1.x = b2Dot(perp, d);
var C1_x = b2Vec2.DotVV(perp, d);
// C1.y = aB - aA - m_referenceAngle;
var C1_y = aB - aA - this.m_referenceAngle;
var linearError = b2Abs(C1_x);
var angularError = b2Abs(C1_y);
var active = false;
var C2 = 0;
if (this.m_enableLimit) {
// float32 translation = b2Dot(axis, d);
var translation = b2Vec2.DotVV(axis, d);
if (b2Abs(this.m_upperTranslation - this.m_lowerTranslation) < 2 * b2_linearSlop) {
// Prevent large angular corrections
C2 = b2Clamp(translation, (-b2_maxLinearCorrection), b2_maxLinearCorrection);
linearError = b2Max(linearError, b2Abs(translation));
active = true;
}
else if (translation <= this.m_lowerTranslation) {
// Prevent large linear corrections and allow some slop.
C2 = b2Clamp(translation - this.m_lowerTranslation + b2_linearSlop, (-b2_maxLinearCorrection), 0);
linearError = b2Max(linearError, this.m_lowerTranslation - translation);
active = true;
}
else if (translation >= this.m_upperTranslation) {
// Prevent large linear corrections and allow some slop.
C2 = b2Clamp(translation - this.m_upperTranslation - b2_linearSlop, 0, b2_maxLinearCorrection);
linearError = b2Max(linearError, translation - this.m_upperTranslation);
active = true;
}
}
if (active) {
// float32 k11 = mA + mB + iA * s1 * s1 + iB * s2 * s2;
var k11 = mA + mB + iA * s1 * s1 + iB * s2 * s2;
// float32 k12 = iA * s1 + iB * s2;
var k12 = iA * s1 + iB * s2;
// float32 k13 = iA * s1 * a1 + iB * s2 * a2;
var k13 = iA * s1 * a1 + iB * s2 * a2;
// float32 k22 = iA + iB;
var k22 = iA + iB;
if (k22 === 0) {
// For fixed rotation
k22 = 1;
}
// float32 k23 = iA * a1 + iB * a2;
var k23 = iA * a1 + iB * a2;
// float32 k33 = mA + mB + iA * a1 * a1 + iB * a2 * a2;
var k33 = mA + mB + iA * a1 * a1 + iB * a2 * a2;
// b2Mat33 K;
var K = this.m_K3;
// K.ex.Set(k11, k12, k13);
K.ex.SetXYZ(k11, k12, k13);
// K.ey.Set(k12, k22, k23);
K.ey.SetXYZ(k12, k22, k23);
// K.ez.Set(k13, k23, k33);
K.ez.SetXYZ(k13, k23, k33);
// b2Vec3 C;
// C.x = C1.x;
// C.y = C1.y;
// C.z = C2;
// impulse = K.Solve33(-C);
impulse = K.Solve33((-C1_x), (-C1_y), (-C2), impulse);
}
else {
// float32 k11 = mA + mB + iA * s1 * s1 + iB * s2 * s2;
var k11 = mA + mB + iA * s1 * s1 + iB * s2 * s2;
// float32 k12 = iA * s1 + iB * s2;
var k12 = iA * s1 + iB * s2;
// float32 k22 = iA + iB;
var k22 = iA + iB;
if (k22 === 0) {
k22 = 1;
}
// b2Mat22 K;
var K2 = this.m_K2;
// K.ex.Set(k11, k12);
K2.ex.Set(k11, k12);
// K.ey.Set(k12, k22);
K2.ey.Set(k12, k22);
// b2Vec2 impulse1 = K.Solve(-C1);
var impulse1 = K2.Solve((-C1_x), (-C1_y), b2PrismaticJoint.SolvePositionConstraints_s_impulse1);
impulse.x = impulse1.x;
impulse.y = impulse1.y;
impulse.z = 0;
}
// b2Vec2 P = impulse.x * perp + impulse.z * axis;
var P = b2Vec2.AddVV(b2Vec2.MulSV(impulse.x, perp, b2Vec2.s_t0), b2Vec2.MulSV(impulse.z, axis, b2Vec2.s_t1), b2PrismaticJoint.SolvePositionConstraints_s_P);
// float32 LA = impulse.x * s1 + impulse.y + impulse.z * a1;
var LA = impulse.x * s1 + impulse.y + impulse.z * a1;
// float32 LB = impulse.x * s2 + impulse.y + impulse.z * a2;
var LB = impulse.x * s2 + impulse.y + impulse.z * a2;
// cA -= mA * P;
cA.SelfMulSub(mA, P);
aA -= iA * LA;
// cB += mB * P;
cB.SelfMulAdd(mB, P);
aB += iB * LB;
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
return linearError <= b2_linearSlop && angularError <= b2_angularSlop;
};
b2PrismaticJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2PrismaticJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2PrismaticJoint.prototype.GetReactionForce = function (inv_dt, out) {
// return inv_dt * (m_impulse.x * m_perp + (m_motorImpulse + m_impulse.z) * m_axis);
out.x = inv_dt * (this.m_impulse.x * this.m_perp.x + (this.m_motorImpulse + this.m_impulse.z) * this.m_axis.x);
out.y = inv_dt * (this.m_impulse.x * this.m_perp.y + (this.m_motorImpulse + this.m_impulse.z) * this.m_axis.y);
return out;
};
b2PrismaticJoint.prototype.GetReactionTorque = function (inv_dt) {
return inv_dt * this.m_impulse.y;
};
b2PrismaticJoint.prototype.GetLocalAnchorA = function () { return this.m_localAnchorA; };
b2PrismaticJoint.prototype.GetLocalAnchorB = function () { return this.m_localAnchorB; };
b2PrismaticJoint.prototype.GetLocalAxisA = function () { return this.m_localXAxisA; };
b2PrismaticJoint.prototype.GetReferenceAngle = function () { return this.m_referenceAngle; };
b2PrismaticJoint.prototype.GetJointTranslation = function () {
// b2Vec2 pA = m_bodyA.GetWorldPoint(m_localAnchorA);
var pA = this.m_bodyA.GetWorldPoint(this.m_localAnchorA, b2PrismaticJoint.GetJointTranslation_s_pA);
// b2Vec2 pB = m_bodyB.GetWorldPoint(m_localAnchorB);
var pB = this.m_bodyB.GetWorldPoint(this.m_localAnchorB, b2PrismaticJoint.GetJointTranslation_s_pB);
// b2Vec2 d = pB - pA;
var d = b2Vec2.SubVV(pB, pA, b2PrismaticJoint.GetJointTranslation_s_d);
// b2Vec2 axis = m_bodyA.GetWorldVector(m_localXAxisA);
var axis = this.m_bodyA.GetWorldVector(this.m_localXAxisA, b2PrismaticJoint.GetJointTranslation_s_axis);
// float32 translation = b2Dot(d, axis);
var translation = b2Vec2.DotVV(d, axis);
return translation;
};
b2PrismaticJoint.prototype.GetJointSpeed = function () {
var bA = this.m_bodyA;
var bB = this.m_bodyB;
// b2Vec2 rA = b2Mul(bA->m_xf.q, m_localAnchorA - bA->m_sweep.localCenter);
b2Vec2.SubVV(this.m_localAnchorA, bA.m_sweep.localCenter, this.m_lalcA);
var rA = b2Rot.MulRV(bA.m_xf.q, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(bB->m_xf.q, m_localAnchorB - bB->m_sweep.localCenter);
b2Vec2.SubVV(this.m_localAnchorB, bB.m_sweep.localCenter, this.m_lalcB);
var rB = b2Rot.MulRV(bB.m_xf.q, this.m_lalcB, this.m_rB);
// b2Vec2 pA = bA->m_sweep.c + rA;
var pA = b2Vec2.AddVV(bA.m_sweep.c, rA, b2Vec2.s_t0); // pA uses s_t0
// b2Vec2 pB = bB->m_sweep.c + rB;
var pB = b2Vec2.AddVV(bB.m_sweep.c, rB, b2Vec2.s_t1); // pB uses s_t1
// b2Vec2 d = pB - pA;
var d = b2Vec2.SubVV(pB, pA, b2Vec2.s_t2); // d uses s_t2
// b2Vec2 axis = b2Mul(bA.m_xf.q, m_localXAxisA);
var axis = bA.GetWorldVector(this.m_localXAxisA, this.m_axis);
var vA = bA.m_linearVelocity;
var vB = bB.m_linearVelocity;
var wA = bA.m_angularVelocity;
var wB = bB.m_angularVelocity;
// float32 speed = b2Dot(d, b2Cross(wA, axis)) + b2Dot(axis, vB + b2Cross(wB, rB) - vA - b2Cross(wA, rA));
var speed = b2Vec2.DotVV(d, b2Vec2.CrossSV(wA, axis, b2Vec2.s_t0)) +
b2Vec2.DotVV(axis, b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, rA, b2Vec2.s_t1), b2Vec2.s_t0));
return speed;
};
b2PrismaticJoint.prototype.IsLimitEnabled = function () {
return this.m_enableLimit;
};
b2PrismaticJoint.prototype.EnableLimit = function (flag) {
if (flag !== this.m_enableLimit) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_enableLimit = flag;
this.m_impulse.z = 0;
}
};
b2PrismaticJoint.prototype.GetLowerLimit = function () {
return this.m_lowerTranslation;
};
b2PrismaticJoint.prototype.GetUpperLimit = function () {
return this.m_upperTranslation;
};
b2PrismaticJoint.prototype.SetLimits = function (lower, upper) {
if (lower !== this.m_lowerTranslation || upper !== this.m_upperTranslation) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_lowerTranslation = lower;
this.m_upperTranslation = upper;
this.m_impulse.z = 0;
}
};
b2PrismaticJoint.prototype.IsMotorEnabled = function () {
return this.m_enableMotor;
};
b2PrismaticJoint.prototype.EnableMotor = function (flag) {
if (flag !== this.m_enableMotor) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_enableMotor = flag;
}
};
b2PrismaticJoint.prototype.SetMotorSpeed = function (speed) {
if (speed !== this.m_motorSpeed) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_motorSpeed = speed;
}
};
b2PrismaticJoint.prototype.GetMotorSpeed = function () {
return this.m_motorSpeed;
};
b2PrismaticJoint.prototype.SetMaxMotorForce = function (force) {
if (force !== this.m_maxMotorForce) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_maxMotorForce = force;
}
};
b2PrismaticJoint.prototype.GetMaxMotorForce = function () { return this.m_maxMotorForce; };
b2PrismaticJoint.prototype.GetMotorForce = function (inv_dt) {
return inv_dt * this.m_motorImpulse;
};
b2PrismaticJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2PrismaticJointDef = new b2PrismaticJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.localAxisA.Set(%.15f, %.15f);\n", this.m_localXAxisA.x, this.m_localXAxisA.y);
log(" jd.referenceAngle = %.15f;\n", this.m_referenceAngle);
log(" jd.enableLimit = %s;\n", (this.m_enableLimit) ? ("true") : ("false"));
log(" jd.lowerTranslation = %.15f;\n", this.m_lowerTranslation);
log(" jd.upperTranslation = %.15f;\n", this.m_upperTranslation);
log(" jd.enableMotor = %s;\n", (this.m_enableMotor) ? ("true") : ("false"));
log(" jd.motorSpeed = %.15f;\n", this.m_motorSpeed);
log(" jd.maxMotorForce = %.15f;\n", this.m_maxMotorForce);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2PrismaticJoint.InitVelocityConstraints_s_d = new b2Vec2();
b2PrismaticJoint.InitVelocityConstraints_s_P = new b2Vec2();
b2PrismaticJoint.SolveVelocityConstraints_s_P = new b2Vec2();
b2PrismaticJoint.SolveVelocityConstraints_s_f2r = new b2Vec2();
b2PrismaticJoint.SolveVelocityConstraints_s_f1 = new b2Vec3();
b2PrismaticJoint.SolveVelocityConstraints_s_df3 = new b2Vec3();
b2PrismaticJoint.SolveVelocityConstraints_s_df2 = new b2Vec2();
// A velocity based solver computes reaction forces(impulses) using the velocity constraint solver.Under this context,
// the position solver is not there to resolve forces.It is only there to cope with integration error.
//
// Therefore, the pseudo impulses in the position solver do not have any physical meaning.Thus it is okay if they suck.
//
// We could take the active state from the velocity solver.However, the joint might push past the limit when the velocity
// solver indicates the limit is inactive.
b2PrismaticJoint.SolvePositionConstraints_s_d = new b2Vec2();
b2PrismaticJoint.SolvePositionConstraints_s_impulse = new b2Vec3();
b2PrismaticJoint.SolvePositionConstraints_s_impulse1 = new b2Vec2();
b2PrismaticJoint.SolvePositionConstraints_s_P = new b2Vec2();
b2PrismaticJoint.GetJointTranslation_s_pA = new b2Vec2();
b2PrismaticJoint.GetJointTranslation_s_pB = new b2Vec2();
b2PrismaticJoint.GetJointTranslation_s_d = new b2Vec2();
b2PrismaticJoint.GetJointTranslation_s_axis = new b2Vec2();
return b2PrismaticJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2_minPulleyLength = 2;
/// Pulley joint definition. This requires two ground anchors,
/// two dynamic body anchor points, and a pulley ratio.
var b2PulleyJointDef = /** @class */ (function (_super) {
__extends(b2PulleyJointDef, _super);
function b2PulleyJointDef() {
var _this = _super.call(this, exports.b2JointType.e_pulleyJoint) || this;
_this.groundAnchorA = new b2Vec2(-1, 1);
_this.groundAnchorB = new b2Vec2(1, 1);
_this.localAnchorA = new b2Vec2(-1, 0);
_this.localAnchorB = new b2Vec2(1, 0);
_this.lengthA = 0;
_this.lengthB = 0;
_this.ratio = 1;
_this.collideConnected = true;
return _this;
}
b2PulleyJointDef.prototype.Initialize = function (bA, bB, groundA, groundB, anchorA, anchorB, r) {
this.bodyA = bA;
this.bodyB = bB;
this.groundAnchorA.Copy(groundA);
this.groundAnchorB.Copy(groundB);
this.bodyA.GetLocalPoint(anchorA, this.localAnchorA);
this.bodyB.GetLocalPoint(anchorB, this.localAnchorB);
this.lengthA = b2Vec2.DistanceVV(anchorA, groundA);
this.lengthB = b2Vec2.DistanceVV(anchorB, groundB);
this.ratio = r;
// DEBUG: b2Assert(this.ratio > b2_epsilon);
};
return b2PulleyJointDef;
}(b2JointDef));
var b2PulleyJoint = /** @class */ (function (_super) {
__extends(b2PulleyJoint, _super);
function b2PulleyJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_groundAnchorA = new b2Vec2();
_this.m_groundAnchorB = new b2Vec2();
_this.m_lengthA = 0;
_this.m_lengthB = 0;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_constant = 0;
_this.m_ratio = 0;
_this.m_impulse = 0;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_uA = new b2Vec2();
_this.m_uB = new b2Vec2();
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_mass = 0;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_groundAnchorA.Copy(b2Maybe(def.groundAnchorA, new b2Vec2(-1, 1)));
_this.m_groundAnchorB.Copy(b2Maybe(def.groundAnchorB, new b2Vec2(1, 0)));
_this.m_localAnchorA.Copy(b2Maybe(def.localAnchorA, new b2Vec2(-1, 0)));
_this.m_localAnchorB.Copy(b2Maybe(def.localAnchorB, new b2Vec2(1, 0)));
_this.m_lengthA = b2Maybe(def.lengthA, 0);
_this.m_lengthB = b2Maybe(def.lengthB, 0);
// DEBUG: b2Assert(b2Maybe(def.ratio, 1) !== 0);
_this.m_ratio = b2Maybe(def.ratio, 1);
_this.m_constant = b2Maybe(def.lengthA, 0) + _this.m_ratio * b2Maybe(def.lengthB, 0);
_this.m_impulse = 0;
return _this;
}
b2PulleyJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// b2Rot qA(aA), qB(aB);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// m_rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// m_rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// Get the pulley axes.
// m_uA = cA + m_rA - m_groundAnchorA;
this.m_uA.Copy(cA).SelfAdd(this.m_rA).SelfSub(this.m_groundAnchorA);
// m_uB = cB + m_rB - m_groundAnchorB;
this.m_uB.Copy(cB).SelfAdd(this.m_rB).SelfSub(this.m_groundAnchorB);
var lengthA = this.m_uA.Length();
var lengthB = this.m_uB.Length();
if (lengthA > 10 * b2_linearSlop) {
this.m_uA.SelfMul(1 / lengthA);
}
else {
this.m_uA.SetZero();
}
if (lengthB > 10 * b2_linearSlop) {
this.m_uB.SelfMul(1 / lengthB);
}
else {
this.m_uB.SetZero();
}
// Compute effective mass.
var ruA = b2Vec2.CrossVV(this.m_rA, this.m_uA);
var ruB = b2Vec2.CrossVV(this.m_rB, this.m_uB);
var mA = this.m_invMassA + this.m_invIA * ruA * ruA;
var mB = this.m_invMassB + this.m_invIB * ruB * ruB;
this.m_mass = mA + this.m_ratio * this.m_ratio * mB;
if (this.m_mass > 0) {
this.m_mass = 1 / this.m_mass;
}
if (data.step.warmStarting) {
// Scale impulses to support variable time steps.
this.m_impulse *= data.step.dtRatio;
// Warm starting.
// b2Vec2 PA = -(m_impulse) * m_uA;
var PA = b2Vec2.MulSV(-(this.m_impulse), this.m_uA, b2PulleyJoint.InitVelocityConstraints_s_PA);
// b2Vec2 PB = (-m_ratio * m_impulse) * m_uB;
var PB = b2Vec2.MulSV((-this.m_ratio * this.m_impulse), this.m_uB, b2PulleyJoint.InitVelocityConstraints_s_PB);
// vA += m_invMassA * PA;
vA.SelfMulAdd(this.m_invMassA, PA);
wA += this.m_invIA * b2Vec2.CrossVV(this.m_rA, PA);
// vB += m_invMassB * PB;
vB.SelfMulAdd(this.m_invMassB, PB);
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, PB);
}
else {
this.m_impulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2PulleyJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// b2Vec2 vpA = vA + b2Cross(wA, m_rA);
var vpA = b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2PulleyJoint.SolveVelocityConstraints_s_vpA);
// b2Vec2 vpB = vB + b2Cross(wB, m_rB);
var vpB = b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2PulleyJoint.SolveVelocityConstraints_s_vpB);
var Cdot = -b2Vec2.DotVV(this.m_uA, vpA) - this.m_ratio * b2Vec2.DotVV(this.m_uB, vpB);
var impulse = -this.m_mass * Cdot;
this.m_impulse += impulse;
// b2Vec2 PA = -impulse * m_uA;
var PA = b2Vec2.MulSV(-impulse, this.m_uA, b2PulleyJoint.SolveVelocityConstraints_s_PA);
// b2Vec2 PB = -m_ratio * impulse * m_uB;
var PB = b2Vec2.MulSV(-this.m_ratio * impulse, this.m_uB, b2PulleyJoint.SolveVelocityConstraints_s_PB);
// vA += m_invMassA * PA;
vA.SelfMulAdd(this.m_invMassA, PA);
wA += this.m_invIA * b2Vec2.CrossVV(this.m_rA, PA);
// vB += m_invMassB * PB;
vB.SelfMulAdd(this.m_invMassB, PB);
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, PB);
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2PulleyJoint.prototype.SolvePositionConstraints = function (data) {
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
// b2Rot qA(aA), qB(aB);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// Get the pulley axes.
// b2Vec2 uA = cA + rA - m_groundAnchorA;
var uA = this.m_uA.Copy(cA).SelfAdd(rA).SelfSub(this.m_groundAnchorA);
// b2Vec2 uB = cB + rB - m_groundAnchorB;
var uB = this.m_uB.Copy(cB).SelfAdd(rB).SelfSub(this.m_groundAnchorB);
var lengthA = uA.Length();
var lengthB = uB.Length();
if (lengthA > 10 * b2_linearSlop) {
uA.SelfMul(1 / lengthA);
}
else {
uA.SetZero();
}
if (lengthB > 10 * b2_linearSlop) {
uB.SelfMul(1 / lengthB);
}
else {
uB.SetZero();
}
// Compute effective mass.
var ruA = b2Vec2.CrossVV(rA, uA);
var ruB = b2Vec2.CrossVV(rB, uB);
var mA = this.m_invMassA + this.m_invIA * ruA * ruA;
var mB = this.m_invMassB + this.m_invIB * ruB * ruB;
var mass = mA + this.m_ratio * this.m_ratio * mB;
if (mass > 0) {
mass = 1 / mass;
}
var C = this.m_constant - lengthA - this.m_ratio * lengthB;
var linearError = b2Abs(C);
var impulse = -mass * C;
// b2Vec2 PA = -impulse * uA;
var PA = b2Vec2.MulSV(-impulse, uA, b2PulleyJoint.SolvePositionConstraints_s_PA);
// b2Vec2 PB = -m_ratio * impulse * uB;
var PB = b2Vec2.MulSV(-this.m_ratio * impulse, uB, b2PulleyJoint.SolvePositionConstraints_s_PB);
// cA += m_invMassA * PA;
cA.SelfMulAdd(this.m_invMassA, PA);
aA += this.m_invIA * b2Vec2.CrossVV(rA, PA);
// cB += m_invMassB * PB;
cB.SelfMulAdd(this.m_invMassB, PB);
aB += this.m_invIB * b2Vec2.CrossVV(rB, PB);
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
return linearError < b2_linearSlop;
};
b2PulleyJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2PulleyJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2PulleyJoint.prototype.GetReactionForce = function (inv_dt, out) {
// b2Vec2 P = m_impulse * m_uB;
// return inv_dt * P;
out.x = inv_dt * this.m_impulse * this.m_uB.x;
out.y = inv_dt * this.m_impulse * this.m_uB.y;
return out;
};
b2PulleyJoint.prototype.GetReactionTorque = function (inv_dt) {
return 0;
};
b2PulleyJoint.prototype.GetGroundAnchorA = function () {
return this.m_groundAnchorA;
};
b2PulleyJoint.prototype.GetGroundAnchorB = function () {
return this.m_groundAnchorB;
};
b2PulleyJoint.prototype.GetLengthA = function () {
return this.m_lengthA;
};
b2PulleyJoint.prototype.GetLengthB = function () {
return this.m_lengthB;
};
b2PulleyJoint.prototype.GetRatio = function () {
return this.m_ratio;
};
b2PulleyJoint.prototype.GetCurrentLengthA = function () {
// b2Vec2 p = m_bodyA->GetWorldPoint(m_localAnchorA);
// b2Vec2 s = m_groundAnchorA;
// b2Vec2 d = p - s;
// return d.Length();
var p = this.m_bodyA.GetWorldPoint(this.m_localAnchorA, b2PulleyJoint.GetCurrentLengthA_s_p);
var s = this.m_groundAnchorA;
return b2Vec2.DistanceVV(p, s);
};
b2PulleyJoint.prototype.GetCurrentLengthB = function () {
// b2Vec2 p = m_bodyB->GetWorldPoint(m_localAnchorB);
// b2Vec2 s = m_groundAnchorB;
// b2Vec2 d = p - s;
// return d.Length();
var p = this.m_bodyB.GetWorldPoint(this.m_localAnchorB, b2PulleyJoint.GetCurrentLengthB_s_p);
var s = this.m_groundAnchorB;
return b2Vec2.DistanceVV(p, s);
};
b2PulleyJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2PulleyJointDef = new b2PulleyJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.groundAnchorA.Set(%.15f, %.15f);\n", this.m_groundAnchorA.x, this.m_groundAnchorA.y);
log(" jd.groundAnchorB.Set(%.15f, %.15f);\n", this.m_groundAnchorB.x, this.m_groundAnchorB.y);
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.lengthA = %.15f;\n", this.m_lengthA);
log(" jd.lengthB = %.15f;\n", this.m_lengthB);
log(" jd.ratio = %.15f;\n", this.m_ratio);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2PulleyJoint.prototype.ShiftOrigin = function (newOrigin) {
this.m_groundAnchorA.SelfSub(newOrigin);
this.m_groundAnchorB.SelfSub(newOrigin);
};
b2PulleyJoint.InitVelocityConstraints_s_PA = new b2Vec2();
b2PulleyJoint.InitVelocityConstraints_s_PB = new b2Vec2();
b2PulleyJoint.SolveVelocityConstraints_s_vpA = new b2Vec2();
b2PulleyJoint.SolveVelocityConstraints_s_vpB = new b2Vec2();
b2PulleyJoint.SolveVelocityConstraints_s_PA = new b2Vec2();
b2PulleyJoint.SolveVelocityConstraints_s_PB = new b2Vec2();
b2PulleyJoint.SolvePositionConstraints_s_PA = new b2Vec2();
b2PulleyJoint.SolvePositionConstraints_s_PB = new b2Vec2();
b2PulleyJoint.GetCurrentLengthA_s_p = new b2Vec2();
b2PulleyJoint.GetCurrentLengthB_s_p = new b2Vec2();
return b2PulleyJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Revolute joint definition. This requires defining an
/// anchor point where the bodies are joined. The definition
/// uses local anchor points so that the initial configuration
/// can violate the constraint slightly. You also need to
/// specify the initial relative angle for joint limits. This
/// helps when saving and loading a game.
/// The local anchor points are measured from the body's origin
/// rather than the center of mass because:
/// 1. you might not know where the center of mass will be.
/// 2. if you add/remove shapes from a body and recompute the mass,
/// the joints will be broken.
var b2RevoluteJointDef = /** @class */ (function (_super) {
__extends(b2RevoluteJointDef, _super);
function b2RevoluteJointDef() {
var _this = _super.call(this, exports.b2JointType.e_revoluteJoint) || this;
_this.localAnchorA = new b2Vec2(0, 0);
_this.localAnchorB = new b2Vec2(0, 0);
_this.referenceAngle = 0;
_this.enableLimit = false;
_this.lowerAngle = 0;
_this.upperAngle = 0;
_this.enableMotor = false;
_this.motorSpeed = 0;
_this.maxMotorTorque = 0;
return _this;
}
b2RevoluteJointDef.prototype.Initialize = function (bA, bB, anchor) {
this.bodyA = bA;
this.bodyB = bB;
this.bodyA.GetLocalPoint(anchor, this.localAnchorA);
this.bodyB.GetLocalPoint(anchor, this.localAnchorB);
this.referenceAngle = this.bodyB.GetAngle() - this.bodyA.GetAngle();
};
return b2RevoluteJointDef;
}(b2JointDef));
var b2RevoluteJoint = /** @class */ (function (_super) {
__extends(b2RevoluteJoint, _super);
function b2RevoluteJoint(def) {
var _this = _super.call(this, def) || this;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_impulse = new b2Vec3();
_this.m_motorImpulse = 0;
_this.m_enableMotor = false;
_this.m_maxMotorTorque = 0;
_this.m_motorSpeed = 0;
_this.m_enableLimit = false;
_this.m_referenceAngle = 0;
_this.m_lowerAngle = 0;
_this.m_upperAngle = 0;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_mass = new b2Mat33(); // effective mass for point-to-point constraint.
_this.m_motorMass = 0; // effective mass for motor/limit angular constraint.
_this.m_limitState = exports.b2LimitState.e_inactiveLimit;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_K = new b2Mat22();
_this.m_localAnchorA.Copy(b2Maybe(def.localAnchorA, b2Vec2.ZERO));
_this.m_localAnchorB.Copy(b2Maybe(def.localAnchorB, b2Vec2.ZERO));
_this.m_referenceAngle = b2Maybe(def.referenceAngle, 0);
_this.m_impulse.SetZero();
_this.m_motorImpulse = 0;
_this.m_lowerAngle = b2Maybe(def.lowerAngle, 0);
_this.m_upperAngle = b2Maybe(def.upperAngle, 0);
_this.m_maxMotorTorque = b2Maybe(def.maxMotorTorque, 0);
_this.m_motorSpeed = b2Maybe(def.motorSpeed, 0);
_this.m_enableLimit = b2Maybe(def.enableLimit, false);
_this.m_enableMotor = b2Maybe(def.enableMotor, false);
_this.m_limitState = exports.b2LimitState.e_inactiveLimit;
return _this;
}
b2RevoluteJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// b2Rot qA(aA), qB(aB);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// m_rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// m_rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// J = [-I -r1_skew I r2_skew]
// [ 0 -1 0 1]
// r_skew = [-ry; rx]
// Matlab
// K = [ mA+r1y^2*iA+mB+r2y^2*iB, -r1y*iA*r1x-r2y*iB*r2x, -r1y*iA-r2y*iB]
// [ -r1y*iA*r1x-r2y*iB*r2x, mA+r1x^2*iA+mB+r2x^2*iB, r1x*iA+r2x*iB]
// [ -r1y*iA-r2y*iB, r1x*iA+r2x*iB, iA+iB]
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var fixedRotation = (iA + iB === 0);
this.m_mass.ex.x = mA + mB + this.m_rA.y * this.m_rA.y * iA + this.m_rB.y * this.m_rB.y * iB;
this.m_mass.ey.x = -this.m_rA.y * this.m_rA.x * iA - this.m_rB.y * this.m_rB.x * iB;
this.m_mass.ez.x = -this.m_rA.y * iA - this.m_rB.y * iB;
this.m_mass.ex.y = this.m_mass.ey.x;
this.m_mass.ey.y = mA + mB + this.m_rA.x * this.m_rA.x * iA + this.m_rB.x * this.m_rB.x * iB;
this.m_mass.ez.y = this.m_rA.x * iA + this.m_rB.x * iB;
this.m_mass.ex.z = this.m_mass.ez.x;
this.m_mass.ey.z = this.m_mass.ez.y;
this.m_mass.ez.z = iA + iB;
this.m_motorMass = iA + iB;
if (this.m_motorMass > 0) {
this.m_motorMass = 1 / this.m_motorMass;
}
if (!this.m_enableMotor || fixedRotation) {
this.m_motorImpulse = 0;
}
if (this.m_enableLimit && !fixedRotation) {
var jointAngle = aB - aA - this.m_referenceAngle;
if (b2Abs(this.m_upperAngle - this.m_lowerAngle) < 2 * b2_angularSlop) {
this.m_limitState = exports.b2LimitState.e_equalLimits;
}
else if (jointAngle <= this.m_lowerAngle) {
if (this.m_limitState !== exports.b2LimitState.e_atLowerLimit) {
this.m_impulse.z = 0;
}
this.m_limitState = exports.b2LimitState.e_atLowerLimit;
}
else if (jointAngle >= this.m_upperAngle) {
if (this.m_limitState !== exports.b2LimitState.e_atUpperLimit) {
this.m_impulse.z = 0;
}
this.m_limitState = exports.b2LimitState.e_atUpperLimit;
}
else {
this.m_limitState = exports.b2LimitState.e_inactiveLimit;
this.m_impulse.z = 0;
}
}
else {
this.m_limitState = exports.b2LimitState.e_inactiveLimit;
}
if (data.step.warmStarting) {
// Scale impulses to support a variable time step.
this.m_impulse.SelfMul(data.step.dtRatio);
this.m_motorImpulse *= data.step.dtRatio;
// b2Vec2 P(m_impulse.x, m_impulse.y);
var P = b2RevoluteJoint.InitVelocityConstraints_s_P.Set(this.m_impulse.x, this.m_impulse.y);
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * (b2Vec2.CrossVV(this.m_rA, P) + this.m_motorImpulse + this.m_impulse.z);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * (b2Vec2.CrossVV(this.m_rB, P) + this.m_motorImpulse + this.m_impulse.z);
}
else {
this.m_impulse.SetZero();
this.m_motorImpulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2RevoluteJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var fixedRotation = (iA + iB === 0);
// Solve motor constraint.
if (this.m_enableMotor && this.m_limitState !== exports.b2LimitState.e_equalLimits && !fixedRotation) {
var Cdot = wB - wA - this.m_motorSpeed;
var impulse = -this.m_motorMass * Cdot;
var oldImpulse = this.m_motorImpulse;
var maxImpulse = data.step.dt * this.m_maxMotorTorque;
this.m_motorImpulse = b2Clamp(this.m_motorImpulse + impulse, -maxImpulse, maxImpulse);
impulse = this.m_motorImpulse - oldImpulse;
wA -= iA * impulse;
wB += iB * impulse;
}
// Solve limit constraint.
if (this.m_enableLimit && this.m_limitState !== exports.b2LimitState.e_inactiveLimit && !fixedRotation) {
// b2Vec2 Cdot1 = vB + b2Cross(wB, m_rB) - vA - b2Cross(wA, m_rA);
var Cdot1 = b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2Vec2.s_t1), b2RevoluteJoint.SolveVelocityConstraints_s_Cdot1);
var Cdot2 = wB - wA;
// b2Vec3 Cdot(Cdot1.x, Cdot1.y, Cdot2);
// b2Vec3 impulse = -this.m_mass.Solve33(Cdot);
var impulse_v3 = this.m_mass.Solve33(Cdot1.x, Cdot1.y, Cdot2, b2RevoluteJoint.SolveVelocityConstraints_s_impulse_v3).SelfNeg();
if (this.m_limitState === exports.b2LimitState.e_equalLimits) {
this.m_impulse.SelfAdd(impulse_v3);
}
else if (this.m_limitState === exports.b2LimitState.e_atLowerLimit) {
var newImpulse = this.m_impulse.z + impulse_v3.z;
if (newImpulse < 0) {
// b2Vec2 rhs = -Cdot1 + m_impulse.z * b2Vec2(m_mass.ez.x, m_mass.ez.y);
var rhs_x = -Cdot1.x + this.m_impulse.z * this.m_mass.ez.x;
var rhs_y = -Cdot1.y + this.m_impulse.z * this.m_mass.ez.y;
var reduced_v2 = this.m_mass.Solve22(rhs_x, rhs_y, b2RevoluteJoint.SolveVelocityConstraints_s_reduced_v2);
impulse_v3.x = reduced_v2.x;
impulse_v3.y = reduced_v2.y;
impulse_v3.z = -this.m_impulse.z;
this.m_impulse.x += reduced_v2.x;
this.m_impulse.y += reduced_v2.y;
this.m_impulse.z = 0;
}
else {
this.m_impulse.SelfAdd(impulse_v3);
}
}
else if (this.m_limitState === exports.b2LimitState.e_atUpperLimit) {
var newImpulse = this.m_impulse.z + impulse_v3.z;
if (newImpulse > 0) {
// b2Vec2 rhs = -Cdot1 + m_impulse.z * b2Vec2(m_mass.ez.x, m_mass.ez.y);
var rhs_x = -Cdot1.x + this.m_impulse.z * this.m_mass.ez.x;
var rhs_y = -Cdot1.y + this.m_impulse.z * this.m_mass.ez.y;
var reduced_v2 = this.m_mass.Solve22(rhs_x, rhs_y, b2RevoluteJoint.SolveVelocityConstraints_s_reduced_v2);
impulse_v3.x = reduced_v2.x;
impulse_v3.y = reduced_v2.y;
impulse_v3.z = -this.m_impulse.z;
this.m_impulse.x += reduced_v2.x;
this.m_impulse.y += reduced_v2.y;
this.m_impulse.z = 0;
}
else {
this.m_impulse.SelfAdd(impulse_v3);
}
}
// b2Vec2 P(impulse.x, impulse.y);
var P = b2RevoluteJoint.SolveVelocityConstraints_s_P.Set(impulse_v3.x, impulse_v3.y);
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * (b2Vec2.CrossVV(this.m_rA, P) + impulse_v3.z);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * (b2Vec2.CrossVV(this.m_rB, P) + impulse_v3.z);
}
else {
// Solve point-to-point constraint
// b2Vec2 Cdot = vB + b2Cross(wB, m_rB) - vA - b2Cross(wA, m_rA);
var Cdot_v2 = b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2Vec2.s_t1), b2RevoluteJoint.SolveVelocityConstraints_s_Cdot_v2);
// b2Vec2 impulse = m_mass.Solve22(-Cdot);
var impulse_v2 = this.m_mass.Solve22(-Cdot_v2.x, -Cdot_v2.y, b2RevoluteJoint.SolveVelocityConstraints_s_impulse_v2);
this.m_impulse.x += impulse_v2.x;
this.m_impulse.y += impulse_v2.y;
// vA -= mA * impulse;
vA.SelfMulSub(mA, impulse_v2);
wA -= iA * b2Vec2.CrossVV(this.m_rA, impulse_v2);
// vB += mB * impulse;
vB.SelfMulAdd(mB, impulse_v2);
wB += iB * b2Vec2.CrossVV(this.m_rB, impulse_v2);
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2RevoluteJoint.prototype.SolvePositionConstraints = function (data) {
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
// b2Rot qA(aA), qB(aB);
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
var angularError = 0;
var positionError = 0;
var fixedRotation = (this.m_invIA + this.m_invIB === 0);
// Solve angular limit constraint.
if (this.m_enableLimit && this.m_limitState !== exports.b2LimitState.e_inactiveLimit && !fixedRotation) {
var angle = aB - aA - this.m_referenceAngle;
var limitImpulse = 0;
if (this.m_limitState === exports.b2LimitState.e_equalLimits) {
// Prevent large angular corrections
var C = b2Clamp(angle - this.m_lowerAngle, -b2_maxAngularCorrection, b2_maxAngularCorrection);
limitImpulse = -this.m_motorMass * C;
angularError = b2Abs(C);
}
else if (this.m_limitState === exports.b2LimitState.e_atLowerLimit) {
var C = angle - this.m_lowerAngle;
angularError = -C;
// Prevent large angular corrections and allow some slop.
C = b2Clamp(C + b2_angularSlop, -b2_maxAngularCorrection, 0);
limitImpulse = -this.m_motorMass * C;
}
else if (this.m_limitState === exports.b2LimitState.e_atUpperLimit) {
var C = angle - this.m_upperAngle;
angularError = C;
// Prevent large angular corrections and allow some slop.
C = b2Clamp(C - b2_angularSlop, 0, b2_maxAngularCorrection);
limitImpulse = -this.m_motorMass * C;
}
aA -= this.m_invIA * limitImpulse;
aB += this.m_invIB * limitImpulse;
}
// Solve point-to-point constraint.
{
qA.SetAngle(aA);
qB.SetAngle(aB);
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// b2Vec2 C = cB + rB - cA - rA;
var C_v2 = b2Vec2.SubVV(b2Vec2.AddVV(cB, rB, b2Vec2.s_t0), b2Vec2.AddVV(cA, rA, b2Vec2.s_t1), b2RevoluteJoint.SolvePositionConstraints_s_C_v2);
// positionError = C.Length();
positionError = C_v2.Length();
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var K = this.m_K;
K.ex.x = mA + mB + iA * rA.y * rA.y + iB * rB.y * rB.y;
K.ex.y = -iA * rA.x * rA.y - iB * rB.x * rB.y;
K.ey.x = K.ex.y;
K.ey.y = mA + mB + iA * rA.x * rA.x + iB * rB.x * rB.x;
// b2Vec2 impulse = -K.Solve(C);
var impulse = K.Solve(C_v2.x, C_v2.y, b2RevoluteJoint.SolvePositionConstraints_s_impulse).SelfNeg();
// cA -= mA * impulse;
cA.SelfMulSub(mA, impulse);
aA -= iA * b2Vec2.CrossVV(rA, impulse);
// cB += mB * impulse;
cB.SelfMulAdd(mB, impulse);
aB += iB * b2Vec2.CrossVV(rB, impulse);
}
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
return positionError <= b2_linearSlop && angularError <= b2_angularSlop;
};
b2RevoluteJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2RevoluteJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2RevoluteJoint.prototype.GetReactionForce = function (inv_dt, out) {
// b2Vec2 P(this.m_impulse.x, this.m_impulse.y);
// return inv_dt * P;
out.x = inv_dt * this.m_impulse.x;
out.y = inv_dt * this.m_impulse.y;
return out;
};
b2RevoluteJoint.prototype.GetReactionTorque = function (inv_dt) {
return inv_dt * this.m_impulse.z;
};
b2RevoluteJoint.prototype.GetLocalAnchorA = function () { return this.m_localAnchorA; };
b2RevoluteJoint.prototype.GetLocalAnchorB = function () { return this.m_localAnchorB; };
b2RevoluteJoint.prototype.GetReferenceAngle = function () { return this.m_referenceAngle; };
b2RevoluteJoint.prototype.GetJointAngle = function () {
// b2Body* bA = this.m_bodyA;
// b2Body* bB = this.m_bodyB;
// return bB->this.m_sweep.a - bA->this.m_sweep.a - this.m_referenceAngle;
return this.m_bodyB.m_sweep.a - this.m_bodyA.m_sweep.a - this.m_referenceAngle;
};
b2RevoluteJoint.prototype.GetJointSpeed = function () {
// b2Body* bA = this.m_bodyA;
// b2Body* bB = this.m_bodyB;
// return bB->this.m_angularVelocity - bA->this.m_angularVelocity;
return this.m_bodyB.m_angularVelocity - this.m_bodyA.m_angularVelocity;
};
b2RevoluteJoint.prototype.IsMotorEnabled = function () {
return this.m_enableMotor;
};
b2RevoluteJoint.prototype.EnableMotor = function (flag) {
if (flag !== this.m_enableMotor) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_enableMotor = flag;
}
};
b2RevoluteJoint.prototype.GetMotorTorque = function (inv_dt) {
return inv_dt * this.m_motorImpulse;
};
b2RevoluteJoint.prototype.GetMotorSpeed = function () {
return this.m_motorSpeed;
};
b2RevoluteJoint.prototype.SetMaxMotorTorque = function (torque) {
if (torque !== this.m_maxMotorTorque) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_maxMotorTorque = torque;
}
};
b2RevoluteJoint.prototype.GetMaxMotorTorque = function () { return this.m_maxMotorTorque; };
b2RevoluteJoint.prototype.IsLimitEnabled = function () {
return this.m_enableLimit;
};
b2RevoluteJoint.prototype.EnableLimit = function (flag) {
if (flag !== this.m_enableLimit) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_enableLimit = flag;
this.m_impulse.z = 0;
}
};
b2RevoluteJoint.prototype.GetLowerLimit = function () {
return this.m_lowerAngle;
};
b2RevoluteJoint.prototype.GetUpperLimit = function () {
return this.m_upperAngle;
};
b2RevoluteJoint.prototype.SetLimits = function (lower, upper) {
if (lower !== this.m_lowerAngle || upper !== this.m_upperAngle) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_impulse.z = 0;
this.m_lowerAngle = lower;
this.m_upperAngle = upper;
}
};
b2RevoluteJoint.prototype.SetMotorSpeed = function (speed) {
if (speed !== this.m_motorSpeed) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_motorSpeed = speed;
}
};
b2RevoluteJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2RevoluteJointDef = new b2RevoluteJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.referenceAngle = %.15f;\n", this.m_referenceAngle);
log(" jd.enableLimit = %s;\n", (this.m_enableLimit) ? ("true") : ("false"));
log(" jd.lowerAngle = %.15f;\n", this.m_lowerAngle);
log(" jd.upperAngle = %.15f;\n", this.m_upperAngle);
log(" jd.enableMotor = %s;\n", (this.m_enableMotor) ? ("true") : ("false"));
log(" jd.motorSpeed = %.15f;\n", this.m_motorSpeed);
log(" jd.maxMotorTorque = %.15f;\n", this.m_maxMotorTorque);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2RevoluteJoint.InitVelocityConstraints_s_P = new b2Vec2();
b2RevoluteJoint.SolveVelocityConstraints_s_P = new b2Vec2();
b2RevoluteJoint.SolveVelocityConstraints_s_Cdot_v2 = new b2Vec2();
b2RevoluteJoint.SolveVelocityConstraints_s_Cdot1 = new b2Vec2();
b2RevoluteJoint.SolveVelocityConstraints_s_impulse_v3 = new b2Vec3();
b2RevoluteJoint.SolveVelocityConstraints_s_reduced_v2 = new b2Vec2();
b2RevoluteJoint.SolveVelocityConstraints_s_impulse_v2 = new b2Vec2();
b2RevoluteJoint.SolvePositionConstraints_s_C_v2 = new b2Vec2();
b2RevoluteJoint.SolvePositionConstraints_s_impulse = new b2Vec2();
return b2RevoluteJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Rope joint definition. This requires two body anchor points and
/// a maximum lengths.
/// Note: by default the connected objects will not collide.
/// see collideConnected in b2JointDef.
var b2RopeJointDef = /** @class */ (function (_super) {
__extends(b2RopeJointDef, _super);
function b2RopeJointDef() {
var _this = _super.call(this, exports.b2JointType.e_ropeJoint) || this;
_this.localAnchorA = new b2Vec2(-1, 0);
_this.localAnchorB = new b2Vec2(1, 0);
_this.maxLength = 0;
return _this;
}
return b2RopeJointDef;
}(b2JointDef));
var b2RopeJoint = /** @class */ (function (_super) {
__extends(b2RopeJoint, _super);
function b2RopeJoint(def) {
var _this = _super.call(this, def) || this;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_maxLength = 0;
_this.m_length = 0;
_this.m_impulse = 0;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_u = new b2Vec2();
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_mass = 0;
_this.m_state = exports.b2LimitState.e_inactiveLimit;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_localAnchorA.Copy(b2Maybe(def.localAnchorA, new b2Vec2(-1, 0)));
_this.m_localAnchorB.Copy(b2Maybe(def.localAnchorB, new b2Vec2(1, 0)));
_this.m_maxLength = b2Maybe(def.maxLength, 0);
return _this;
}
b2RopeJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// this.m_rA = b2Mul(qA, this.m_localAnchorA - this.m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// this.m_rB = b2Mul(qB, this.m_localAnchorB - this.m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// this.m_u = cB + this.m_rB - cA - this.m_rA;
this.m_u.Copy(cB).SelfAdd(this.m_rB).SelfSub(cA).SelfSub(this.m_rA);
this.m_length = this.m_u.Length();
var C = this.m_length - this.m_maxLength;
if (C > 0) {
this.m_state = exports.b2LimitState.e_atUpperLimit;
}
else {
this.m_state = exports.b2LimitState.e_inactiveLimit;
}
if (this.m_length > b2_linearSlop) {
this.m_u.SelfMul(1 / this.m_length);
}
else {
this.m_u.SetZero();
this.m_mass = 0;
this.m_impulse = 0;
return;
}
// Compute effective mass.
var crA = b2Vec2.CrossVV(this.m_rA, this.m_u);
var crB = b2Vec2.CrossVV(this.m_rB, this.m_u);
var invMass = this.m_invMassA + this.m_invIA * crA * crA + this.m_invMassB + this.m_invIB * crB * crB;
this.m_mass = invMass !== 0 ? 1 / invMass : 0;
if (data.step.warmStarting) {
// Scale the impulse to support a variable time step.
this.m_impulse *= data.step.dtRatio;
// b2Vec2 P = m_impulse * m_u;
var P = b2Vec2.MulSV(this.m_impulse, this.m_u, b2RopeJoint.InitVelocityConstraints_s_P);
// vA -= m_invMassA * P;
vA.SelfMulSub(this.m_invMassA, P);
wA -= this.m_invIA * b2Vec2.CrossVV(this.m_rA, P);
// vB += m_invMassB * P;
vB.SelfMulAdd(this.m_invMassB, P);
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, P);
}
else {
this.m_impulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2RopeJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// Cdot = dot(u, v + cross(w, r))
// b2Vec2 vpA = vA + b2Cross(wA, m_rA);
var vpA = b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2RopeJoint.SolveVelocityConstraints_s_vpA);
// b2Vec2 vpB = vB + b2Cross(wB, m_rB);
var vpB = b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2RopeJoint.SolveVelocityConstraints_s_vpB);
// float32 C = m_length - m_maxLength;
var C = this.m_length - this.m_maxLength;
// float32 Cdot = b2Dot(m_u, vpB - vpA);
var Cdot = b2Vec2.DotVV(this.m_u, b2Vec2.SubVV(vpB, vpA, b2Vec2.s_t0));
// Predictive constraint.
if (C < 0) {
Cdot += data.step.inv_dt * C;
}
var impulse = -this.m_mass * Cdot;
var oldImpulse = this.m_impulse;
this.m_impulse = b2Min(0, this.m_impulse + impulse);
impulse = this.m_impulse - oldImpulse;
// b2Vec2 P = impulse * m_u;
var P = b2Vec2.MulSV(impulse, this.m_u, b2RopeJoint.SolveVelocityConstraints_s_P);
// vA -= m_invMassA * P;
vA.SelfMulSub(this.m_invMassA, P);
wA -= this.m_invIA * b2Vec2.CrossVV(this.m_rA, P);
// vB += m_invMassB * P;
vB.SelfMulAdd(this.m_invMassB, P);
wB += this.m_invIB * b2Vec2.CrossVV(this.m_rB, P);
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2RopeJoint.prototype.SolvePositionConstraints = function (data) {
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// b2Vec2 rA = b2Mul(qA, this.m_localAnchorA - this.m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, this.m_localAnchorB - this.m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// b2Vec2 u = cB + rB - cA - rA;
var u = this.m_u.Copy(cB).SelfAdd(rB).SelfSub(cA).SelfSub(rA);
var length = u.Normalize();
var C = length - this.m_maxLength;
C = b2Clamp(C, 0, b2_maxLinearCorrection);
var impulse = -this.m_mass * C;
// b2Vec2 P = impulse * u;
var P = b2Vec2.MulSV(impulse, u, b2RopeJoint.SolvePositionConstraints_s_P);
// cA -= m_invMassA * P;
cA.SelfMulSub(this.m_invMassA, P);
aA -= this.m_invIA * b2Vec2.CrossVV(rA, P);
// cB += m_invMassB * P;
cB.SelfMulAdd(this.m_invMassB, P);
aB += this.m_invIB * b2Vec2.CrossVV(rB, P);
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
return length - this.m_maxLength < b2_linearSlop;
};
b2RopeJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2RopeJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2RopeJoint.prototype.GetReactionForce = function (inv_dt, out) {
// return out.Set(inv_dt * this.m_linearImpulse.x, inv_dt * this.m_linearImpulse.y);
return b2Vec2.MulSV((inv_dt * this.m_impulse), this.m_u, out);
};
b2RopeJoint.prototype.GetReactionTorque = function (inv_dt) {
return 0;
};
b2RopeJoint.prototype.GetLocalAnchorA = function () { return this.m_localAnchorA; };
b2RopeJoint.prototype.GetLocalAnchorB = function () { return this.m_localAnchorB; };
b2RopeJoint.prototype.SetMaxLength = function (length) { this.m_maxLength = length; };
b2RopeJoint.prototype.GetMaxLength = function () {
return this.m_maxLength;
};
b2RopeJoint.prototype.GetLimitState = function () {
return this.m_state;
};
b2RopeJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2RopeJointDef = new b2RopeJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.maxLength = %.15f;\n", this.m_maxLength);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2RopeJoint.InitVelocityConstraints_s_P = new b2Vec2();
b2RopeJoint.SolveVelocityConstraints_s_vpA = new b2Vec2();
b2RopeJoint.SolveVelocityConstraints_s_vpB = new b2Vec2();
b2RopeJoint.SolveVelocityConstraints_s_P = new b2Vec2();
b2RopeJoint.SolvePositionConstraints_s_P = new b2Vec2();
return b2RopeJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Weld joint definition. You need to specify local anchor points
/// where they are attached and the relative body angle. The position
/// of the anchor points is important for computing the reaction torque.
var b2WeldJointDef = /** @class */ (function (_super) {
__extends(b2WeldJointDef, _super);
function b2WeldJointDef() {
var _this = _super.call(this, exports.b2JointType.e_weldJoint) || this;
_this.localAnchorA = new b2Vec2();
_this.localAnchorB = new b2Vec2();
_this.referenceAngle = 0;
_this.frequencyHz = 0;
_this.dampingRatio = 0;
return _this;
}
b2WeldJointDef.prototype.Initialize = function (bA, bB, anchor) {
this.bodyA = bA;
this.bodyB = bB;
this.bodyA.GetLocalPoint(anchor, this.localAnchorA);
this.bodyB.GetLocalPoint(anchor, this.localAnchorB);
this.referenceAngle = this.bodyB.GetAngle() - this.bodyA.GetAngle();
};
return b2WeldJointDef;
}(b2JointDef));
var b2WeldJoint = /** @class */ (function (_super) {
__extends(b2WeldJoint, _super);
function b2WeldJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_frequencyHz = 0;
_this.m_dampingRatio = 0;
_this.m_bias = 0;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_referenceAngle = 0;
_this.m_gamma = 0;
_this.m_impulse = new b2Vec3(0, 0, 0);
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_mass = new b2Mat33();
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_K = new b2Mat33();
_this.m_frequencyHz = b2Maybe(def.frequencyHz, 0);
_this.m_dampingRatio = b2Maybe(def.dampingRatio, 0);
_this.m_localAnchorA.Copy(b2Maybe(def.localAnchorA, b2Vec2.ZERO));
_this.m_localAnchorB.Copy(b2Maybe(def.localAnchorB, b2Vec2.ZERO));
_this.m_referenceAngle = b2Maybe(def.referenceAngle, 0);
_this.m_impulse.SetZero();
return _this;
}
b2WeldJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// m_rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// m_rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// J = [-I -r1_skew I r2_skew]
// [ 0 -1 0 1]
// r_skew = [-ry; rx]
// Matlab
// K = [ mA+r1y^2*iA+mB+r2y^2*iB, -r1y*iA*r1x-r2y*iB*r2x, -r1y*iA-r2y*iB]
// [ -r1y*iA*r1x-r2y*iB*r2x, mA+r1x^2*iA+mB+r2x^2*iB, r1x*iA+r2x*iB]
// [ -r1y*iA-r2y*iB, r1x*iA+r2x*iB, iA+iB]
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var K = this.m_K;
K.ex.x = mA + mB + this.m_rA.y * this.m_rA.y * iA + this.m_rB.y * this.m_rB.y * iB;
K.ey.x = -this.m_rA.y * this.m_rA.x * iA - this.m_rB.y * this.m_rB.x * iB;
K.ez.x = -this.m_rA.y * iA - this.m_rB.y * iB;
K.ex.y = K.ey.x;
K.ey.y = mA + mB + this.m_rA.x * this.m_rA.x * iA + this.m_rB.x * this.m_rB.x * iB;
K.ez.y = this.m_rA.x * iA + this.m_rB.x * iB;
K.ex.z = K.ez.x;
K.ey.z = K.ez.y;
K.ez.z = iA + iB;
if (this.m_frequencyHz > 0) {
K.GetInverse22(this.m_mass);
var invM = iA + iB;
var m = invM > 0 ? 1 / invM : 0;
var C = aB - aA - this.m_referenceAngle;
// Frequency
var omega = 2 * b2_pi * this.m_frequencyHz;
// Damping coefficient
var d = 2 * m * this.m_dampingRatio * omega;
// Spring stiffness
var k = m * omega * omega;
// magic formulas
var h = data.step.dt;
this.m_gamma = h * (d + h * k);
this.m_gamma = this.m_gamma !== 0 ? 1 / this.m_gamma : 0;
this.m_bias = C * h * k * this.m_gamma;
invM += this.m_gamma;
this.m_mass.ez.z = invM !== 0 ? 1 / invM : 0;
}
else {
K.GetSymInverse33(this.m_mass);
this.m_gamma = 0;
this.m_bias = 0;
}
if (data.step.warmStarting) {
// Scale impulses to support a variable time step.
this.m_impulse.SelfMul(data.step.dtRatio);
// b2Vec2 P(m_impulse.x, m_impulse.y);
var P = b2WeldJoint.InitVelocityConstraints_s_P.Set(this.m_impulse.x, this.m_impulse.y);
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * (b2Vec2.CrossVV(this.m_rA, P) + this.m_impulse.z);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * (b2Vec2.CrossVV(this.m_rB, P) + this.m_impulse.z);
}
else {
this.m_impulse.SetZero();
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2WeldJoint.prototype.SolveVelocityConstraints = function (data) {
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
if (this.m_frequencyHz > 0) {
var Cdot2 = wB - wA;
var impulse2 = -this.m_mass.ez.z * (Cdot2 + this.m_bias + this.m_gamma * this.m_impulse.z);
this.m_impulse.z += impulse2;
wA -= iA * impulse2;
wB += iB * impulse2;
// b2Vec2 Cdot1 = vB + b2Vec2.CrossSV(wB, this.m_rB) - vA - b2Vec2.CrossSV(wA, this.m_rA);
var Cdot1 = b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2Vec2.s_t1), b2WeldJoint.SolveVelocityConstraints_s_Cdot1);
// b2Vec2 impulse1 = -b2Mul22(m_mass, Cdot1);
var impulse1 = b2Mat33.MulM33XY(this.m_mass, Cdot1.x, Cdot1.y, b2WeldJoint.SolveVelocityConstraints_s_impulse1).SelfNeg();
this.m_impulse.x += impulse1.x;
this.m_impulse.y += impulse1.y;
// b2Vec2 P = impulse1;
var P = impulse1;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
// wA -= iA * b2Cross(m_rA, P);
wA -= iA * b2Vec2.CrossVV(this.m_rA, P);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
// wB += iB * b2Cross(m_rB, P);
wB += iB * b2Vec2.CrossVV(this.m_rB, P);
}
else {
// b2Vec2 Cdot1 = vB + b2Cross(wB, this.m_rB) - vA - b2Cross(wA, this.m_rA);
var Cdot1 = b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, this.m_rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, this.m_rA, b2Vec2.s_t1), b2WeldJoint.SolveVelocityConstraints_s_Cdot1);
var Cdot2 = wB - wA;
// b2Vec3 const Cdot(Cdot1.x, Cdot1.y, Cdot2);
// b2Vec3 impulse = -b2Mul(m_mass, Cdot);
var impulse = b2Mat33.MulM33XYZ(this.m_mass, Cdot1.x, Cdot1.y, Cdot2, b2WeldJoint.SolveVelocityConstraints_s_impulse).SelfNeg();
this.m_impulse.SelfAdd(impulse);
// b2Vec2 P(impulse.x, impulse.y);
var P = b2WeldJoint.SolveVelocityConstraints_s_P.Set(impulse.x, impulse.y);
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * (b2Vec2.CrossVV(this.m_rA, P) + impulse.z);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * (b2Vec2.CrossVV(this.m_rB, P) + impulse.z);
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2WeldJoint.prototype.SolvePositionConstraints = function (data) {
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
var positionError, angularError;
var K = this.m_K;
K.ex.x = mA + mB + rA.y * rA.y * iA + rB.y * rB.y * iB;
K.ey.x = -rA.y * rA.x * iA - rB.y * rB.x * iB;
K.ez.x = -rA.y * iA - rB.y * iB;
K.ex.y = K.ey.x;
K.ey.y = mA + mB + rA.x * rA.x * iA + rB.x * rB.x * iB;
K.ez.y = rA.x * iA + rB.x * iB;
K.ex.z = K.ez.x;
K.ey.z = K.ez.y;
K.ez.z = iA + iB;
if (this.m_frequencyHz > 0) {
// b2Vec2 C1 = cB + rB - cA - rA;
var C1 = b2Vec2.SubVV(b2Vec2.AddVV(cB, rB, b2Vec2.s_t0), b2Vec2.AddVV(cA, rA, b2Vec2.s_t1), b2WeldJoint.SolvePositionConstraints_s_C1);
positionError = C1.Length();
angularError = 0;
// b2Vec2 P = -K.Solve22(C1);
var P = K.Solve22(C1.x, C1.y, b2WeldJoint.SolvePositionConstraints_s_P).SelfNeg();
// cA -= mA * P;
cA.SelfMulSub(mA, P);
aA -= iA * b2Vec2.CrossVV(rA, P);
// cB += mB * P;
cB.SelfMulAdd(mB, P);
aB += iB * b2Vec2.CrossVV(rB, P);
}
else {
// b2Vec2 C1 = cB + rB - cA - rA;
var C1 = b2Vec2.SubVV(b2Vec2.AddVV(cB, rB, b2Vec2.s_t0), b2Vec2.AddVV(cA, rA, b2Vec2.s_t1), b2WeldJoint.SolvePositionConstraints_s_C1);
var C2 = aB - aA - this.m_referenceAngle;
positionError = C1.Length();
angularError = b2Abs(C2);
// b2Vec3 C(C1.x, C1.y, C2);
// b2Vec3 impulse = -K.Solve33(C);
var impulse = K.Solve33(C1.x, C1.y, C2, b2WeldJoint.SolvePositionConstraints_s_impulse).SelfNeg();
// b2Vec2 P(impulse.x, impulse.y);
var P = b2WeldJoint.SolvePositionConstraints_s_P.Set(impulse.x, impulse.y);
// cA -= mA * P;
cA.SelfMulSub(mA, P);
aA -= iA * (b2Vec2.CrossVV(this.m_rA, P) + impulse.z);
// cB += mB * P;
cB.SelfMulAdd(mB, P);
aB += iB * (b2Vec2.CrossVV(this.m_rB, P) + impulse.z);
}
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
return positionError <= b2_linearSlop && angularError <= b2_angularSlop;
};
b2WeldJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2WeldJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2WeldJoint.prototype.GetReactionForce = function (inv_dt, out) {
// b2Vec2 P(this.m_impulse.x, this.m_impulse.y);
// return inv_dt * P;
out.x = inv_dt * this.m_impulse.x;
out.y = inv_dt * this.m_impulse.y;
return out;
};
b2WeldJoint.prototype.GetReactionTorque = function (inv_dt) {
return inv_dt * this.m_impulse.z;
};
b2WeldJoint.prototype.GetLocalAnchorA = function () { return this.m_localAnchorA; };
b2WeldJoint.prototype.GetLocalAnchorB = function () { return this.m_localAnchorB; };
b2WeldJoint.prototype.GetReferenceAngle = function () { return this.m_referenceAngle; };
b2WeldJoint.prototype.SetFrequency = function (hz) { this.m_frequencyHz = hz; };
b2WeldJoint.prototype.GetFrequency = function () { return this.m_frequencyHz; };
b2WeldJoint.prototype.SetDampingRatio = function (ratio) { this.m_dampingRatio = ratio; };
b2WeldJoint.prototype.GetDampingRatio = function () { return this.m_dampingRatio; };
b2WeldJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2WeldJointDef = new b2WeldJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.referenceAngle = %.15f;\n", this.m_referenceAngle);
log(" jd.frequencyHz = %.15f;\n", this.m_frequencyHz);
log(" jd.dampingRatio = %.15f;\n", this.m_dampingRatio);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2WeldJoint.InitVelocityConstraints_s_P = new b2Vec2();
b2WeldJoint.SolveVelocityConstraints_s_Cdot1 = new b2Vec2();
b2WeldJoint.SolveVelocityConstraints_s_impulse1 = new b2Vec2();
b2WeldJoint.SolveVelocityConstraints_s_impulse = new b2Vec3();
b2WeldJoint.SolveVelocityConstraints_s_P = new b2Vec2();
b2WeldJoint.SolvePositionConstraints_s_C1 = new b2Vec2();
b2WeldJoint.SolvePositionConstraints_s_P = new b2Vec2();
b2WeldJoint.SolvePositionConstraints_s_impulse = new b2Vec3();
return b2WeldJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Wheel joint definition. This requires defining a line of
/// motion using an axis and an anchor point. The definition uses local
/// anchor points and a local axis so that the initial configuration
/// can violate the constraint slightly. The joint translation is zero
/// when the local anchor points coincide in world space. Using local
/// anchors and a local axis helps when saving and loading a game.
var b2WheelJointDef = /** @class */ (function (_super) {
__extends(b2WheelJointDef, _super);
function b2WheelJointDef() {
var _this = _super.call(this, exports.b2JointType.e_wheelJoint) || this;
_this.localAnchorA = new b2Vec2(0, 0);
_this.localAnchorB = new b2Vec2(0, 0);
_this.localAxisA = new b2Vec2(1, 0);
_this.enableMotor = false;
_this.maxMotorTorque = 0;
_this.motorSpeed = 0;
_this.frequencyHz = 2;
_this.dampingRatio = 0.7;
return _this;
}
b2WheelJointDef.prototype.Initialize = function (bA, bB, anchor, axis) {
this.bodyA = bA;
this.bodyB = bB;
this.bodyA.GetLocalPoint(anchor, this.localAnchorA);
this.bodyB.GetLocalPoint(anchor, this.localAnchorB);
this.bodyA.GetLocalVector(axis, this.localAxisA);
};
return b2WheelJointDef;
}(b2JointDef));
var b2WheelJoint = /** @class */ (function (_super) {
__extends(b2WheelJoint, _super);
function b2WheelJoint(def) {
var _this = _super.call(this, def) || this;
_this.m_frequencyHz = 0;
_this.m_dampingRatio = 0;
// Solver shared
_this.m_localAnchorA = new b2Vec2();
_this.m_localAnchorB = new b2Vec2();
_this.m_localXAxisA = new b2Vec2();
_this.m_localYAxisA = new b2Vec2();
_this.m_impulse = 0;
_this.m_motorImpulse = 0;
_this.m_springImpulse = 0;
_this.m_maxMotorTorque = 0;
_this.m_motorSpeed = 0;
_this.m_enableMotor = false;
// Solver temp
_this.m_indexA = 0;
_this.m_indexB = 0;
_this.m_localCenterA = new b2Vec2();
_this.m_localCenterB = new b2Vec2();
_this.m_invMassA = 0;
_this.m_invMassB = 0;
_this.m_invIA = 0;
_this.m_invIB = 0;
_this.m_ax = new b2Vec2();
_this.m_ay = new b2Vec2();
_this.m_sAx = 0;
_this.m_sBx = 0;
_this.m_sAy = 0;
_this.m_sBy = 0;
_this.m_mass = 0;
_this.m_motorMass = 0;
_this.m_springMass = 0;
_this.m_bias = 0;
_this.m_gamma = 0;
_this.m_qA = new b2Rot();
_this.m_qB = new b2Rot();
_this.m_lalcA = new b2Vec2();
_this.m_lalcB = new b2Vec2();
_this.m_rA = new b2Vec2();
_this.m_rB = new b2Vec2();
_this.m_frequencyHz = b2Maybe(def.frequencyHz, 2);
_this.m_dampingRatio = b2Maybe(def.dampingRatio, 0.7);
_this.m_localAnchorA.Copy(b2Maybe(def.localAnchorA, b2Vec2.ZERO));
_this.m_localAnchorB.Copy(b2Maybe(def.localAnchorB, b2Vec2.ZERO));
_this.m_localXAxisA.Copy(b2Maybe(def.localAxisA, b2Vec2.UNITX));
b2Vec2.CrossOneV(_this.m_localXAxisA, _this.m_localYAxisA);
_this.m_maxMotorTorque = b2Maybe(def.maxMotorTorque, 0);
_this.m_motorSpeed = b2Maybe(def.motorSpeed, 0);
_this.m_enableMotor = b2Maybe(def.enableMotor, false);
_this.m_ax.SetZero();
_this.m_ay.SetZero();
return _this;
}
b2WheelJoint.prototype.GetMotorSpeed = function () {
return this.m_motorSpeed;
};
b2WheelJoint.prototype.GetMaxMotorTorque = function () {
return this.m_maxMotorTorque;
};
b2WheelJoint.prototype.SetSpringFrequencyHz = function (hz) {
this.m_frequencyHz = hz;
};
b2WheelJoint.prototype.GetSpringFrequencyHz = function () {
return this.m_frequencyHz;
};
b2WheelJoint.prototype.SetSpringDampingRatio = function (ratio) {
this.m_dampingRatio = ratio;
};
b2WheelJoint.prototype.GetSpringDampingRatio = function () {
return this.m_dampingRatio;
};
b2WheelJoint.prototype.InitVelocityConstraints = function (data) {
this.m_indexA = this.m_bodyA.m_islandIndex;
this.m_indexB = this.m_bodyB.m_islandIndex;
this.m_localCenterA.Copy(this.m_bodyA.m_sweep.localCenter);
this.m_localCenterB.Copy(this.m_bodyB.m_sweep.localCenter);
this.m_invMassA = this.m_bodyA.m_invMass;
this.m_invMassB = this.m_bodyB.m_invMass;
this.m_invIA = this.m_bodyA.m_invI;
this.m_invIB = this.m_bodyB.m_invI;
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// Compute the effective masses.
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// b2Vec2 d = cB + rB - cA - rA;
var d = b2Vec2.SubVV(b2Vec2.AddVV(cB, rB, b2Vec2.s_t0), b2Vec2.AddVV(cA, rA, b2Vec2.s_t1), b2WheelJoint.InitVelocityConstraints_s_d);
// Point to line constraint
{
// m_ay = b2Mul(qA, m_localYAxisA);
b2Rot.MulRV(qA, this.m_localYAxisA, this.m_ay);
// m_sAy = b2Cross(d + rA, m_ay);
this.m_sAy = b2Vec2.CrossVV(b2Vec2.AddVV(d, rA, b2Vec2.s_t0), this.m_ay);
// m_sBy = b2Cross(rB, m_ay);
this.m_sBy = b2Vec2.CrossVV(rB, this.m_ay);
this.m_mass = mA + mB + iA * this.m_sAy * this.m_sAy + iB * this.m_sBy * this.m_sBy;
if (this.m_mass > 0) {
this.m_mass = 1 / this.m_mass;
}
}
// Spring constraint
this.m_springMass = 0;
this.m_bias = 0;
this.m_gamma = 0;
if (this.m_frequencyHz > 0) {
// m_ax = b2Mul(qA, m_localXAxisA);
b2Rot.MulRV(qA, this.m_localXAxisA, this.m_ax);
// m_sAx = b2Cross(d + rA, m_ax);
this.m_sAx = b2Vec2.CrossVV(b2Vec2.AddVV(d, rA, b2Vec2.s_t0), this.m_ax);
// m_sBx = b2Cross(rB, m_ax);
this.m_sBx = b2Vec2.CrossVV(rB, this.m_ax);
var invMass = mA + mB + iA * this.m_sAx * this.m_sAx + iB * this.m_sBx * this.m_sBx;
if (invMass > 0) {
this.m_springMass = 1 / invMass;
var C = b2Vec2.DotVV(d, this.m_ax);
// Frequency
var omega = 2 * b2_pi * this.m_frequencyHz;
// Damping coefficient
var damp = 2 * this.m_springMass * this.m_dampingRatio * omega;
// Spring stiffness
var k = this.m_springMass * omega * omega;
// magic formulas
var h = data.step.dt;
this.m_gamma = h * (damp + h * k);
if (this.m_gamma > 0) {
this.m_gamma = 1 / this.m_gamma;
}
this.m_bias = C * h * k * this.m_gamma;
this.m_springMass = invMass + this.m_gamma;
if (this.m_springMass > 0) {
this.m_springMass = 1 / this.m_springMass;
}
}
}
else {
this.m_springImpulse = 0;
}
// Rotational motor
if (this.m_enableMotor) {
this.m_motorMass = iA + iB;
if (this.m_motorMass > 0) {
this.m_motorMass = 1 / this.m_motorMass;
}
}
else {
this.m_motorMass = 0;
this.m_motorImpulse = 0;
}
if (data.step.warmStarting) {
// Account for variable time step.
this.m_impulse *= data.step.dtRatio;
this.m_springImpulse *= data.step.dtRatio;
this.m_motorImpulse *= data.step.dtRatio;
// b2Vec2 P = m_impulse * m_ay + m_springImpulse * m_ax;
var P = b2Vec2.AddVV(b2Vec2.MulSV(this.m_impulse, this.m_ay, b2Vec2.s_t0), b2Vec2.MulSV(this.m_springImpulse, this.m_ax, b2Vec2.s_t1), b2WheelJoint.InitVelocityConstraints_s_P);
// float32 LA = m_impulse * m_sAy + m_springImpulse * m_sAx + m_motorImpulse;
var LA = this.m_impulse * this.m_sAy + this.m_springImpulse * this.m_sAx + this.m_motorImpulse;
// float32 LB = m_impulse * m_sBy + m_springImpulse * m_sBx + m_motorImpulse;
var LB = this.m_impulse * this.m_sBy + this.m_springImpulse * this.m_sBx + this.m_motorImpulse;
// vA -= m_invMassA * P;
vA.SelfMulSub(this.m_invMassA, P);
wA -= this.m_invIA * LA;
// vB += m_invMassB * P;
vB.SelfMulAdd(this.m_invMassB, P);
wB += this.m_invIB * LB;
}
else {
this.m_impulse = 0;
this.m_springImpulse = 0;
this.m_motorImpulse = 0;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2WheelJoint.prototype.SolveVelocityConstraints = function (data) {
var mA = this.m_invMassA, mB = this.m_invMassB;
var iA = this.m_invIA, iB = this.m_invIB;
var vA = data.velocities[this.m_indexA].v;
var wA = data.velocities[this.m_indexA].w;
var vB = data.velocities[this.m_indexB].v;
var wB = data.velocities[this.m_indexB].w;
// Solve spring constraint
{
var Cdot = b2Vec2.DotVV(this.m_ax, b2Vec2.SubVV(vB, vA, b2Vec2.s_t0)) + this.m_sBx * wB - this.m_sAx * wA;
var impulse = -this.m_springMass * (Cdot + this.m_bias + this.m_gamma * this.m_springImpulse);
this.m_springImpulse += impulse;
// b2Vec2 P = impulse * m_ax;
var P = b2Vec2.MulSV(impulse, this.m_ax, b2WheelJoint.SolveVelocityConstraints_s_P);
var LA = impulse * this.m_sAx;
var LB = impulse * this.m_sBx;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * LA;
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * LB;
}
// Solve rotational motor constraint
{
var Cdot = wB - wA - this.m_motorSpeed;
var impulse = -this.m_motorMass * Cdot;
var oldImpulse = this.m_motorImpulse;
var maxImpulse = data.step.dt * this.m_maxMotorTorque;
this.m_motorImpulse = b2Clamp(this.m_motorImpulse + impulse, -maxImpulse, maxImpulse);
impulse = this.m_motorImpulse - oldImpulse;
wA -= iA * impulse;
wB += iB * impulse;
}
// Solve point to line constraint
{
var Cdot = b2Vec2.DotVV(this.m_ay, b2Vec2.SubVV(vB, vA, b2Vec2.s_t0)) + this.m_sBy * wB - this.m_sAy * wA;
var impulse = -this.m_mass * Cdot;
this.m_impulse += impulse;
// b2Vec2 P = impulse * m_ay;
var P = b2Vec2.MulSV(impulse, this.m_ay, b2WheelJoint.SolveVelocityConstraints_s_P);
var LA = impulse * this.m_sAy;
var LB = impulse * this.m_sBy;
// vA -= mA * P;
vA.SelfMulSub(mA, P);
wA -= iA * LA;
// vB += mB * P;
vB.SelfMulAdd(mB, P);
wB += iB * LB;
}
// data.velocities[this.m_indexA].v = vA;
data.velocities[this.m_indexA].w = wA;
// data.velocities[this.m_indexB].v = vB;
data.velocities[this.m_indexB].w = wB;
};
b2WheelJoint.prototype.SolvePositionConstraints = function (data) {
var cA = data.positions[this.m_indexA].c;
var aA = data.positions[this.m_indexA].a;
var cB = data.positions[this.m_indexB].c;
var aB = data.positions[this.m_indexB].a;
var qA = this.m_qA.SetAngle(aA), qB = this.m_qB.SetAngle(aB);
// b2Vec2 rA = b2Mul(qA, m_localAnchorA - m_localCenterA);
b2Vec2.SubVV(this.m_localAnchorA, this.m_localCenterA, this.m_lalcA);
var rA = b2Rot.MulRV(qA, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(qB, m_localAnchorB - m_localCenterB);
b2Vec2.SubVV(this.m_localAnchorB, this.m_localCenterB, this.m_lalcB);
var rB = b2Rot.MulRV(qB, this.m_lalcB, this.m_rB);
// b2Vec2 d = (cB - cA) + rB - rA;
var d = b2Vec2.AddVV(b2Vec2.SubVV(cB, cA, b2Vec2.s_t0), b2Vec2.SubVV(rB, rA, b2Vec2.s_t1), b2WheelJoint.SolvePositionConstraints_s_d);
// b2Vec2 ay = b2Mul(qA, m_localYAxisA);
var ay = b2Rot.MulRV(qA, this.m_localYAxisA, this.m_ay);
// float32 sAy = b2Cross(d + rA, ay);
var sAy = b2Vec2.CrossVV(b2Vec2.AddVV(d, rA, b2Vec2.s_t0), ay);
// float32 sBy = b2Cross(rB, ay);
var sBy = b2Vec2.CrossVV(rB, ay);
// float32 C = b2Dot(d, ay);
var C = b2Vec2.DotVV(d, this.m_ay);
var k = this.m_invMassA + this.m_invMassB + this.m_invIA * this.m_sAy * this.m_sAy + this.m_invIB * this.m_sBy * this.m_sBy;
var impulse;
if (k !== 0) {
impulse = -C / k;
}
else {
impulse = 0;
}
// b2Vec2 P = impulse * ay;
var P = b2Vec2.MulSV(impulse, ay, b2WheelJoint.SolvePositionConstraints_s_P);
var LA = impulse * sAy;
var LB = impulse * sBy;
// cA -= m_invMassA * P;
cA.SelfMulSub(this.m_invMassA, P);
aA -= this.m_invIA * LA;
// cB += m_invMassB * P;
cB.SelfMulAdd(this.m_invMassB, P);
aB += this.m_invIB * LB;
// data.positions[this.m_indexA].c = cA;
data.positions[this.m_indexA].a = aA;
// data.positions[this.m_indexB].c = cB;
data.positions[this.m_indexB].a = aB;
return b2Abs(C) <= b2_linearSlop;
};
b2WheelJoint.prototype.GetDefinition = function (def) {
// DEBUG: b2Assert(false); // TODO
return def;
};
b2WheelJoint.prototype.GetAnchorA = function (out) {
return this.m_bodyA.GetWorldPoint(this.m_localAnchorA, out);
};
b2WheelJoint.prototype.GetAnchorB = function (out) {
return this.m_bodyB.GetWorldPoint(this.m_localAnchorB, out);
};
b2WheelJoint.prototype.GetReactionForce = function (inv_dt, out) {
// return inv_dt * (m_impulse * m_ay + m_springImpulse * m_ax);
out.x = inv_dt * (this.m_impulse * this.m_ay.x + this.m_springImpulse * this.m_ax.x);
out.y = inv_dt * (this.m_impulse * this.m_ay.y + this.m_springImpulse * this.m_ax.y);
return out;
};
b2WheelJoint.prototype.GetReactionTorque = function (inv_dt) {
return inv_dt * this.m_motorImpulse;
};
b2WheelJoint.prototype.GetLocalAnchorA = function () { return this.m_localAnchorA; };
b2WheelJoint.prototype.GetLocalAnchorB = function () { return this.m_localAnchorB; };
b2WheelJoint.prototype.GetLocalAxisA = function () { return this.m_localXAxisA; };
b2WheelJoint.prototype.GetJointTranslation = function () {
return this.GetPrismaticJointTranslation();
};
b2WheelJoint.prototype.GetJointLinearSpeed = function () {
return this.GetPrismaticJointSpeed();
};
b2WheelJoint.prototype.GetJointAngle = function () {
return this.GetRevoluteJointAngle();
};
b2WheelJoint.prototype.GetJointAngularSpeed = function () {
return this.GetRevoluteJointSpeed();
};
b2WheelJoint.prototype.GetPrismaticJointTranslation = function () {
var bA = this.m_bodyA;
var bB = this.m_bodyB;
var pA = bA.GetWorldPoint(this.m_localAnchorA, new b2Vec2());
var pB = bB.GetWorldPoint(this.m_localAnchorB, new b2Vec2());
var d = b2Vec2.SubVV(pB, pA, new b2Vec2());
var axis = bA.GetWorldVector(this.m_localXAxisA, new b2Vec2());
var translation = b2Vec2.DotVV(d, axis);
return translation;
};
b2WheelJoint.prototype.GetPrismaticJointSpeed = function () {
var bA = this.m_bodyA;
var bB = this.m_bodyB;
// b2Vec2 rA = b2Mul(bA->m_xf.q, m_localAnchorA - bA->m_sweep.localCenter);
b2Vec2.SubVV(this.m_localAnchorA, bA.m_sweep.localCenter, this.m_lalcA);
var rA = b2Rot.MulRV(bA.m_xf.q, this.m_lalcA, this.m_rA);
// b2Vec2 rB = b2Mul(bB->m_xf.q, m_localAnchorB - bB->m_sweep.localCenter);
b2Vec2.SubVV(this.m_localAnchorB, bB.m_sweep.localCenter, this.m_lalcB);
var rB = b2Rot.MulRV(bB.m_xf.q, this.m_lalcB, this.m_rB);
// b2Vec2 pA = bA->m_sweep.c + rA;
var pA = b2Vec2.AddVV(bA.m_sweep.c, rA, b2Vec2.s_t0); // pA uses s_t0
// b2Vec2 pB = bB->m_sweep.c + rB;
var pB = b2Vec2.AddVV(bB.m_sweep.c, rB, b2Vec2.s_t1); // pB uses s_t1
// b2Vec2 d = pB - pA;
var d = b2Vec2.SubVV(pB, pA, b2Vec2.s_t2); // d uses s_t2
// b2Vec2 axis = b2Mul(bA.m_xf.q, m_localXAxisA);
var axis = bA.GetWorldVector(this.m_localXAxisA, new b2Vec2());
var vA = bA.m_linearVelocity;
var vB = bB.m_linearVelocity;
var wA = bA.m_angularVelocity;
var wB = bB.m_angularVelocity;
// float32 speed = b2Dot(d, b2Cross(wA, axis)) + b2Dot(axis, vB + b2Cross(wB, rB) - vA - b2Cross(wA, rA));
var speed = b2Vec2.DotVV(d, b2Vec2.CrossSV(wA, axis, b2Vec2.s_t0)) +
b2Vec2.DotVV(axis, b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, rA, b2Vec2.s_t1), b2Vec2.s_t0));
return speed;
};
b2WheelJoint.prototype.GetRevoluteJointAngle = function () {
// b2Body* bA = this.m_bodyA;
// b2Body* bB = this.m_bodyB;
// return bB->this.m_sweep.a - bA->this.m_sweep.a;
return this.m_bodyB.m_sweep.a - this.m_bodyA.m_sweep.a;
};
b2WheelJoint.prototype.GetRevoluteJointSpeed = function () {
var wA = this.m_bodyA.m_angularVelocity;
var wB = this.m_bodyB.m_angularVelocity;
return wB - wA;
};
b2WheelJoint.prototype.IsMotorEnabled = function () {
return this.m_enableMotor;
};
b2WheelJoint.prototype.EnableMotor = function (flag) {
if (flag !== this.m_enableMotor) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_enableMotor = flag;
}
};
b2WheelJoint.prototype.SetMotorSpeed = function (speed) {
if (speed !== this.m_motorSpeed) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_motorSpeed = speed;
}
};
b2WheelJoint.prototype.SetMaxMotorTorque = function (force) {
if (force !== this.m_maxMotorTorque) {
this.m_bodyA.SetAwake(true);
this.m_bodyB.SetAwake(true);
this.m_maxMotorTorque = force;
}
};
b2WheelJoint.prototype.GetMotorTorque = function (inv_dt) {
return inv_dt * this.m_motorImpulse;
};
b2WheelJoint.prototype.Dump = function (log) {
var indexA = this.m_bodyA.m_islandIndex;
var indexB = this.m_bodyB.m_islandIndex;
log(" const jd: b2WheelJointDef = new b2WheelJointDef();\n");
log(" jd.bodyA = bodies[%d];\n", indexA);
log(" jd.bodyB = bodies[%d];\n", indexB);
log(" jd.collideConnected = %s;\n", (this.m_collideConnected) ? ("true") : ("false"));
log(" jd.localAnchorA.Set(%.15f, %.15f);\n", this.m_localAnchorA.x, this.m_localAnchorA.y);
log(" jd.localAnchorB.Set(%.15f, %.15f);\n", this.m_localAnchorB.x, this.m_localAnchorB.y);
log(" jd.localAxisA.Set(%.15f, %.15f);\n", this.m_localXAxisA.x, this.m_localXAxisA.y);
log(" jd.enableMotor = %s;\n", (this.m_enableMotor) ? ("true") : ("false"));
log(" jd.motorSpeed = %.15f;\n", this.m_motorSpeed);
log(" jd.maxMotorTorque = %.15f;\n", this.m_maxMotorTorque);
log(" jd.frequencyHz = %.15f;\n", this.m_frequencyHz);
log(" jd.dampingRatio = %.15f;\n", this.m_dampingRatio);
log(" joints[%d] = this.m_world.CreateJoint(jd);\n", this.m_index);
};
b2WheelJoint.InitVelocityConstraints_s_d = new b2Vec2();
b2WheelJoint.InitVelocityConstraints_s_P = new b2Vec2();
b2WheelJoint.SolveVelocityConstraints_s_P = new b2Vec2();
b2WheelJoint.SolvePositionConstraints_s_d = new b2Vec2();
b2WheelJoint.SolvePositionConstraints_s_P = new b2Vec2();
return b2WheelJoint;
}(b2Joint));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Friction mixing law. The idea is to allow either fixture to drive the friction to zero.
/// For example, anything slides on ice.
function b2MixFriction(friction1, friction2) {
return b2Sqrt(friction1 * friction2);
}
/// Restitution mixing law. The idea is allow for anything to bounce off an inelastic surface.
/// For example, a superball bounces on anything.
function b2MixRestitution(restitution1, restitution2) {
return restitution1 > restitution2 ? restitution1 : restitution2;
}
var b2ContactEdge = /** @class */ (function () {
function b2ContactEdge(contact) {
this.prev = null; ///< the previous contact edge in the body's contact list
this.next = null; ///< the next contact edge in the body's contact list
this.contact = contact;
}
return b2ContactEdge;
}());
var b2Contact = /** @class */ (function () {
function b2Contact() {
this.m_islandFlag = false; /// Used when crawling contact graph when forming islands.
this.m_touchingFlag = false; /// Set when the shapes are touching.
this.m_enabledFlag = false; /// This contact can be disabled (by user)
this.m_filterFlag = false; /// This contact needs filtering because a fixture filter was changed.
this.m_bulletHitFlag = false; /// This bullet contact had a TOI event
this.m_toiFlag = false; /// This contact has a valid TOI in m_toi
this.m_prev = null;
this.m_next = null;
this.m_indexA = 0;
this.m_indexB = 0;
this.m_manifold = new b2Manifold(); // TODO: readonly
this.m_toiCount = 0;
this.m_toi = 0;
this.m_friction = 0;
this.m_restitution = 0;
this.m_tangentSpeed = 0;
this.m_oldManifold = new b2Manifold(); // TODO: readonly
this.m_nodeA = new b2ContactEdge(this);
this.m_nodeB = new b2ContactEdge(this);
}
b2Contact.prototype.GetManifold = function () {
return this.m_manifold;
};
b2Contact.prototype.GetWorldManifold = function (worldManifold) {
var bodyA = this.m_fixtureA.GetBody();
var bodyB = this.m_fixtureB.GetBody();
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
worldManifold.Initialize(this.m_manifold, bodyA.GetTransform(), shapeA.m_radius, bodyB.GetTransform(), shapeB.m_radius);
};
b2Contact.prototype.IsTouching = function () {
return this.m_touchingFlag;
};
b2Contact.prototype.SetEnabled = function (flag) {
this.m_enabledFlag = flag;
};
b2Contact.prototype.IsEnabled = function () {
return this.m_enabledFlag;
};
b2Contact.prototype.GetNext = function () {
return this.m_next;
};
b2Contact.prototype.GetFixtureA = function () {
return this.m_fixtureA;
};
b2Contact.prototype.GetChildIndexA = function () {
return this.m_indexA;
};
b2Contact.prototype.GetFixtureB = function () {
return this.m_fixtureB;
};
b2Contact.prototype.GetChildIndexB = function () {
return this.m_indexB;
};
b2Contact.prototype.FlagForFiltering = function () {
this.m_filterFlag = true;
};
b2Contact.prototype.SetFriction = function (friction) {
this.m_friction = friction;
};
b2Contact.prototype.GetFriction = function () {
return this.m_friction;
};
b2Contact.prototype.ResetFriction = function () {
this.m_friction = b2MixFriction(this.m_fixtureA.m_friction, this.m_fixtureB.m_friction);
};
b2Contact.prototype.SetRestitution = function (restitution) {
this.m_restitution = restitution;
};
b2Contact.prototype.GetRestitution = function () {
return this.m_restitution;
};
b2Contact.prototype.ResetRestitution = function () {
this.m_restitution = b2MixRestitution(this.m_fixtureA.m_restitution, this.m_fixtureB.m_restitution);
};
b2Contact.prototype.SetTangentSpeed = function (speed) {
this.m_tangentSpeed = speed;
};
b2Contact.prototype.GetTangentSpeed = function () {
return this.m_tangentSpeed;
};
b2Contact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
this.m_islandFlag = false;
this.m_touchingFlag = false;
this.m_enabledFlag = true;
this.m_filterFlag = false;
this.m_bulletHitFlag = false;
this.m_toiFlag = false;
this.m_fixtureA = fixtureA;
this.m_fixtureB = fixtureB;
this.m_indexA = indexA;
this.m_indexB = indexB;
this.m_manifold.pointCount = 0;
this.m_prev = null;
this.m_next = null;
delete this.m_nodeA.contact; // = null;
this.m_nodeA.prev = null;
this.m_nodeA.next = null;
delete this.m_nodeA.other; // = null;
delete this.m_nodeB.contact; // = null;
this.m_nodeB.prev = null;
this.m_nodeB.next = null;
delete this.m_nodeB.other; // = null;
this.m_toiCount = 0;
this.m_friction = b2MixFriction(this.m_fixtureA.m_friction, this.m_fixtureB.m_friction);
this.m_restitution = b2MixRestitution(this.m_fixtureA.m_restitution, this.m_fixtureB.m_restitution);
};
b2Contact.prototype.Update = function (listener) {
var tManifold = this.m_oldManifold;
this.m_oldManifold = this.m_manifold;
this.m_manifold = tManifold;
// Re-enable this contact.
this.m_enabledFlag = true;
var touching = false;
var wasTouching = this.m_touchingFlag;
var sensorA = this.m_fixtureA.IsSensor();
var sensorB = this.m_fixtureB.IsSensor();
var sensor = sensorA || sensorB;
var bodyA = this.m_fixtureA.GetBody();
var bodyB = this.m_fixtureB.GetBody();
var xfA = bodyA.GetTransform();
var xfB = bodyB.GetTransform();
///const aabbOverlap = b2TestOverlapAABB(this.m_fixtureA.GetAABB(0), this.m_fixtureB.GetAABB(0));
// Is this contact a sensor?
if (sensor) {
///if (aabbOverlap)
///{
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
touching = b2TestOverlapShape(shapeA, this.m_indexA, shapeB, this.m_indexB, xfA, xfB);
///}
// Sensors don't generate manifolds.
this.m_manifold.pointCount = 0;
}
else {
///if (aabbOverlap)
///{
this.Evaluate(this.m_manifold, xfA, xfB);
touching = this.m_manifold.pointCount > 0;
// Match old contact ids to new contact ids and copy the
// stored impulses to warm start the solver.
for (var i = 0; i < this.m_manifold.pointCount; ++i) {
var mp2 = this.m_manifold.points[i];
mp2.normalImpulse = 0;
mp2.tangentImpulse = 0;
var id2 = mp2.id;
for (var j = 0; j < this.m_oldManifold.pointCount; ++j) {
var mp1 = this.m_oldManifold.points[j];
if (mp1.id.key === id2.key) {
mp2.normalImpulse = mp1.normalImpulse;
mp2.tangentImpulse = mp1.tangentImpulse;
break;
}
}
}
///}
///else
///{
/// this.m_manifold.pointCount = 0;
///}
if (touching !== wasTouching) {
bodyA.SetAwake(true);
bodyB.SetAwake(true);
}
}
this.m_touchingFlag = touching;
if (!wasTouching && touching && listener) {
listener.BeginContact(this);
}
if (wasTouching && !touching && listener) {
listener.EndContact(this);
}
if (!sensor && touching && listener) {
listener.PreSolve(this, this.m_oldManifold);
}
};
b2Contact.prototype.ComputeTOI = function (sweepA, sweepB) {
var input = b2Contact.ComputeTOI_s_input;
input.proxyA.SetShape(this.m_fixtureA.GetShape(), this.m_indexA);
input.proxyB.SetShape(this.m_fixtureB.GetShape(), this.m_indexB);
input.sweepA.Copy(sweepA);
input.sweepB.Copy(sweepB);
input.tMax = b2_linearSlop;
var output = b2Contact.ComputeTOI_s_output;
b2TimeOfImpact(output, input);
return output.t;
};
b2Contact.ComputeTOI_s_input = new b2TOIInput();
b2Contact.ComputeTOI_s_output = new b2TOIOutput();
return b2Contact;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2CircleContact = /** @class */ (function (_super) {
__extends(b2CircleContact, _super);
function b2CircleContact() {
return _super.call(this) || this;
}
b2CircleContact.Create = function (allocator) {
return new b2CircleContact();
};
b2CircleContact.Destroy = function (contact, allocator) {
};
b2CircleContact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
_super.prototype.Reset.call(this, fixtureA, indexA, fixtureB, indexB);
};
b2CircleContact.prototype.Evaluate = function (manifold, xfA, xfB) {
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
// DEBUG: b2Assert(shapeA.GetType() === b2ShapeType.e_circleShape);
// DEBUG: b2Assert(shapeB.GetType() === b2ShapeType.e_circleShape);
b2CollideCircles(manifold, shapeA, xfA, shapeB, xfB);
};
return b2CircleContact;
}(b2Contact));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2PolygonContact = /** @class */ (function (_super) {
__extends(b2PolygonContact, _super);
function b2PolygonContact() {
return _super.call(this) || this;
}
b2PolygonContact.Create = function (allocator) {
return new b2PolygonContact();
};
b2PolygonContact.Destroy = function (contact, allocator) {
};
b2PolygonContact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
_super.prototype.Reset.call(this, fixtureA, indexA, fixtureB, indexB);
};
b2PolygonContact.prototype.Evaluate = function (manifold, xfA, xfB) {
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
// DEBUG: b2Assert(shapeA.GetType() === b2ShapeType.e_polygonShape);
// DEBUG: b2Assert(shapeB.GetType() === b2ShapeType.e_polygonShape);
b2CollidePolygons(manifold, shapeA, xfA, shapeB, xfB);
};
return b2PolygonContact;
}(b2Contact));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2PolygonAndCircleContact = /** @class */ (function (_super) {
__extends(b2PolygonAndCircleContact, _super);
function b2PolygonAndCircleContact() {
return _super.call(this) || this;
}
b2PolygonAndCircleContact.Create = function (allocator) {
return new b2PolygonAndCircleContact();
};
b2PolygonAndCircleContact.Destroy = function (contact, allocator) {
};
b2PolygonAndCircleContact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
_super.prototype.Reset.call(this, fixtureA, indexA, fixtureB, indexB);
// DEBUG: b2Assert(fixtureA.GetType() === b2ShapeType.e_polygonShape);
// DEBUG: b2Assert(fixtureB.GetType() === b2ShapeType.e_circleShape);
};
b2PolygonAndCircleContact.prototype.Evaluate = function (manifold, xfA, xfB) {
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
// DEBUG: b2Assert(shapeA instanceof b2PolygonShape);
// DEBUG: b2Assert(shapeB instanceof b2CircleShape);
b2CollidePolygonAndCircle(manifold, shapeA, xfA, shapeB, xfB);
};
return b2PolygonAndCircleContact;
}(b2Contact));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2EdgeAndCircleContact = /** @class */ (function (_super) {
__extends(b2EdgeAndCircleContact, _super);
function b2EdgeAndCircleContact() {
return _super.call(this) || this;
}
b2EdgeAndCircleContact.Create = function (allocator) {
return new b2EdgeAndCircleContact();
};
b2EdgeAndCircleContact.Destroy = function (contact, allocator) {
};
b2EdgeAndCircleContact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
_super.prototype.Reset.call(this, fixtureA, indexA, fixtureB, indexB);
// DEBUG: b2Assert(fixtureA.GetType() === b2ShapeType.e_edgeShape);
// DEBUG: b2Assert(fixtureB.GetType() === b2ShapeType.e_circleShape);
};
b2EdgeAndCircleContact.prototype.Evaluate = function (manifold, xfA, xfB) {
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
// DEBUG: b2Assert(shapeA instanceof b2EdgeShape);
// DEBUG: b2Assert(shapeB instanceof b2CircleShape);
b2CollideEdgeAndCircle(manifold, shapeA, xfA, shapeB, xfB);
};
return b2EdgeAndCircleContact;
}(b2Contact));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2EdgeAndPolygonContact = /** @class */ (function (_super) {
__extends(b2EdgeAndPolygonContact, _super);
function b2EdgeAndPolygonContact() {
return _super.call(this) || this;
}
b2EdgeAndPolygonContact.Create = function (allocator) {
return new b2EdgeAndPolygonContact();
};
b2EdgeAndPolygonContact.Destroy = function (contact, allocator) {
};
b2EdgeAndPolygonContact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
_super.prototype.Reset.call(this, fixtureA, indexA, fixtureB, indexB);
// DEBUG: b2Assert(fixtureA.GetType() === b2ShapeType.e_edgeShape);
// DEBUG: b2Assert(fixtureB.GetType() === b2ShapeType.e_polygonShape);
};
b2EdgeAndPolygonContact.prototype.Evaluate = function (manifold, xfA, xfB) {
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
// DEBUG: b2Assert(shapeA instanceof b2EdgeShape);
// DEBUG: b2Assert(shapeB instanceof b2PolygonShape);
b2CollideEdgeAndPolygon(manifold, shapeA, xfA, shapeB, xfB);
};
return b2EdgeAndPolygonContact;
}(b2Contact));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2ChainAndCircleContact = /** @class */ (function (_super) {
__extends(b2ChainAndCircleContact, _super);
function b2ChainAndCircleContact() {
return _super.call(this) || this;
}
b2ChainAndCircleContact.Create = function (allocator) {
return new b2ChainAndCircleContact();
};
b2ChainAndCircleContact.Destroy = function (contact, allocator) {
};
b2ChainAndCircleContact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
_super.prototype.Reset.call(this, fixtureA, indexA, fixtureB, indexB);
// DEBUG: b2Assert(fixtureA.GetType() === b2ShapeType.e_chainShape);
// DEBUG: b2Assert(fixtureB.GetType() === b2ShapeType.e_circleShape);
};
b2ChainAndCircleContact.prototype.Evaluate = function (manifold, xfA, xfB) {
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
// DEBUG: b2Assert(shapeA instanceof b2ChainShape);
// DEBUG: b2Assert(shapeB instanceof b2CircleShape);
var chain = shapeA;
var edge = b2ChainAndCircleContact.Evaluate_s_edge;
chain.GetChildEdge(edge, this.m_indexA);
b2CollideEdgeAndCircle(manifold, edge, xfA, shapeB, xfB);
};
b2ChainAndCircleContact.Evaluate_s_edge = new b2EdgeShape();
return b2ChainAndCircleContact;
}(b2Contact));
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2ChainAndPolygonContact = /** @class */ (function (_super) {
__extends(b2ChainAndPolygonContact, _super);
function b2ChainAndPolygonContact() {
return _super.call(this) || this;
}
b2ChainAndPolygonContact.Create = function (allocator) {
return new b2ChainAndPolygonContact();
};
b2ChainAndPolygonContact.Destroy = function (contact, allocator) {
};
b2ChainAndPolygonContact.prototype.Reset = function (fixtureA, indexA, fixtureB, indexB) {
_super.prototype.Reset.call(this, fixtureA, indexA, fixtureB, indexB);
// DEBUG: b2Assert(fixtureA.GetType() === b2ShapeType.e_chainShape);
// DEBUG: b2Assert(fixtureB.GetType() === b2ShapeType.e_polygonShape);
};
b2ChainAndPolygonContact.prototype.Evaluate = function (manifold, xfA, xfB) {
var shapeA = this.m_fixtureA.GetShape();
var shapeB = this.m_fixtureB.GetShape();
// DEBUG: b2Assert(shapeA instanceof b2ChainShape);
// DEBUG: b2Assert(shapeB instanceof b2PolygonShape);
var chain = shapeA;
var edge = b2ChainAndPolygonContact.Evaluate_s_edge;
chain.GetChildEdge(edge, this.m_indexA);
b2CollideEdgeAndPolygon(manifold, edge, xfA, shapeB, xfB);
};
b2ChainAndPolygonContact.Evaluate_s_edge = new b2EdgeShape();
return b2ChainAndPolygonContact;
}(b2Contact));
// DEBUG: import { b2Assert } from "../../Common/b2Settings";
var b2ContactRegister = /** @class */ (function () {
function b2ContactRegister() {
// public pool: b2Contact[];
this.createFcn = null;
this.destroyFcn = null;
this.primary = false;
}
return b2ContactRegister;
}());
var b2ContactFactory = /** @class */ (function () {
function b2ContactFactory(allocator) {
this.m_allocator = null;
this.m_allocator = allocator;
this.InitializeRegisters();
}
b2ContactFactory.prototype.AddType = function (createFcn, destroyFcn, type1, type2) {
var _this = this;
var pool = b2MakeArray(256, function (i) { return createFcn(_this.m_allocator); }); // TODO: b2Settings
function poolCreateFcn(allocator) {
// if (pool.length > 0) {
// return pool.pop();
// }
// return createFcn(allocator);
return pool.pop() || createFcn(allocator);
}
function poolDestroyFcn(contact, allocator) {
pool.push(contact);
}
// this.m_registers[type1][type2].pool = pool;
this.m_registers[type1][type2].createFcn = poolCreateFcn;
this.m_registers[type1][type2].destroyFcn = poolDestroyFcn;
this.m_registers[type1][type2].primary = true;
if (type1 !== type2) {
// this.m_registers[type2][type1].pool = pool;
this.m_registers[type2][type1].createFcn = poolCreateFcn;
this.m_registers[type2][type1].destroyFcn = poolDestroyFcn;
this.m_registers[type2][type1].primary = false;
}
/*
this.m_registers[type1][type2].createFcn = createFcn;
this.m_registers[type1][type2].destroyFcn = destroyFcn;
this.m_registers[type1][type2].primary = true;
if (type1 !== type2) {
this.m_registers[type2][type1].createFcn = createFcn;
this.m_registers[type2][type1].destroyFcn = destroyFcn;
this.m_registers[type2][type1].primary = false;
}
*/
};
b2ContactFactory.prototype.InitializeRegisters = function () {
this.m_registers = [ /*b2ShapeType.e_shapeTypeCount*/];
for (var i = 0; i < exports.b2ShapeType.e_shapeTypeCount; i++) {
this.m_registers[i] = [ /*b2ShapeType.e_shapeTypeCount*/];
for (var j = 0; j < exports.b2ShapeType.e_shapeTypeCount; j++) {
this.m_registers[i][j] = new b2ContactRegister();
}
}
this.AddType(b2CircleContact.Create, b2CircleContact.Destroy, exports.b2ShapeType.e_circleShape, exports.b2ShapeType.e_circleShape);
this.AddType(b2PolygonAndCircleContact.Create, b2PolygonAndCircleContact.Destroy, exports.b2ShapeType.e_polygonShape, exports.b2ShapeType.e_circleShape);
this.AddType(b2PolygonContact.Create, b2PolygonContact.Destroy, exports.b2ShapeType.e_polygonShape, exports.b2ShapeType.e_polygonShape);
this.AddType(b2EdgeAndCircleContact.Create, b2EdgeAndCircleContact.Destroy, exports.b2ShapeType.e_edgeShape, exports.b2ShapeType.e_circleShape);
this.AddType(b2EdgeAndPolygonContact.Create, b2EdgeAndPolygonContact.Destroy, exports.b2ShapeType.e_edgeShape, exports.b2ShapeType.e_polygonShape);
this.AddType(b2ChainAndCircleContact.Create, b2ChainAndCircleContact.Destroy, exports.b2ShapeType.e_chainShape, exports.b2ShapeType.e_circleShape);
this.AddType(b2ChainAndPolygonContact.Create, b2ChainAndPolygonContact.Destroy, exports.b2ShapeType.e_chainShape, exports.b2ShapeType.e_polygonShape);
};
b2ContactFactory.prototype.Create = function (fixtureA, indexA, fixtureB, indexB) {
var type1 = fixtureA.GetType();
var type2 = fixtureB.GetType();
// DEBUG: b2Assert(0 <= type1 && type1 < b2ShapeType.e_shapeTypeCount);
// DEBUG: b2Assert(0 <= type2 && type2 < b2ShapeType.e_shapeTypeCount);
var reg = this.m_registers[type1][type2];
if (reg.createFcn) {
var c = reg.createFcn(this.m_allocator);
if (reg.primary) {
c.Reset(fixtureA, indexA, fixtureB, indexB);
}
else {
c.Reset(fixtureB, indexB, fixtureA, indexA);
}
return c;
}
else {
return null;
}
};
b2ContactFactory.prototype.Destroy = function (contact) {
var fixtureA = contact.m_fixtureA;
var fixtureB = contact.m_fixtureB;
if (contact.m_manifold.pointCount > 0 &&
!fixtureA.IsSensor() &&
!fixtureB.IsSensor()) {
fixtureA.GetBody().SetAwake(true);
fixtureB.GetBody().SetAwake(true);
}
var typeA = fixtureA.GetType();
var typeB = fixtureB.GetType();
// DEBUG: b2Assert(0 <= typeA && typeB < b2ShapeType.e_shapeTypeCount);
// DEBUG: b2Assert(0 <= typeA && typeB < b2ShapeType.e_shapeTypeCount);
var reg = this.m_registers[typeA][typeB];
if (reg.destroyFcn) {
reg.destroyFcn(contact, this.m_allocator);
}
};
return b2ContactFactory;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
// #endif
/// Joints and fixtures are destroyed when their associated
/// body is destroyed. Implement this listener so that you
/// may nullify references to these joints and shapes.
var b2DestructionListener = /** @class */ (function () {
function b2DestructionListener() {
}
/// Called when any joint is about to be destroyed due
/// to the destruction of one of its attached bodies.
b2DestructionListener.prototype.SayGoodbyeJoint = function (joint) { };
/// Called when any fixture is about to be destroyed due
/// to the destruction of its parent body.
b2DestructionListener.prototype.SayGoodbyeFixture = function (fixture) { };
// #if B2_ENABLE_PARTICLE
/// Called when any particle group is about to be destroyed.
b2DestructionListener.prototype.SayGoodbyeParticleGroup = function (group) { };
/// Called when a particle is about to be destroyed.
/// The index can be used in conjunction with
/// b2ParticleSystem::GetUserDataBuffer() or
/// b2ParticleSystem::GetParticleHandleFromIndex() to determine which
/// particle has been destroyed.
b2DestructionListener.prototype.SayGoodbyeParticle = function (system, index) { };
return b2DestructionListener;
}());
/// Implement this class to provide collision filtering. In other words, you can implement
/// this class if you want finer control over contact creation.
var b2ContactFilter = /** @class */ (function () {
function b2ContactFilter() {
}
/// Return true if contact calculations should be performed between these two shapes.
/// @warning for performance reasons this is only called when the AABBs begin to overlap.
b2ContactFilter.prototype.ShouldCollide = function (fixtureA, fixtureB) {
var bodyA = fixtureA.GetBody();
var bodyB = fixtureB.GetBody();
// At least one body should be dynamic or kinematic.
if (bodyB.GetType() === exports.b2BodyType.b2_staticBody && bodyA.GetType() === exports.b2BodyType.b2_staticBody) {
return false;
}
// Does a joint prevent collision?
if (!bodyB.ShouldCollideConnected(bodyA)) {
return false;
}
var filter1 = fixtureA.GetFilterData();
var filter2 = fixtureB.GetFilterData();
if (filter1.groupIndex === filter2.groupIndex && filter1.groupIndex !== 0) {
return (filter1.groupIndex > 0);
}
var collide = (((filter1.maskBits & filter2.categoryBits) !== 0) && ((filter1.categoryBits & filter2.maskBits) !== 0));
return collide;
};
// #if B2_ENABLE_PARTICLE
b2ContactFilter.prototype.ShouldCollideFixtureParticle = function (fixture, system, index) {
return true;
};
b2ContactFilter.prototype.ShouldCollideParticleParticle = function (system, indexA, indexB) {
return true;
};
// #endif
b2ContactFilter.b2_defaultFilter = new b2ContactFilter();
return b2ContactFilter;
}());
/// Contact impulses for reporting. Impulses are used instead of forces because
/// sub-step forces may approach infinity for rigid body collisions. These
/// match up one-to-one with the contact points in b2Manifold.
var b2ContactImpulse = /** @class */ (function () {
function b2ContactImpulse() {
this.normalImpulses = b2MakeNumberArray(b2_maxManifoldPoints);
this.tangentImpulses = b2MakeNumberArray(b2_maxManifoldPoints);
this.count = 0;
}
return b2ContactImpulse;
}());
/// Implement this class to get contact information. You can use these results for
/// things like sounds and game logic. You can also get contact results by
/// traversing the contact lists after the time step. However, you might miss
/// some contacts because continuous physics leads to sub-stepping.
/// Additionally you may receive multiple callbacks for the same contact in a
/// single time step.
/// You should strive to make your callbacks efficient because there may be
/// many callbacks per time step.
/// @warning You cannot create/destroy Box2D entities inside these callbacks.
var b2ContactListener = /** @class */ (function () {
function b2ContactListener() {
}
/// Called when two fixtures begin to touch.
b2ContactListener.prototype.BeginContact = function (contact) { };
/// Called when two fixtures cease to touch.
b2ContactListener.prototype.EndContact = function (contact) { };
// #if B2_ENABLE_PARTICLE
b2ContactListener.prototype.BeginContactFixtureParticle = function (system, contact) { };
b2ContactListener.prototype.EndContactFixtureParticle = function (system, contact) { };
b2ContactListener.prototype.BeginContactParticleParticle = function (system, contact) { };
b2ContactListener.prototype.EndContactParticleParticle = function (system, contact) { };
// #endif
/// This is called after a contact is updated. This allows you to inspect a
/// contact before it goes to the solver. If you are careful, you can modify the
/// contact manifold (e.g. disable contact).
/// A copy of the old manifold is provided so that you can detect changes.
/// Note: this is called only for awake bodies.
/// Note: this is called even when the number of contact points is zero.
/// Note: this is not called for sensors.
/// Note: if you set the number of contact points to zero, you will not
/// get an EndContact callback. However, you may get a BeginContact callback
/// the next step.
b2ContactListener.prototype.PreSolve = function (contact, oldManifold) { };
/// This lets you inspect a contact after the solver is finished. This is useful
/// for inspecting impulses.
/// Note: the contact manifold does not include time of impact impulses, which can be
/// arbitrarily large if the sub-step is small. Hence the impulse is provided explicitly
/// in a separate data structure.
/// Note: this is only called for contacts that are touching, solid, and awake.
b2ContactListener.prototype.PostSolve = function (contact, impulse) { };
b2ContactListener.b2_defaultListener = new b2ContactListener();
return b2ContactListener;
}());
/// Callback class for AABB queries.
/// See b2World::Query
var b2QueryCallback = /** @class */ (function () {
function b2QueryCallback() {
}
/// Called for each fixture found in the query AABB.
/// @return false to terminate the query.
b2QueryCallback.prototype.ReportFixture = function (fixture) {
return true;
};
// #if B2_ENABLE_PARTICLE
b2QueryCallback.prototype.ReportParticle = function (system, index) {
return false;
};
b2QueryCallback.prototype.ShouldQueryParticleSystem = function (system) {
return true;
};
return b2QueryCallback;
}());
/// Callback class for ray casts.
/// See b2World::RayCast
var b2RayCastCallback = /** @class */ (function () {
function b2RayCastCallback() {
}
/// Called for each fixture found in the query. You control how the ray cast
/// proceeds by returning a float:
/// return -1: ignore this fixture and continue
/// return 0: terminate the ray cast
/// return fraction: clip the ray to this point
/// return 1: don't clip the ray and continue
/// @param fixture the fixture hit by the ray
/// @param point the point of initial intersection
/// @param normal the normal vector at the point of intersection
/// @return -1 to filter, 0 to terminate, fraction to clip the ray for
/// closest hit, 1 to continue
b2RayCastCallback.prototype.ReportFixture = function (fixture, point, normal, fraction) {
return fraction;
};
// #if B2_ENABLE_PARTICLE
b2RayCastCallback.prototype.ReportParticle = function (system, index, point, normal, fraction) {
return 0;
};
b2RayCastCallback.prototype.ShouldQueryParticleSystem = function (system) {
return true;
};
return b2RayCastCallback;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
// Delegate of b2World.
var b2ContactManager = /** @class */ (function () {
function b2ContactManager() {
this.m_broadPhase = new b2BroadPhase();
this.m_contactList = null;
this.m_contactCount = 0;
this.m_contactFilter = b2ContactFilter.b2_defaultFilter;
this.m_contactListener = b2ContactListener.b2_defaultListener;
this.m_allocator = null;
this.m_contactFactory = new b2ContactFactory(this.m_allocator);
}
// Broad-phase callback.
b2ContactManager.prototype.AddPair = function (proxyA, proxyB) {
// DEBUG: b2Assert(proxyA instanceof b2FixtureProxy);
// DEBUG: b2Assert(proxyB instanceof b2FixtureProxy);
var fixtureA = proxyA.fixture;
var fixtureB = proxyB.fixture;
var indexA = proxyA.childIndex;
var indexB = proxyB.childIndex;
var bodyA = fixtureA.GetBody();
var bodyB = fixtureB.GetBody();
// Are the fixtures on the same body?
if (bodyA === bodyB) {
return;
}
// TODO_ERIN use a hash table to remove a potential bottleneck when both
// bodies have a lot of contacts.
// Does a contact already exist?
var edge = bodyB.GetContactList();
while (edge) {
if (edge.other === bodyA) {
var fA = edge.contact.GetFixtureA();
var fB = edge.contact.GetFixtureB();
var iA = edge.contact.GetChildIndexA();
var iB = edge.contact.GetChildIndexB();
if (fA === fixtureA && fB === fixtureB && iA === indexA && iB === indexB) {
// A contact already exists.
return;
}
if (fA === fixtureB && fB === fixtureA && iA === indexB && iB === indexA) {
// A contact already exists.
return;
}
}
edge = edge.next;
}
// Check user filtering.
if (this.m_contactFilter && !this.m_contactFilter.ShouldCollide(fixtureA, fixtureB)) {
return;
}
// Call the factory.
var c = this.m_contactFactory.Create(fixtureA, indexA, fixtureB, indexB);
if (c === null) {
return;
}
// Contact creation may swap fixtures.
fixtureA = c.GetFixtureA();
fixtureB = c.GetFixtureB();
indexA = c.GetChildIndexA();
indexB = c.GetChildIndexB();
bodyA = fixtureA.m_body;
bodyB = fixtureB.m_body;
// Insert into the world.
c.m_prev = null;
c.m_next = this.m_contactList;
if (this.m_contactList !== null) {
this.m_contactList.m_prev = c;
}
this.m_contactList = c;
// Connect to island graph.
// Connect to body A
c.m_nodeA.contact = c;
c.m_nodeA.other = bodyB;
c.m_nodeA.prev = null;
c.m_nodeA.next = bodyA.m_contactList;
if (bodyA.m_contactList !== null) {
bodyA.m_contactList.prev = c.m_nodeA;
}
bodyA.m_contactList = c.m_nodeA;
// Connect to body B
c.m_nodeB.contact = c;
c.m_nodeB.other = bodyA;
c.m_nodeB.prev = null;
c.m_nodeB.next = bodyB.m_contactList;
if (bodyB.m_contactList !== null) {
bodyB.m_contactList.prev = c.m_nodeB;
}
bodyB.m_contactList = c.m_nodeB;
// Wake up the bodies
if (!fixtureA.IsSensor() && !fixtureB.IsSensor()) {
bodyA.SetAwake(true);
bodyB.SetAwake(true);
}
++this.m_contactCount;
};
b2ContactManager.prototype.FindNewContacts = function () {
var _this = this;
this.m_broadPhase.UpdatePairs(function (proxyA, proxyB) {
_this.AddPair(proxyA, proxyB);
});
};
b2ContactManager.prototype.Destroy = function (c) {
var fixtureA = c.GetFixtureA();
var fixtureB = c.GetFixtureB();
var bodyA = fixtureA.GetBody();
var bodyB = fixtureB.GetBody();
if (this.m_contactListener && c.IsTouching()) {
this.m_contactListener.EndContact(c);
}
// Remove from the world.
if (c.m_prev) {
c.m_prev.m_next = c.m_next;
}
if (c.m_next) {
c.m_next.m_prev = c.m_prev;
}
if (c === this.m_contactList) {
this.m_contactList = c.m_next;
}
// Remove from body 1
if (c.m_nodeA.prev) {
c.m_nodeA.prev.next = c.m_nodeA.next;
}
if (c.m_nodeA.next) {
c.m_nodeA.next.prev = c.m_nodeA.prev;
}
if (c.m_nodeA === bodyA.m_contactList) {
bodyA.m_contactList = c.m_nodeA.next;
}
// Remove from body 2
if (c.m_nodeB.prev) {
c.m_nodeB.prev.next = c.m_nodeB.next;
}
if (c.m_nodeB.next) {
c.m_nodeB.next.prev = c.m_nodeB.prev;
}
if (c.m_nodeB === bodyB.m_contactList) {
bodyB.m_contactList = c.m_nodeB.next;
}
// Call the factory.
this.m_contactFactory.Destroy(c);
--this.m_contactCount;
};
// This is the top level collision call for the time step. Here
// all the narrow phase collision is processed for the world
// contact list.
b2ContactManager.prototype.Collide = function () {
// Update awake contacts.
var c = this.m_contactList;
while (c) {
var fixtureA = c.GetFixtureA();
var fixtureB = c.GetFixtureB();
var indexA = c.GetChildIndexA();
var indexB = c.GetChildIndexB();
var bodyA = fixtureA.GetBody();
var bodyB = fixtureB.GetBody();
// Is this contact flagged for filtering?
if (c.m_filterFlag) {
// Check user filtering.
if (this.m_contactFilter && !this.m_contactFilter.ShouldCollide(fixtureA, fixtureB)) {
var cNuke = c;
c = cNuke.m_next;
this.Destroy(cNuke);
continue;
}
// Clear the filtering flag.
c.m_filterFlag = false;
}
var activeA = bodyA.IsAwake() && bodyA.m_type !== exports.b2BodyType.b2_staticBody;
var activeB = bodyB.IsAwake() && bodyB.m_type !== exports.b2BodyType.b2_staticBody;
// At least one body must be awake and it must be dynamic or kinematic.
if (!activeA && !activeB) {
c = c.m_next;
continue;
}
var proxyA = fixtureA.m_proxies[indexA].treeNode;
var proxyB = fixtureB.m_proxies[indexB].treeNode;
var overlap = b2TestOverlapAABB(proxyA.aabb, proxyB.aabb);
// Here we destroy contacts that cease to overlap in the broad-phase.
if (!overlap) {
var cNuke = c;
c = cNuke.m_next;
this.Destroy(cNuke);
continue;
}
// The contact persists.
c.Update(this.m_contactListener);
c = c.m_next;
}
};
return b2ContactManager;
}());
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/// Profiling data. Times are in milliseconds.
var b2Profile = /** @class */ (function () {
function b2Profile() {
this.step = 0;
this.collide = 0;
this.solve = 0;
this.solveInit = 0;
this.solveVelocity = 0;
this.solvePosition = 0;
this.broadphase = 0;
this.solveTOI = 0;
}
b2Profile.prototype.Reset = function () {
this.step = 0;
this.collide = 0;
this.solve = 0;
this.solveInit = 0;
this.solveVelocity = 0;
this.solvePosition = 0;
this.broadphase = 0;
this.solveTOI = 0;
return this;
};
return b2Profile;
}());
/// This is an internal structure.
var b2TimeStep = /** @class */ (function () {
function b2TimeStep() {
this.dt = 0; // time step
this.inv_dt = 0; // inverse time step (0 if dt == 0).
this.dtRatio = 0; // dt * inv_dt0
this.velocityIterations = 0;
this.positionIterations = 0;
// #if B2_ENABLE_PARTICLE
this.particleIterations = 0;
// #endif
this.warmStarting = false;
}
b2TimeStep.prototype.Copy = function (step) {
this.dt = step.dt;
this.inv_dt = step.inv_dt;
this.dtRatio = step.dtRatio;
this.positionIterations = step.positionIterations;
this.velocityIterations = step.velocityIterations;
// #if B2_ENABLE_PARTICLE
this.particleIterations = step.particleIterations;
// #endif
this.warmStarting = step.warmStarting;
return this;
};
return b2TimeStep;
}());
var b2Position = /** @class */ (function () {
function b2Position() {
this.c = new b2Vec2();
this.a = 0;
}
b2Position.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2Position(); });
};
return b2Position;
}());
var b2Velocity = /** @class */ (function () {
function b2Velocity() {
this.v = new b2Vec2();
this.w = 0;
}
b2Velocity.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2Velocity(); });
};
return b2Velocity;
}());
var b2SolverData = /** @class */ (function () {
function b2SolverData() {
this.step = new b2TimeStep();
}
return b2SolverData;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
// Solver debugging is normally disabled because the block solver sometimes has to deal with a poorly conditioned effective mass matrix.
// #define B2_DEBUG_SOLVER 0
var g_blockSolve = false;
var b2VelocityConstraintPoint = /** @class */ (function () {
function b2VelocityConstraintPoint() {
this.rA = new b2Vec2();
this.rB = new b2Vec2();
this.normalImpulse = 0;
this.tangentImpulse = 0;
this.normalMass = 0;
this.tangentMass = 0;
this.velocityBias = 0;
}
b2VelocityConstraintPoint.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2VelocityConstraintPoint(); });
};
return b2VelocityConstraintPoint;
}());
var b2ContactVelocityConstraint = /** @class */ (function () {
function b2ContactVelocityConstraint() {
this.points = b2VelocityConstraintPoint.MakeArray(b2_maxManifoldPoints);
this.normal = new b2Vec2();
this.tangent = new b2Vec2();
this.normalMass = new b2Mat22();
this.K = new b2Mat22();
this.indexA = 0;
this.indexB = 0;
this.invMassA = 0;
this.invMassB = 0;
this.invIA = 0;
this.invIB = 0;
this.friction = 0;
this.restitution = 0;
this.tangentSpeed = 0;
this.pointCount = 0;
this.contactIndex = 0;
}
b2ContactVelocityConstraint.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2ContactVelocityConstraint(); });
};
return b2ContactVelocityConstraint;
}());
var b2ContactPositionConstraint = /** @class */ (function () {
function b2ContactPositionConstraint() {
this.localPoints = b2Vec2.MakeArray(b2_maxManifoldPoints);
this.localNormal = new b2Vec2();
this.localPoint = new b2Vec2();
this.indexA = 0;
this.indexB = 0;
this.invMassA = 0;
this.invMassB = 0;
this.localCenterA = new b2Vec2();
this.localCenterB = new b2Vec2();
this.invIA = 0;
this.invIB = 0;
this.type = exports.b2ManifoldType.e_unknown;
this.radiusA = 0;
this.radiusB = 0;
this.pointCount = 0;
}
b2ContactPositionConstraint.MakeArray = function (length) {
return b2MakeArray(length, function (i) { return new b2ContactPositionConstraint(); });
};
return b2ContactPositionConstraint;
}());
var b2ContactSolverDef = /** @class */ (function () {
function b2ContactSolverDef() {
this.step = new b2TimeStep();
this.count = 0;
this.allocator = null;
}
return b2ContactSolverDef;
}());
var b2PositionSolverManifold = /** @class */ (function () {
function b2PositionSolverManifold() {
this.normal = new b2Vec2();
this.point = new b2Vec2();
this.separation = 0;
}
b2PositionSolverManifold.prototype.Initialize = function (pc, xfA, xfB, index) {
var pointA = b2PositionSolverManifold.Initialize_s_pointA;
var pointB = b2PositionSolverManifold.Initialize_s_pointB;
var planePoint = b2PositionSolverManifold.Initialize_s_planePoint;
var clipPoint = b2PositionSolverManifold.Initialize_s_clipPoint;
// DEBUG: b2Assert(pc.pointCount > 0);
switch (pc.type) {
case exports.b2ManifoldType.e_circles: {
// b2Vec2 pointA = b2Mul(xfA, pc->localPoint);
b2Transform.MulXV(xfA, pc.localPoint, pointA);
// b2Vec2 pointB = b2Mul(xfB, pc->localPoints[0]);
b2Transform.MulXV(xfB, pc.localPoints[0], pointB);
// normal = pointB - pointA;
// normal.Normalize();
b2Vec2.SubVV(pointB, pointA, this.normal).SelfNormalize();
// point = 0.5f * (pointA + pointB);
b2Vec2.MidVV(pointA, pointB, this.point);
// separation = b2Dot(pointB - pointA, normal) - pc->radius;
this.separation = b2Vec2.DotVV(b2Vec2.SubVV(pointB, pointA, b2Vec2.s_t0), this.normal) - pc.radiusA - pc.radiusB;
break;
}
case exports.b2ManifoldType.e_faceA: {
// normal = b2Mul(xfA.q, pc->localNormal);
b2Rot.MulRV(xfA.q, pc.localNormal, this.normal);
// b2Vec2 planePoint = b2Mul(xfA, pc->localPoint);
b2Transform.MulXV(xfA, pc.localPoint, planePoint);
// b2Vec2 clipPoint = b2Mul(xfB, pc->localPoints[index]);
b2Transform.MulXV(xfB, pc.localPoints[index], clipPoint);
// separation = b2Dot(clipPoint - planePoint, normal) - pc->radius;
this.separation = b2Vec2.DotVV(b2Vec2.SubVV(clipPoint, planePoint, b2Vec2.s_t0), this.normal) - pc.radiusA - pc.radiusB;
// point = clipPoint;
this.point.Copy(clipPoint);
break;
}
case exports.b2ManifoldType.e_faceB: {
// normal = b2Mul(xfB.q, pc->localNormal);
b2Rot.MulRV(xfB.q, pc.localNormal, this.normal);
// b2Vec2 planePoint = b2Mul(xfB, pc->localPoint);
b2Transform.MulXV(xfB, pc.localPoint, planePoint);
// b2Vec2 clipPoint = b2Mul(xfA, pc->localPoints[index]);
b2Transform.MulXV(xfA, pc.localPoints[index], clipPoint);
// separation = b2Dot(clipPoint - planePoint, normal) - pc->radius;
this.separation = b2Vec2.DotVV(b2Vec2.SubVV(clipPoint, planePoint, b2Vec2.s_t0), this.normal) - pc.radiusA - pc.radiusB;
// point = clipPoint;
this.point.Copy(clipPoint);
// Ensure normal points from A to B
// normal = -normal;
this.normal.SelfNeg();
break;
}
}
};
b2PositionSolverManifold.Initialize_s_pointA = new b2Vec2();
b2PositionSolverManifold.Initialize_s_pointB = new b2Vec2();
b2PositionSolverManifold.Initialize_s_planePoint = new b2Vec2();
b2PositionSolverManifold.Initialize_s_clipPoint = new b2Vec2();
return b2PositionSolverManifold;
}());
var b2ContactSolver = /** @class */ (function () {
function b2ContactSolver() {
this.m_step = new b2TimeStep();
this.m_allocator = null;
this.m_positionConstraints = b2ContactPositionConstraint.MakeArray(1024); // TODO: b2Settings
this.m_velocityConstraints = b2ContactVelocityConstraint.MakeArray(1024); // TODO: b2Settings
this.m_count = 0;
}
b2ContactSolver.prototype.Initialize = function (def) {
this.m_step.Copy(def.step);
this.m_allocator = def.allocator;
this.m_count = def.count;
// TODO:
if (this.m_positionConstraints.length < this.m_count) {
var new_length = b2Max(this.m_positionConstraints.length * 2, this.m_count);
while (this.m_positionConstraints.length < new_length) {
this.m_positionConstraints[this.m_positionConstraints.length] = new b2ContactPositionConstraint();
}
}
// TODO:
if (this.m_velocityConstraints.length < this.m_count) {
var new_length = b2Max(this.m_velocityConstraints.length * 2, this.m_count);
while (this.m_velocityConstraints.length < new_length) {
this.m_velocityConstraints[this.m_velocityConstraints.length] = new b2ContactVelocityConstraint();
}
}
this.m_positions = def.positions;
this.m_velocities = def.velocities;
this.m_contacts = def.contacts;
// Initialize position independent portions of the constraints.
for (var i = 0; i < this.m_count; ++i) {
var contact = this.m_contacts[i];
var fixtureA = contact.m_fixtureA;
var fixtureB = contact.m_fixtureB;
var shapeA = fixtureA.GetShape();
var shapeB = fixtureB.GetShape();
var radiusA = shapeA.m_radius;
var radiusB = shapeB.m_radius;
var bodyA = fixtureA.GetBody();
var bodyB = fixtureB.GetBody();
var manifold = contact.GetManifold();
var pointCount = manifold.pointCount;
// DEBUG: b2Assert(pointCount > 0);
var vc = this.m_velocityConstraints[i];
vc.friction = contact.m_friction;
vc.restitution = contact.m_restitution;
vc.tangentSpeed = contact.m_tangentSpeed;
vc.indexA = bodyA.m_islandIndex;
vc.indexB = bodyB.m_islandIndex;
vc.invMassA = bodyA.m_invMass;
vc.invMassB = bodyB.m_invMass;
vc.invIA = bodyA.m_invI;
vc.invIB = bodyB.m_invI;
vc.contactIndex = i;
vc.pointCount = pointCount;
vc.K.SetZero();
vc.normalMass.SetZero();
var pc = this.m_positionConstraints[i];
pc.indexA = bodyA.m_islandIndex;
pc.indexB = bodyB.m_islandIndex;
pc.invMassA = bodyA.m_invMass;
pc.invMassB = bodyB.m_invMass;
pc.localCenterA.Copy(bodyA.m_sweep.localCenter);
pc.localCenterB.Copy(bodyB.m_sweep.localCenter);
pc.invIA = bodyA.m_invI;
pc.invIB = bodyB.m_invI;
pc.localNormal.Copy(manifold.localNormal);
pc.localPoint.Copy(manifold.localPoint);
pc.pointCount = pointCount;
pc.radiusA = radiusA;
pc.radiusB = radiusB;
pc.type = manifold.type;
for (var j = 0; j < pointCount; ++j) {
var cp = manifold.points[j];
var vcp = vc.points[j];
if (this.m_step.warmStarting) {
vcp.normalImpulse = this.m_step.dtRatio * cp.normalImpulse;
vcp.tangentImpulse = this.m_step.dtRatio * cp.tangentImpulse;
}
else {
vcp.normalImpulse = 0;
vcp.tangentImpulse = 0;
}
vcp.rA.SetZero();
vcp.rB.SetZero();
vcp.normalMass = 0;
vcp.tangentMass = 0;
vcp.velocityBias = 0;
pc.localPoints[j].Copy(cp.localPoint);
}
}
return this;
};
b2ContactSolver.prototype.InitializeVelocityConstraints = function () {
var xfA = b2ContactSolver.InitializeVelocityConstraints_s_xfA;
var xfB = b2ContactSolver.InitializeVelocityConstraints_s_xfB;
var worldManifold = b2ContactSolver.InitializeVelocityConstraints_s_worldManifold;
var k_maxConditionNumber = 1000;
for (var i = 0; i < this.m_count; ++i) {
var vc = this.m_velocityConstraints[i];
var pc = this.m_positionConstraints[i];
var radiusA = pc.radiusA;
var radiusB = pc.radiusB;
var manifold = this.m_contacts[vc.contactIndex].GetManifold();
var indexA = vc.indexA;
var indexB = vc.indexB;
var mA = vc.invMassA;
var mB = vc.invMassB;
var iA = vc.invIA;
var iB = vc.invIB;
var localCenterA = pc.localCenterA;
var localCenterB = pc.localCenterB;
var cA = this.m_positions[indexA].c;
var aA = this.m_positions[indexA].a;
var vA = this.m_velocities[indexA].v;
var wA = this.m_velocities[indexA].w;
var cB = this.m_positions[indexB].c;
var aB = this.m_positions[indexB].a;
var vB = this.m_velocities[indexB].v;
var wB = this.m_velocities[indexB].w;
// DEBUG: b2Assert(manifold.pointCount > 0);
xfA.q.SetAngle(aA);
xfB.q.SetAngle(aB);
b2Vec2.SubVV(cA, b2Rot.MulRV(xfA.q, localCenterA, b2Vec2.s_t0), xfA.p);
b2Vec2.SubVV(cB, b2Rot.MulRV(xfB.q, localCenterB, b2Vec2.s_t0), xfB.p);
worldManifold.Initialize(manifold, xfA, radiusA, xfB, radiusB);
vc.normal.Copy(worldManifold.normal);
b2Vec2.CrossVOne(vc.normal, vc.tangent); // compute from normal
var pointCount = vc.pointCount;
for (var j = 0; j < pointCount; ++j) {
var vcp = vc.points[j];
// vcp->rA = worldManifold.points[j] - cA;
b2Vec2.SubVV(worldManifold.points[j], cA, vcp.rA);
// vcp->rB = worldManifold.points[j] - cB;
b2Vec2.SubVV(worldManifold.points[j], cB, vcp.rB);
var rnA = b2Vec2.CrossVV(vcp.rA, vc.normal);
var rnB = b2Vec2.CrossVV(vcp.rB, vc.normal);
var kNormal = mA + mB + iA * rnA * rnA + iB * rnB * rnB;
vcp.normalMass = kNormal > 0 ? 1 / kNormal : 0;
// b2Vec2 tangent = b2Cross(vc->normal, 1.0f);
var tangent = vc.tangent; // precomputed from normal
var rtA = b2Vec2.CrossVV(vcp.rA, tangent);
var rtB = b2Vec2.CrossVV(vcp.rB, tangent);
var kTangent = mA + mB + iA * rtA * rtA + iB * rtB * rtB;
vcp.tangentMass = kTangent > 0 ? 1 / kTangent : 0;
// Setup a velocity bias for restitution.
vcp.velocityBias = 0;
// float32 vRel = b2Dot(vc->normal, vB + b2Cross(wB, vcp->rB) - vA - b2Cross(wA, vcp->rA));
var vRel = b2Vec2.DotVV(vc.normal, b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, vcp.rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, vcp.rA, b2Vec2.s_t1), b2Vec2.s_t0));
if (vRel < (-b2_velocityThreshold)) {
vcp.velocityBias += (-vc.restitution * vRel);
}
}
// If we have two points, then prepare the block solver.
if (vc.pointCount === 2 && g_blockSolve) {
var vcp1 = vc.points[0];
var vcp2 = vc.points[1];
var rn1A = b2Vec2.CrossVV(vcp1.rA, vc.normal);
var rn1B = b2Vec2.CrossVV(vcp1.rB, vc.normal);
var rn2A = b2Vec2.CrossVV(vcp2.rA, vc.normal);
var rn2B = b2Vec2.CrossVV(vcp2.rB, vc.normal);
var k11 = mA + mB + iA * rn1A * rn1A + iB * rn1B * rn1B;
var k22 = mA + mB + iA * rn2A * rn2A + iB * rn2B * rn2B;
var k12 = mA + mB + iA * rn1A * rn2A + iB * rn1B * rn2B;
// Ensure a reasonable condition number.
// float32 k_maxConditionNumber = 1000.0f;
if (k11 * k11 < k_maxConditionNumber * (k11 * k22 - k12 * k12)) {
// K is safe to invert.
vc.K.ex.Set(k11, k12);
vc.K.ey.Set(k12, k22);
vc.K.GetInverse(vc.normalMass);
}
else {
// The constraints are redundant, just use one.
// TODO_ERIN use deepest?
vc.pointCount = 1;
}
}
}
};
b2ContactSolver.prototype.WarmStart = function () {
var P = b2ContactSolver.WarmStart_s_P;
// Warm start.
for (var i = 0; i < this.m_count; ++i) {
var vc = this.m_velocityConstraints[i];
var indexA = vc.indexA;
var indexB = vc.indexB;
var mA = vc.invMassA;
var iA = vc.invIA;
var mB = vc.invMassB;
var iB = vc.invIB;
var pointCount = vc.pointCount;
var vA = this.m_velocities[indexA].v;
var wA = this.m_velocities[indexA].w;
var vB = this.m_velocities[indexB].v;
var wB = this.m_velocities[indexB].w;
var normal = vc.normal;
// b2Vec2 tangent = b2Cross(normal, 1.0f);
var tangent = vc.tangent; // precomputed from normal
for (var j = 0; j < pointCount; ++j) {
var vcp = vc.points[j];
// b2Vec2 P = vcp->normalImpulse * normal + vcp->tangentImpulse * tangent;
b2Vec2.AddVV(b2Vec2.MulSV(vcp.normalImpulse, normal, b2Vec2.s_t0), b2Vec2.MulSV(vcp.tangentImpulse, tangent, b2Vec2.s_t1), P);
// wA -= iA * b2Cross(vcp->rA, P);
wA -= iA * b2Vec2.CrossVV(vcp.rA, P);
// vA -= mA * P;
vA.SelfMulSub(mA, P);
// wB += iB * b2Cross(vcp->rB, P);
wB += iB * b2Vec2.CrossVV(vcp.rB, P);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
}
// this.m_velocities[indexA].v = vA;
this.m_velocities[indexA].w = wA;
// this.m_velocities[indexB].v = vB;
this.m_velocities[indexB].w = wB;
}
};
b2ContactSolver.prototype.SolveVelocityConstraints = function () {
var dv = b2ContactSolver.SolveVelocityConstraints_s_dv;
var dv1 = b2ContactSolver.SolveVelocityConstraints_s_dv1;
var dv2 = b2ContactSolver.SolveVelocityConstraints_s_dv2;
var P = b2ContactSolver.SolveVelocityConstraints_s_P;
var a = b2ContactSolver.SolveVelocityConstraints_s_a;
var b = b2ContactSolver.SolveVelocityConstraints_s_b;
var x = b2ContactSolver.SolveVelocityConstraints_s_x;
var d = b2ContactSolver.SolveVelocityConstraints_s_d;
var P1 = b2ContactSolver.SolveVelocityConstraints_s_P1;
var P2 = b2ContactSolver.SolveVelocityConstraints_s_P2;
var P1P2 = b2ContactSolver.SolveVelocityConstraints_s_P1P2;
for (var i = 0; i < this.m_count; ++i) {
var vc = this.m_velocityConstraints[i];
var indexA = vc.indexA;
var indexB = vc.indexB;
var mA = vc.invMassA;
var iA = vc.invIA;
var mB = vc.invMassB;
var iB = vc.invIB;
var pointCount = vc.pointCount;
var vA = this.m_velocities[indexA].v;
var wA = this.m_velocities[indexA].w;
var vB = this.m_velocities[indexB].v;
var wB = this.m_velocities[indexB].w;
// b2Vec2 normal = vc->normal;
var normal = vc.normal;
// b2Vec2 tangent = b2Cross(normal, 1.0f);
var tangent = vc.tangent; // precomputed from normal
var friction = vc.friction;
// DEBUG: b2Assert(pointCount === 1 || pointCount === 2);
// Solve tangent constraints first because non-penetration is more important
// than friction.
for (var j = 0; j < pointCount; ++j) {
var vcp = vc.points[j];
// Relative velocity at contact
// b2Vec2 dv = vB + b2Cross(wB, vcp->rB) - vA - b2Cross(wA, vcp->rA);
b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, vcp.rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, vcp.rA, b2Vec2.s_t1), dv);
// Compute tangent force
// float32 vt = b2Dot(dv, tangent) - vc->tangentSpeed;
var vt = b2Vec2.DotVV(dv, tangent) - vc.tangentSpeed;
var lambda = vcp.tangentMass * (-vt);
// b2Clamp the accumulated force
var maxFriction = friction * vcp.normalImpulse;
var newImpulse = b2Clamp(vcp.tangentImpulse + lambda, (-maxFriction), maxFriction);
lambda = newImpulse - vcp.tangentImpulse;
vcp.tangentImpulse = newImpulse;
// Apply contact impulse
// b2Vec2 P = lambda * tangent;
b2Vec2.MulSV(lambda, tangent, P);
// vA -= mA * P;
vA.SelfMulSub(mA, P);
// wA -= iA * b2Cross(vcp->rA, P);
wA -= iA * b2Vec2.CrossVV(vcp.rA, P);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
// wB += iB * b2Cross(vcp->rB, P);
wB += iB * b2Vec2.CrossVV(vcp.rB, P);
}
// Solve normal constraints
if (vc.pointCount === 1 || g_blockSolve === false) {
for (var j = 0; j < pointCount; ++j) {
var vcp = vc.points[j];
// Relative velocity at contact
// b2Vec2 dv = vB + b2Cross(wB, vcp->rB) - vA - b2Cross(wA, vcp->rA);
b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, vcp.rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, vcp.rA, b2Vec2.s_t1), dv);
// Compute normal impulse
// float32 vn = b2Dot(dv, normal);
var vn = b2Vec2.DotVV(dv, normal);
var lambda = (-vcp.normalMass * (vn - vcp.velocityBias));
// b2Clamp the accumulated impulse
// float32 newImpulse = b2Max(vcp->normalImpulse + lambda, 0.0f);
var newImpulse = b2Max(vcp.normalImpulse + lambda, 0);
lambda = newImpulse - vcp.normalImpulse;
vcp.normalImpulse = newImpulse;
// Apply contact impulse
// b2Vec2 P = lambda * normal;
b2Vec2.MulSV(lambda, normal, P);
// vA -= mA * P;
vA.SelfMulSub(mA, P);
// wA -= iA * b2Cross(vcp->rA, P);
wA -= iA * b2Vec2.CrossVV(vcp.rA, P);
// vB += mB * P;
vB.SelfMulAdd(mB, P);
// wB += iB * b2Cross(vcp->rB, P);
wB += iB * b2Vec2.CrossVV(vcp.rB, P);
}
}
else {
// Block solver developed in collaboration with Dirk Gregorius (back in 01/07 on Box2D_Lite).
// Build the mini LCP for this contact patch
//
// vn = A * x + b, vn >= 0, x >= 0 and vn_i * x_i = 0 with i = 1..2
//
// A = J * W * JT and J = ( -n, -r1 x n, n, r2 x n )
// b = vn0 - velocityBias
//
// The system is solved using the "Total enumeration method" (s. Murty). The complementary constraint vn_i * x_i
// implies that we must have in any solution either vn_i = 0 or x_i = 0. So for the 2D contact problem the cases
// vn1 = 0 and vn2 = 0, x1 = 0 and x2 = 0, x1 = 0 and vn2 = 0, x2 = 0 and vn1 = 0 need to be tested. The first valid
// solution that satisfies the problem is chosen.
//
// In order to account of the accumulated impulse 'a' (because of the iterative nature of the solver which only requires
// that the accumulated impulse is clamped and not the incremental impulse) we change the impulse variable (x_i).
//
// Substitute:
//
// x = a + d
//
// a := old total impulse
// x := new total impulse
// d := incremental impulse
//
// For the current iteration we extend the formula for the incremental impulse
// to compute the new total impulse:
//
// vn = A * d + b
// = A * (x - a) + b
// = A * x + b - A * a
// = A * x + b'
// b' = b - A * a;
var cp1 = vc.points[0];
var cp2 = vc.points[1];
// b2Vec2 a(cp1->normalImpulse, cp2->normalImpulse);
a.Set(cp1.normalImpulse, cp2.normalImpulse);
// DEBUG: b2Assert(a.x >= 0 && a.y >= 0);
// Relative velocity at contact
// b2Vec2 dv1 = vB + b2Cross(wB, cp1->rB) - vA - b2Cross(wA, cp1->rA);
b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, cp1.rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, cp1.rA, b2Vec2.s_t1), dv1);
// b2Vec2 dv2 = vB + b2Cross(wB, cp2->rB) - vA - b2Cross(wA, cp2->rA);
b2Vec2.SubVV(b2Vec2.AddVCrossSV(vB, wB, cp2.rB, b2Vec2.s_t0), b2Vec2.AddVCrossSV(vA, wA, cp2.rA, b2Vec2.s_t1), dv2);
// Compute normal velocity
// float32 vn1 = b2Dot(dv1, normal);
var vn1 = b2Vec2.DotVV(dv1, normal);
// float32 vn2 = b2Dot(dv2, normal);
var vn2 = b2Vec2.DotVV(dv2, normal);
// b2Vec2 b;
b.x = vn1 - cp1.velocityBias;
b.y = vn2 - cp2.velocityBias;
// Compute b'
// b -= b2Mul(vc->K, a);
b.SelfSub(b2Mat22.MulMV(vc.K, a, b2Vec2.s_t0));
/*
#if B2_DEBUG_SOLVER === 1
const k_errorTol: number = 0.001;
#endif
*/
for (;;) {
//
// Case 1: vn = 0
//
// 0 = A * x + b'
//
// Solve for x:
//
// x = - inv(A) * b'
//
// b2Vec2 x = - b2Mul(vc->normalMass, b);
b2Mat22.MulMV(vc.normalMass, b, x).SelfNeg();
if (x.x >= 0 && x.y >= 0) {
// Get the incremental impulse
// b2Vec2 d = x - a;
b2Vec2.SubVV(x, a, d);
// Apply incremental impulse
// b2Vec2 P1 = d.x * normal;
b2Vec2.MulSV(d.x, normal, P1);
// b2Vec2 P2 = d.y * normal;
b2Vec2.MulSV(d.y, normal, P2);
b2Vec2.AddVV(P1, P2, P1P2);
// vA -= mA * (P1 + P2);
vA.SelfMulSub(mA, P1P2);
// wA -= iA * (b2Cross(cp1->rA, P1) + b2Cross(cp2->rA, P2));
wA -= iA * (b2Vec2.CrossVV(cp1.rA, P1) + b2Vec2.CrossVV(cp2.rA, P2));
// vB += mB * (P1 + P2);
vB.SelfMulAdd(mB, P1P2);
// wB += iB * (b2Cross(cp1->rB, P1) + b2Cross(cp2->rB, P2));
wB += iB * (b2Vec2.CrossVV(cp1.rB, P1) + b2Vec2.CrossVV(cp2.rB, P2));
// Accumulate
cp1.normalImpulse = x.x;
cp2.normalImpulse = x.y;
/*
#if B2_DEBUG_SOLVER === 1
// Postconditions
dv1 = vB + b2Cross(wB, cp1->rB) - vA - b2Cross(wA, cp1->rA);
dv2 = vB + b2Cross(wB, cp2->rB) - vA - b2Cross(wA, cp2->rA);
// Compute normal velocity
vn1 = b2Dot(dv1, normal);
vn2 = b2Dot(dv2, normal);
b2Assert(b2Abs(vn1 - cp1->velocityBias) < k_errorTol);
b2Assert(b2Abs(vn2 - cp2->velocityBias) < k_errorTol);
#endif
*/
break;
}
//
// Case 2: vn1 = 0 and x2 = 0
//
// 0 = a11 * x1 + a12 * 0 + b1'
// vn2 = a21 * x1 + a22 * 0 + b2'
//
x.x = (-cp1.normalMass * b.x);
x.y = 0;
vn1 = 0;
vn2 = vc.K.ex.y * x.x + b.y;
if (x.x >= 0 && vn2 >= 0) {
// Get the incremental impulse
// b2Vec2 d = x - a;
b2Vec2.SubVV(x, a, d);
// Apply incremental impulse
// b2Vec2 P1 = d.x * normal;
b2Vec2.MulSV(d.x, normal, P1);
// b2Vec2 P2 = d.y * normal;
b2Vec2.MulSV(d.y, normal, P2);
b2Vec2.AddVV(P1, P2, P1P2);
// vA -= mA * (P1 + P2);
vA.SelfMulSub(mA, P1P2);
// wA -= iA * (b2Cross(cp1->rA, P1) + b2Cross(cp2->rA, P2));
wA -= iA * (b2Vec2.CrossVV(cp1.rA, P1) + b2Vec2.CrossVV(cp2.rA, P2));
// vB += mB * (P1 + P2);
vB.SelfMulAdd(mB, P1P2);
// wB += iB * (b2Cross(cp1->rB, P1) + b2Cross(cp2->rB, P2));
wB += iB * (b2Vec2.CrossVV(cp1.rB, P1) + b2Vec2.CrossVV(cp2.rB, P2));
// Accumulate
cp1.normalImpulse = x.x;
cp2.normalImpulse = x.y;
/*
#if B2_DEBUG_SOLVER === 1
// Postconditions
dv1 = vB + b2Cross(wB, cp1->rB) - vA - b2Cross(wA, cp1->rA);
// Compute normal velocity
vn1 = b2Dot(dv1, normal);
b2Assert(b2Abs(vn1 - cp1->velocityBias) < k_errorTol);
#endif
*/
break;
}
//
// Case 3: vn2 = 0 and x1 = 0
//
// vn1 = a11 * 0 + a12 * x2 + b1'
// 0 = a21 * 0 + a22 * x2 + b2'
//
x.x = 0;
x.y = (-cp2.normalMass * b.y);
vn1 = vc.K.ey.x * x.y + b.x;
vn2 = 0;
if (x.y >= 0 && vn1 >= 0) {
// Resubstitute for the incremental impulse
// b2Vec2 d = x - a;
b2Vec2.SubVV(x, a, d);
// Apply incremental impulse
// b2Vec2 P1 = d.x * normal;
b2Vec2.MulSV(d.x, normal, P1);
// b2Vec2 P2 = d.y * normal;
b2Vec2.MulSV(d.y, normal, P2);
b2Vec2.AddVV(P1, P2, P1P2);
// vA -= mA * (P1 + P2);
vA.SelfMulSub(mA, P1P2);
// wA -= iA * (b2Cross(cp1->rA, P1) + b2Cross(cp2->rA, P2));
wA -= iA * (b2Vec2.CrossVV(cp1.rA, P1) + b2Vec2.CrossVV(cp2.rA, P2));
// vB += mB * (P1 + P2);
vB.SelfMulAdd(mB, P1P2);
// wB += iB * (b2Cross(cp1->rB, P1) + b2Cross(cp2->rB, P2));
wB += iB * (b2Vec2.CrossVV(cp1.rB, P1) + b2Vec2.CrossVV(cp2.rB, P2));
// Accumulate
cp1.normalImpulse = x.x;
cp2.normalImpulse = x.y;
/*
#if B2_DEBUG_SOLVER === 1
// Postconditions
dv2 = vB + b2Cross(wB, cp2->rB) - vA - b2Cross(wA, cp2->rA);
// Compute normal velocity
vn2 = b2Dot(dv2, normal);
b2Assert(b2Abs(vn2 - cp2->velocityBias) < k_errorTol);
#endif
*/
break;
}
//
// Case 4: x1 = 0 and x2 = 0
//
// vn1 = b1
// vn2 = b2;
x.x = 0;
x.y = 0;
vn1 = b.x;
vn2 = b.y;
if (vn1 >= 0 && vn2 >= 0) {
// Resubstitute for the incremental impulse
// b2Vec2 d = x - a;
b2Vec2.SubVV(x, a, d);
// Apply incremental impulse
// b2Vec2 P1 = d.x * normal;
b2Vec2.MulSV(d.x, normal, P1);
// b2Vec2 P2 = d.y * normal;
b2Vec2.MulSV(d.y, normal, P2);
b2Vec2.AddVV(P1, P2, P1P2);
// vA -= mA * (P1 + P2);
vA.SelfMulSub(mA, P1P2);
// wA -= iA * (b2Cross(cp1->rA, P1) + b2Cross(cp2->rA, P2));
wA -= iA * (b2Vec2.CrossVV(cp1.rA, P1) + b2Vec2.CrossVV(cp2.rA, P2));
// vB += mB * (P1 + P2);
vB.SelfMulAdd(mB, P1P2);
// wB += iB * (b2Cross(cp1->rB, P1) + b2Cross(cp2->rB, P2));
wB += iB * (b2Vec2.CrossVV(cp1.rB, P1) + b2Vec2.CrossVV(cp2.rB, P2));
// Accumulate
cp1.normalImpulse = x.x;
cp2.normalImpulse = x.y;
break;
}
// No solution, give up. This is hit sometimes, but it doesn't seem to matter.
break;
}
}
// this.m_velocities[indexA].v = vA;
this.m_velocities[indexA].w = wA;
// this.m_velocities[indexB].v = vB;
this.m_velocities[indexB].w = wB;
}
};
b2ContactSolver.prototype.StoreImpulses = function () {
for (var i = 0; i < this.m_count; ++i) {
var vc = this.m_velocityConstraints[i];
var manifold = this.m_contacts[vc.contactIndex].GetManifold();
for (var j = 0; j < vc.pointCount; ++j) {
manifold.points[j].normalImpulse = vc.points[j].normalImpulse;
manifold.points[j].tangentImpulse = vc.points[j].tangentImpulse;
}
}
};
b2ContactSolver.prototype.SolvePositionConstraints = function () {
var xfA = b2ContactSolver.SolvePositionConstraints_s_xfA;
var xfB = b2ContactSolver.SolvePositionConstraints_s_xfB;
var psm = b2ContactSolver.SolvePositionConstraints_s_psm;
var rA = b2ContactSolver.SolvePositionConstraints_s_rA;
var rB = b2ContactSolver.SolvePositionConstraints_s_rB;
var P = b2ContactSolver.SolvePositionConstraints_s_P;
var minSeparation = 0;
for (var i = 0; i < this.m_count; ++i) {
var pc = this.m_positionConstraints[i];
var indexA = pc.indexA;
var indexB = pc.indexB;
var localCenterA = pc.localCenterA;
var mA = pc.invMassA;
var iA = pc.invIA;
var localCenterB = pc.localCenterB;
var mB = pc.invMassB;
var iB = pc.invIB;
var pointCount = pc.pointCount;
var cA = this.m_positions[indexA].c;
var aA = this.m_positions[indexA].a;
var cB = this.m_positions[indexB].c;
var aB = this.m_positions[indexB].a;
// Solve normal constraints
for (var j = 0; j < pointCount; ++j) {
xfA.q.SetAngle(aA);
xfB.q.SetAngle(aB);
b2Vec2.SubVV(cA, b2Rot.MulRV(xfA.q, localCenterA, b2Vec2.s_t0), xfA.p);
b2Vec2.SubVV(cB, b2Rot.MulRV(xfB.q, localCenterB, b2Vec2.s_t0), xfB.p);
psm.Initialize(pc, xfA, xfB, j);
var normal = psm.normal;
var point = psm.point;
var separation = psm.separation;
// b2Vec2 rA = point - cA;
b2Vec2.SubVV(point, cA, rA);
// b2Vec2 rB = point - cB;
b2Vec2.SubVV(point, cB, rB);
// Track max constraint error.
minSeparation = b2Min(minSeparation, separation);
// Prevent large corrections and allow slop.
var C = b2Clamp(b2_baumgarte * (separation + b2_linearSlop), (-b2_maxLinearCorrection), 0);
// Compute the effective mass.
// float32 rnA = b2Cross(rA, normal);
var rnA = b2Vec2.CrossVV(rA, normal);
// float32 rnB = b2Cross(rB, normal);
var rnB = b2Vec2.CrossVV(rB, normal);
// float32 K = mA + mB + iA * rnA * rnA + iB * rnB * rnB;
var K = mA + mB + iA * rnA * rnA + iB * rnB * rnB;
// Compute normal impulse
var impulse = K > 0 ? -C / K : 0;
// b2Vec2 P = impulse * normal;
b2Vec2.MulSV(impulse, normal, P);
// cA -= mA * P;
cA.SelfMulSub(mA, P);
// aA -= iA * b2Cross(rA, P);
aA -= iA * b2Vec2.CrossVV(rA, P);
// cB += mB * P;
cB.SelfMulAdd(mB, P);
// aB += iB * b2Cross(rB, P);
aB += iB * b2Vec2.CrossVV(rB, P);
}
// this.m_positions[indexA].c = cA;
this.m_positions[indexA].a = aA;
// this.m_positions[indexB].c = cB;
this.m_positions[indexB].a = aB;
}
// We can't expect minSpeparation >= -b2_linearSlop because we don't
// push the separation above -b2_linearSlop.
return minSeparation > (-3 * b2_linearSlop);
};
b2ContactSolver.prototype.SolveTOIPositionConstraints = function (toiIndexA, toiIndexB) {
var xfA = b2ContactSolver.SolveTOIPositionConstraints_s_xfA;
var xfB = b2ContactSolver.SolveTOIPositionConstraints_s_xfB;
var psm = b2ContactSolver.SolveTOIPositionConstraints_s_psm;
var rA = b2ContactSolver.SolveTOIPositionConstraints_s_rA;
var rB = b2ContactSolver.SolveTOIPositionConstraints_s_rB;
var P = b2ContactSolver.SolveTOIPositionConstraints_s_P;
var minSeparation = 0;
for (var i = 0; i < this.m_count; ++i) {
var pc = this.m_positionConstraints[i];
var indexA = pc.indexA;
var indexB = pc.indexB;
var localCenterA = pc.localCenterA;
var localCenterB = pc.localCenterB;
var pointCount = pc.pointCount;
var mA = 0;
var iA = 0;
if (indexA === toiIndexA || indexA === toiIndexB) {
mA = pc.invMassA;
iA = pc.invIA;
}
var mB = 0;
var iB = 0;
if (indexB === toiIndexA || indexB === toiIndexB) {
mB = pc.invMassB;
iB = pc.invIB;
}
var cA = this.m_positions[indexA].c;
var aA = this.m_positions[indexA].a;
var cB = this.m_positions[indexB].c;
var aB = this.m_positions[indexB].a;
// Solve normal constraints
for (var j = 0; j < pointCount; ++j) {
xfA.q.SetAngle(aA);
xfB.q.SetAngle(aB);
b2Vec2.SubVV(cA, b2Rot.MulRV(xfA.q, localCenterA, b2Vec2.s_t0), xfA.p);
b2Vec2.SubVV(cB, b2Rot.MulRV(xfB.q, localCenterB, b2Vec2.s_t0), xfB.p);
psm.Initialize(pc, xfA, xfB, j);
var normal = psm.normal;
var point = psm.point;
var separation = psm.separation;
// b2Vec2 rA = point - cA;
b2Vec2.SubVV(point, cA, rA);
// b2Vec2 rB = point - cB;
b2Vec2.SubVV(point, cB, rB);
// Track max constraint error.
minSeparation = b2Min(minSeparation, separation);
// Prevent large corrections and allow slop.
var C = b2Clamp(b2_toiBaumgarte * (separation + b2_linearSlop), (-b2_maxLinearCorrection), 0);
// Compute the effective mass.
// float32 rnA = b2Cross(rA, normal);
var rnA = b2Vec2.CrossVV(rA, normal);
// float32 rnB = b2Cross(rB, normal);
var rnB = b2Vec2.CrossVV(rB, normal);
// float32 K = mA + mB + iA * rnA * rnA + iB * rnB * rnB;
var K = mA + mB + iA * rnA * rnA + iB * rnB * rnB;
// Compute normal impulse
var impulse = K > 0 ? -C / K : 0;
// b2Vec2 P = impulse * normal;
b2Vec2.MulSV(impulse, normal, P);
// cA -= mA * P;
cA.SelfMulSub(mA, P);
// aA -= iA * b2Cross(rA, P);
aA -= iA * b2Vec2.CrossVV(rA, P);
// cB += mB * P;
cB.SelfMulAdd(mB, P);
// aB += iB * b2Cross(rB, P);
aB += iB * b2Vec2.CrossVV(rB, P);
}
// this.m_positions[indexA].c = cA;
this.m_positions[indexA].a = aA;
// this.m_positions[indexB].c = cB;
this.m_positions[indexB].a = aB;
}
// We can't expect minSpeparation >= -b2_linearSlop because we don't
// push the separation above -b2_linearSlop.
return minSeparation >= -1.5 * b2_linearSlop;
};
b2ContactSolver.InitializeVelocityConstraints_s_xfA = new b2Transform();
b2ContactSolver.InitializeVelocityConstraints_s_xfB = new b2Transform();
b2ContactSolver.InitializeVelocityConstraints_s_worldManifold = new b2WorldManifold();
b2ContactSolver.WarmStart_s_P = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_dv = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_dv1 = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_dv2 = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_P = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_a = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_b = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_x = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_d = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_P1 = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_P2 = new b2Vec2();
b2ContactSolver.SolveVelocityConstraints_s_P1P2 = new b2Vec2();
b2ContactSolver.SolvePositionConstraints_s_xfA = new b2Transform();
b2ContactSolver.SolvePositionConstraints_s_xfB = new b2Transform();
b2ContactSolver.SolvePositionConstraints_s_psm = new b2PositionSolverManifold();
b2ContactSolver.SolvePositionConstraints_s_rA = new b2Vec2();
b2ContactSolver.SolvePositionConstraints_s_rB = new b2Vec2();
b2ContactSolver.SolvePositionConstraints_s_P = new b2Vec2();
b2ContactSolver.SolveTOIPositionConstraints_s_xfA = new b2Transform();
b2ContactSolver.SolveTOIPositionConstraints_s_xfB = new b2Transform();
b2ContactSolver.SolveTOIPositionConstraints_s_psm = new b2PositionSolverManifold();
b2ContactSolver.SolveTOIPositionConstraints_s_rA = new b2Vec2();
b2ContactSolver.SolveTOIPositionConstraints_s_rB = new b2Vec2();
b2ContactSolver.SolveTOIPositionConstraints_s_P = new b2Vec2();
return b2ContactSolver;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/*
Position Correction Notes
=========================
I tried the several algorithms for position correction of the 2D revolute joint.
I looked at these systems:
- simple pendulum (1m diameter sphere on massless 5m stick) with initial angular velocity of 100 rad/s.
- suspension bridge with 30 1m long planks of length 1m.
- multi-link chain with 30 1m long links.
Here are the algorithms:
Baumgarte - A fraction of the position error is added to the velocity error. There is no
separate position solver.
Pseudo Velocities - After the velocity solver and position integration,
the position error, Jacobian, and effective mass are recomputed. Then
the velocity constraints are solved with pseudo velocities and a fraction
of the position error is added to the pseudo velocity error. The pseudo
velocities are initialized to zero and there is no warm-starting. After
the position solver, the pseudo velocities are added to the positions.
This is also called the First Order World method or the Position LCP method.
Modified Nonlinear Gauss-Seidel (NGS) - Like Pseudo Velocities except the
position error is re-computed for each constraint and the positions are updated
after the constraint is solved. The radius vectors (aka Jacobians) are
re-computed too (otherwise the algorithm has horrible instability). The pseudo
velocity states are not needed because they are effectively zero at the beginning
of each iteration. Since we have the current position error, we allow the
iterations to terminate early if the error becomes smaller than b2_linearSlop.
Full NGS or just NGS - Like Modified NGS except the effective mass are re-computed
each time a constraint is solved.
Here are the results:
Baumgarte - this is the cheapest algorithm but it has some stability problems,
especially with the bridge. The chain links separate easily close to the root
and they jitter as they struggle to pull together. This is one of the most common
methods in the field. The big drawback is that the position correction artificially
affects the momentum, thus leading to instabilities and false bounce. I used a
bias factor of 0.2. A larger bias factor makes the bridge less stable, a smaller
factor makes joints and contacts more spongy.
Pseudo Velocities - the is more stable than the Baumgarte method. The bridge is
stable. However, joints still separate with large angular velocities. Drag the
simple pendulum in a circle quickly and the joint will separate. The chain separates
easily and does not recover. I used a bias factor of 0.2. A larger value lead to
the bridge collapsing when a heavy cube drops on it.
Modified NGS - this algorithm is better in some ways than Baumgarte and Pseudo
Velocities, but in other ways it is worse. The bridge and chain are much more
stable, but the simple pendulum goes unstable at high angular velocities.
Full NGS - stable in all tests. The joints display good stiffness. The bridge
still sags, but this is better than infinite forces.
Recommendations
Pseudo Velocities are not really worthwhile because the bridge and chain cannot
recover from joint separation. In other cases the benefit over Baumgarte is small.
Modified NGS is not a robust method for the revolute joint due to the violent
instability seen in the simple pendulum. Perhaps it is viable with other constraint
types, especially scalar constraints where the effective mass is a scalar.
This leaves Baumgarte and Full NGS. Baumgarte has small, but manageable instabilities
and is very fast. I don't think we can escape Baumgarte, especially in highly
demanding cases where high constraint fidelity is not needed.
Full NGS is robust and easy on the eyes. I recommend this as an option for
higher fidelity simulation and certainly for suspension bridges and long chains.
Full NGS might be a good choice for ragdolls, especially motorized ragdolls where
joint separation can be problematic. The number of NGS iterations can be reduced
for better performance without harming robustness much.
Each joint in a can be handled differently in the position solver. So I recommend
a system where the user can select the algorithm on a per joint basis. I would
probably default to the slower Full NGS and let the user select the faster
Baumgarte method in performance critical scenarios.
*/
/*
Cache Performance
The Box2D solvers are dominated by cache misses. Data structures are designed
to increase the number of cache hits. Much of misses are due to random access
to body data. The constraint structures are iterated over linearly, which leads
to few cache misses.
The bodies are not accessed during iteration. Instead read only data, such as
the mass values are stored with the constraints. The mutable data are the constraint
impulses and the bodies velocities/positions. The impulses are held inside the
constraint structures. The body velocities/positions are held in compact, temporary
arrays to increase the number of cache hits. Linear and angular velocity are
stored in a single array since multiple arrays lead to multiple misses.
*/
/*
2D Rotation
R = [cos(theta) -sin(theta)]
[sin(theta) cos(theta) ]
thetaDot = omega
Let q1 = cos(theta), q2 = sin(theta).
R = [q1 -q2]
[q2 q1]
q1Dot = -thetaDot * q2
q2Dot = thetaDot * q1
q1_new = q1_old - dt * w * q2
q2_new = q2_old + dt * w * q1
then normalize.
This might be faster than computing sin+cos.
However, we can compute sin+cos of the same angle fast.
*/
var b2Island = /** @class */ (function () {
function b2Island() {
this.m_allocator = null;
this.m_bodies = [ /*1024*/]; // TODO: b2Settings
this.m_contacts = [ /*1024*/]; // TODO: b2Settings
this.m_joints = [ /*1024*/]; // TODO: b2Settings
this.m_positions = b2Position.MakeArray(1024); // TODO: b2Settings
this.m_velocities = b2Velocity.MakeArray(1024); // TODO: b2Settings
this.m_bodyCount = 0;
this.m_jointCount = 0;
this.m_contactCount = 0;
this.m_bodyCapacity = 0;
this.m_contactCapacity = 0;
this.m_jointCapacity = 0;
}
b2Island.prototype.Initialize = function (bodyCapacity, contactCapacity, jointCapacity, allocator, listener) {
this.m_bodyCapacity = bodyCapacity;
this.m_contactCapacity = contactCapacity;
this.m_jointCapacity = jointCapacity;
this.m_bodyCount = 0;
this.m_contactCount = 0;
this.m_jointCount = 0;
this.m_allocator = allocator;
this.m_listener = listener;
// TODO:
// while (this.m_bodies.length < bodyCapacity) {
// this.m_bodies[this.m_bodies.length] = null;
// }
// TODO:
// while (this.m_contacts.length < contactCapacity) {
// this.m_contacts[this.m_contacts.length] = null;
// }
// TODO:
// while (this.m_joints.length < jointCapacity) {
// this.m_joints[this.m_joints.length] = null;
// }
// TODO:
if (this.m_positions.length < bodyCapacity) {
var new_length = b2Max(this.m_positions.length * 2, bodyCapacity);
while (this.m_positions.length < new_length) {
this.m_positions[this.m_positions.length] = new b2Position();
}
}
// TODO:
if (this.m_velocities.length < bodyCapacity) {
var new_length = b2Max(this.m_velocities.length * 2, bodyCapacity);
while (this.m_velocities.length < new_length) {
this.m_velocities[this.m_velocities.length] = new b2Velocity();
}
}
};
b2Island.prototype.Clear = function () {
this.m_bodyCount = 0;
this.m_contactCount = 0;
this.m_jointCount = 0;
};
b2Island.prototype.AddBody = function (body) {
// DEBUG: b2Assert(this.m_bodyCount < this.m_bodyCapacity);
body.m_islandIndex = this.m_bodyCount;
this.m_bodies[this.m_bodyCount++] = body;
};
b2Island.prototype.AddContact = function (contact) {
// DEBUG: b2Assert(this.m_contactCount < this.m_contactCapacity);
this.m_contacts[this.m_contactCount++] = contact;
};
b2Island.prototype.AddJoint = function (joint) {
// DEBUG: b2Assert(this.m_jointCount < this.m_jointCapacity);
this.m_joints[this.m_jointCount++] = joint;
};
b2Island.prototype.Solve = function (profile, step, gravity, allowSleep) {
var timer = b2Island.s_timer.Reset();
var h = step.dt;
// Integrate velocities and apply damping. Initialize the body state.
for (var i = 0; i < this.m_bodyCount; ++i) {
var b = this.m_bodies[i];
// const c: b2Vec2 =
this.m_positions[i].c.Copy(b.m_sweep.c);
var a = b.m_sweep.a;
var v = this.m_velocities[i].v.Copy(b.m_linearVelocity);
var w = b.m_angularVelocity;
// Store positions for continuous collision.
b.m_sweep.c0.Copy(b.m_sweep.c);
b.m_sweep.a0 = b.m_sweep.a;
if (b.m_type === exports.b2BodyType.b2_dynamicBody) {
// Integrate velocities.
v.x += h * (b.m_gravityScale * gravity.x + b.m_invMass * b.m_force.x);
v.y += h * (b.m_gravityScale * gravity.y + b.m_invMass * b.m_force.y);
w += h * b.m_invI * b.m_torque;
// Apply damping.
// ODE: dv/dt + c * v = 0
// Solution: v(t) = v0 * exp(-c * t)
// Time step: v(t + dt) = v0 * exp(-c * (t + dt)) = v0 * exp(-c * t) * exp(-c * dt) = v * exp(-c * dt)
// v2 = exp(-c * dt) * v1
// Pade approximation:
// v2 = v1 * 1 / (1 + c * dt)
v.SelfMul(1.0 / (1.0 + h * b.m_linearDamping));
w *= 1.0 / (1.0 + h * b.m_angularDamping);
}
// this.m_positions[i].c = c;
this.m_positions[i].a = a;
// this.m_velocities[i].v = v;
this.m_velocities[i].w = w;
}
timer.Reset();
// Solver data
var solverData = b2Island.s_solverData;
solverData.step.Copy(step);
solverData.positions = this.m_positions;
solverData.velocities = this.m_velocities;
// Initialize velocity constraints.
var contactSolverDef = b2Island.s_contactSolverDef;
contactSolverDef.step.Copy(step);
contactSolverDef.contacts = this.m_contacts;
contactSolverDef.count = this.m_contactCount;
contactSolverDef.positions = this.m_positions;
contactSolverDef.velocities = this.m_velocities;
contactSolverDef.allocator = this.m_allocator;
var contactSolver = b2Island.s_contactSolver.Initialize(contactSolverDef);
contactSolver.InitializeVelocityConstraints();
if (step.warmStarting) {
contactSolver.WarmStart();
}
for (var i = 0; i < this.m_jointCount; ++i) {
this.m_joints[i].InitVelocityConstraints(solverData);
}
profile.solveInit = timer.GetMilliseconds();
// Solve velocity constraints.
timer.Reset();
for (var i = 0; i < step.velocityIterations; ++i) {
for (var j = 0; j < this.m_jointCount; ++j) {
this.m_joints[j].SolveVelocityConstraints(solverData);
}
contactSolver.SolveVelocityConstraints();
}
// Store impulses for warm starting
contactSolver.StoreImpulses();
profile.solveVelocity = timer.GetMilliseconds();
// Integrate positions.
for (var i = 0; i < this.m_bodyCount; ++i) {
var c = this.m_positions[i].c;
var a = this.m_positions[i].a;
var v = this.m_velocities[i].v;
var w = this.m_velocities[i].w;
// Check for large velocities
var translation = b2Vec2.MulSV(h, v, b2Island.s_translation);
if (b2Vec2.DotVV(translation, translation) > b2_maxTranslationSquared) {
var ratio = b2_maxTranslation / translation.Length();
v.SelfMul(ratio);
}
var rotation = h * w;
if (rotation * rotation > b2_maxRotationSquared) {
var ratio = b2_maxRotation / b2Abs(rotation);
w *= ratio;
}
// Integrate
c.x += h * v.x;
c.y += h * v.y;
a += h * w;
// this.m_positions[i].c = c;
this.m_positions[i].a = a;
// this.m_velocities[i].v = v;
this.m_velocities[i].w = w;
}
// Solve position constraints
timer.Reset();
var positionSolved = false;
for (var i = 0; i < step.positionIterations; ++i) {
var contactsOkay = contactSolver.SolvePositionConstraints();
var jointsOkay = true;
for (var j = 0; j < this.m_jointCount; ++j) {
var jointOkay = this.m_joints[j].SolvePositionConstraints(solverData);
jointsOkay = jointsOkay && jointOkay;
}
if (contactsOkay && jointsOkay) {
// Exit early if the position errors are small.
positionSolved = true;
break;
}
}
// Copy state buffers back to the bodies
for (var i = 0; i < this.m_bodyCount; ++i) {
var body = this.m_bodies[i];
body.m_sweep.c.Copy(this.m_positions[i].c);
body.m_sweep.a = this.m_positions[i].a;
body.m_linearVelocity.Copy(this.m_velocities[i].v);
body.m_angularVelocity = this.m_velocities[i].w;
body.SynchronizeTransform();
}
profile.solvePosition = timer.GetMilliseconds();
this.Report(contactSolver.m_velocityConstraints);
if (allowSleep) {
var minSleepTime = b2_maxFloat;
var linTolSqr = b2_linearSleepTolerance * b2_linearSleepTolerance;
var angTolSqr = b2_angularSleepTolerance * b2_angularSleepTolerance;
for (var i = 0; i < this.m_bodyCount; ++i) {
var b = this.m_bodies[i];
if (b.GetType() === exports.b2BodyType.b2_staticBody) {
continue;
}
if (!b.m_autoSleepFlag ||
b.m_angularVelocity * b.m_angularVelocity > angTolSqr ||
b2Vec2.DotVV(b.m_linearVelocity, b.m_linearVelocity) > linTolSqr) {
b.m_sleepTime = 0;
minSleepTime = 0;
}
else {
b.m_sleepTime += h;
minSleepTime = b2Min(minSleepTime, b.m_sleepTime);
}
}
if (minSleepTime >= b2_timeToSleep && positionSolved) {
for (var i = 0; i < this.m_bodyCount; ++i) {
var b = this.m_bodies[i];
b.SetAwake(false);
}
}
}
};
b2Island.prototype.SolveTOI = function (subStep, toiIndexA, toiIndexB) {
// DEBUG: b2Assert(toiIndexA < this.m_bodyCount);
// DEBUG: b2Assert(toiIndexB < this.m_bodyCount);
// Initialize the body state.
for (var i = 0; i < this.m_bodyCount; ++i) {
var b = this.m_bodies[i];
this.m_positions[i].c.Copy(b.m_sweep.c);
this.m_positions[i].a = b.m_sweep.a;
this.m_velocities[i].v.Copy(b.m_linearVelocity);
this.m_velocities[i].w = b.m_angularVelocity;
}
var contactSolverDef = b2Island.s_contactSolverDef;
contactSolverDef.contacts = this.m_contacts;
contactSolverDef.count = this.m_contactCount;
contactSolverDef.allocator = this.m_allocator;
contactSolverDef.step.Copy(subStep);
contactSolverDef.positions = this.m_positions;
contactSolverDef.velocities = this.m_velocities;
var contactSolver = b2Island.s_contactSolver.Initialize(contactSolverDef);
// Solve position constraints.
for (var i = 0; i < subStep.positionIterations; ++i) {
var contactsOkay = contactSolver.SolveTOIPositionConstraints(toiIndexA, toiIndexB);
if (contactsOkay) {
break;
}
}
/*
#if 0
// Is the new position really safe?
for (int32 i = 0; i < this.m_contactCount; ++i) {
b2Contact* c = this.m_contacts[i];
b2Fixture* fA = c.GetFixtureA();
b2Fixture* fB = c.GetFixtureB();
b2Body* bA = fA.GetBody();
b2Body* bB = fB.GetBody();
int32 indexA = c.GetChildIndexA();
int32 indexB = c.GetChildIndexB();
b2DistanceInput input;
input.proxyA.Set(fA.GetShape(), indexA);
input.proxyB.Set(fB.GetShape(), indexB);
input.transformA = bA.GetTransform();
input.transformB = bB.GetTransform();
input.useRadii = false;
b2DistanceOutput output;
b2SimplexCache cache;
cache.count = 0;
b2Distance(&output, &cache, &input);
if (output.distance === 0 || cache.count === 3) {
cache.count += 0;
}
}
#endif
*/
// Leap of faith to new safe state.
this.m_bodies[toiIndexA].m_sweep.c0.Copy(this.m_positions[toiIndexA].c);
this.m_bodies[toiIndexA].m_sweep.a0 = this.m_positions[toiIndexA].a;
this.m_bodies[toiIndexB].m_sweep.c0.Copy(this.m_positions[toiIndexB].c);
this.m_bodies[toiIndexB].m_sweep.a0 = this.m_positions[toiIndexB].a;
// No warm starting is needed for TOI events because warm
// starting impulses were applied in the discrete solver.
contactSolver.InitializeVelocityConstraints();
// Solve velocity constraints.
for (var i = 0; i < subStep.velocityIterations; ++i) {
contactSolver.SolveVelocityConstraints();
}
// Don't store the TOI contact forces for warm starting
// because they can be quite large.
var h = subStep.dt;
// Integrate positions
for (var i = 0; i < this.m_bodyCount; ++i) {
var c = this.m_positions[i].c;
var a = this.m_positions[i].a;
var v = this.m_velocities[i].v;
var w = this.m_velocities[i].w;
// Check for large velocities
var translation = b2Vec2.MulSV(h, v, b2Island.s_translation);
if (b2Vec2.DotVV(translation, translation) > b2_maxTranslationSquared) {
var ratio = b2_maxTranslation / translation.Length();
v.SelfMul(ratio);
}
var rotation = h * w;
if (rotation * rotation > b2_maxRotationSquared) {
var ratio = b2_maxRotation / b2Abs(rotation);
w *= ratio;
}
// Integrate
c.SelfMulAdd(h, v);
a += h * w;
// this.m_positions[i].c = c;
this.m_positions[i].a = a;
// this.m_velocities[i].v = v;
this.m_velocities[i].w = w;
// Sync bodies
var body = this.m_bodies[i];
body.m_sweep.c.Copy(c);
body.m_sweep.a = a;
body.m_linearVelocity.Copy(v);
body.m_angularVelocity = w;
body.SynchronizeTransform();
}
this.Report(contactSolver.m_velocityConstraints);
};
b2Island.prototype.Report = function (constraints) {
if (this.m_listener === null) {
return;
}
for (var i = 0; i < this.m_contactCount; ++i) {
var c = this.m_contacts[i];
if (!c) {
continue;
}
var vc = constraints[i];
var impulse = b2Island.s_impulse;
impulse.count = vc.pointCount;
for (var j = 0; j < vc.pointCount; ++j) {
impulse.normalImpulses[j] = vc.points[j].normalImpulse;
impulse.tangentImpulses[j] = vc.points[j].tangentImpulse;
}
this.m_listener.PostSolve(c, impulse);
}
};
b2Island.s_timer = new b2Timer();
b2Island.s_solverData = new b2SolverData();
b2Island.s_contactSolverDef = new b2ContactSolverDef();
b2Island.s_contactSolver = new b2ContactSolver();
b2Island.s_translation = new b2Vec2();
b2Island.s_impulse = new b2ContactImpulse();
return b2Island;
}());
/*
* Copyright (c) 2013 Google, Inc.
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
(function (b2ParticleFlag) {
/// Water particle.
b2ParticleFlag[b2ParticleFlag["b2_waterParticle"] = 0] = "b2_waterParticle";
/// Removed after next simulation step.
b2ParticleFlag[b2ParticleFlag["b2_zombieParticle"] = 2] = "b2_zombieParticle";
/// Zero velocity.
b2ParticleFlag[b2ParticleFlag["b2_wallParticle"] = 4] = "b2_wallParticle";
/// With restitution from stretching.
b2ParticleFlag[b2ParticleFlag["b2_springParticle"] = 8] = "b2_springParticle";
/// With restitution from deformation.
b2ParticleFlag[b2ParticleFlag["b2_elasticParticle"] = 16] = "b2_elasticParticle";
/// With viscosity.
b2ParticleFlag[b2ParticleFlag["b2_viscousParticle"] = 32] = "b2_viscousParticle";
/// Without isotropic pressure.
b2ParticleFlag[b2ParticleFlag["b2_powderParticle"] = 64] = "b2_powderParticle";
/// With surface tension.
b2ParticleFlag[b2ParticleFlag["b2_tensileParticle"] = 128] = "b2_tensileParticle";
/// Mix color between contacting particles.
b2ParticleFlag[b2ParticleFlag["b2_colorMixingParticle"] = 256] = "b2_colorMixingParticle";
/// Call b2DestructionListener on destruction.
b2ParticleFlag[b2ParticleFlag["b2_destructionListenerParticle"] = 512] = "b2_destructionListenerParticle";
/// Prevents other particles from leaking.
b2ParticleFlag[b2ParticleFlag["b2_barrierParticle"] = 1024] = "b2_barrierParticle";
/// Less compressibility.
b2ParticleFlag[b2ParticleFlag["b2_staticPressureParticle"] = 2048] = "b2_staticPressureParticle";
/// Makes pairs or triads with other particles.
b2ParticleFlag[b2ParticleFlag["b2_reactiveParticle"] = 4096] = "b2_reactiveParticle";
/// With high repulsive force.
b2ParticleFlag[b2ParticleFlag["b2_repulsiveParticle"] = 8192] = "b2_repulsiveParticle";
/// Call b2ContactListener when this particle is about to interact with
/// a rigid body or stops interacting with a rigid body.
/// This results in an expensive operation compared to using
/// b2_fixtureContactFilterParticle to detect collisions between
/// particles.
b2ParticleFlag[b2ParticleFlag["b2_fixtureContactListenerParticle"] = 16384] = "b2_fixtureContactListenerParticle";
/// Call b2ContactListener when this particle is about to interact with
/// another particle or stops interacting with another particle.
/// This results in an expensive operation compared to using
/// b2_particleContactFilterParticle to detect collisions between
/// particles.
b2ParticleFlag[b2ParticleFlag["b2_particleContactListenerParticle"] = 32768] = "b2_particleContactListenerParticle";
/// Call b2ContactFilter when this particle interacts with rigid bodies.
b2ParticleFlag[b2ParticleFlag["b2_fixtureContactFilterParticle"] = 65536] = "b2_fixtureContactFilterParticle";
/// Call b2ContactFilter when this particle interacts with other
/// particles.
b2ParticleFlag[b2ParticleFlag["b2_particleContactFilterParticle"] = 131072] = "b2_particleContactFilterParticle";
})(exports.b2ParticleFlag || (exports.b2ParticleFlag = {}));
var b2ParticleDef = /** @class */ (function () {
function b2ParticleDef() {
this.flags = 0;
this.position = new b2Vec2();
this.velocity = new b2Vec2();
this.color = new b2Color(0, 0, 0, 0);
this.lifetime = 0.0;
this.userData = null;
this.group = null;
}
return b2ParticleDef;
}());
function b2CalculateParticleIterations(gravity, radius, timeStep) {
// In some situations you may want more particle iterations than this,
// but to avoid excessive cycle cost, don't recommend more than this.
var B2_MAX_RECOMMENDED_PARTICLE_ITERATIONS = 8;
var B2_RADIUS_THRESHOLD = 0.01;
var iterations = Math.ceil(Math.sqrt(gravity / (B2_RADIUS_THRESHOLD * radius)) * timeStep);
return b2Clamp(iterations, 1, B2_MAX_RECOMMENDED_PARTICLE_ITERATIONS);
}
var b2ParticleHandle = /** @class */ (function () {
function b2ParticleHandle() {
this.m_index = b2_invalidParticleIndex;
}
b2ParticleHandle.prototype.GetIndex = function () { return this.m_index; };
b2ParticleHandle.prototype.SetIndex = function (index) { this.m_index = index; };
return b2ParticleHandle;
}());
// #endif
/*
* Copyright (c) 2013 Google, Inc.
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
(function (b2ParticleGroupFlag) {
/// Prevents overlapping or leaking.
b2ParticleGroupFlag[b2ParticleGroupFlag["b2_solidParticleGroup"] = 1] = "b2_solidParticleGroup";
/// Keeps its shape.
b2ParticleGroupFlag[b2ParticleGroupFlag["b2_rigidParticleGroup"] = 2] = "b2_rigidParticleGroup";
/// Won't be destroyed if it gets empty.
b2ParticleGroupFlag[b2ParticleGroupFlag["b2_particleGroupCanBeEmpty"] = 4] = "b2_particleGroupCanBeEmpty";
/// Will be destroyed on next simulation step.
b2ParticleGroupFlag[b2ParticleGroupFlag["b2_particleGroupWillBeDestroyed"] = 8] = "b2_particleGroupWillBeDestroyed";
/// Updates depth data on next simulation step.
b2ParticleGroupFlag[b2ParticleGroupFlag["b2_particleGroupNeedsUpdateDepth"] = 16] = "b2_particleGroupNeedsUpdateDepth";
b2ParticleGroupFlag[b2ParticleGroupFlag["b2_particleGroupInternalMask"] = 24] = "b2_particleGroupInternalMask";
})(exports.b2ParticleGroupFlag || (exports.b2ParticleGroupFlag = {}));
var b2ParticleGroupDef = /** @class */ (function () {
function b2ParticleGroupDef() {
this.flags = 0;
this.groupFlags = 0;
this.position = new b2Vec2();
this.angle = 0.0;
this.linearVelocity = new b2Vec2();
this.angularVelocity = 0.0;
this.color = new b2Color();
this.strength = 1.0;
this.shapeCount = 0;
this.stride = 0;
this.particleCount = 0;
this.lifetime = 0;
this.userData = null;
this.group = null;
}
return b2ParticleGroupDef;
}());
var b2ParticleGroup = /** @class */ (function () {
function b2ParticleGroup(system) {
this.m_firstIndex = 0;
this.m_lastIndex = 0;
this.m_groupFlags = 0;
this.m_strength = 1.0;
this.m_prev = null;
this.m_next = null;
this.m_timestamp = -1;
this.m_mass = 0.0;
this.m_inertia = 0.0;
this.m_center = new b2Vec2();
this.m_linearVelocity = new b2Vec2();
this.m_angularVelocity = 0.0;
this.m_transform = new b2Transform();
///m_transform.SetIdentity();
this.m_userData = null;
this.m_system = system;
}
b2ParticleGroup.prototype.GetNext = function () {
return this.m_next;
};
b2ParticleGroup.prototype.GetParticleSystem = function () {
return this.m_system;
};
b2ParticleGroup.prototype.GetParticleCount = function () {
return this.m_lastIndex - this.m_firstIndex;
};
b2ParticleGroup.prototype.GetBufferIndex = function () {
return this.m_firstIndex;
};
b2ParticleGroup.prototype.ContainsParticle = function (index) {
return this.m_firstIndex <= index && index < this.m_lastIndex;
};
b2ParticleGroup.prototype.GetAllParticleFlags = function () {
if (!this.m_system.m_flagsBuffer.data) {
throw new Error();
}
var flags = 0;
for (var i = this.m_firstIndex; i < this.m_lastIndex; i++) {
flags |= this.m_system.m_flagsBuffer.data[i];
}
return flags;
};
b2ParticleGroup.prototype.GetGroupFlags = function () {
return this.m_groupFlags;
};
b2ParticleGroup.prototype.SetGroupFlags = function (flags) {
// DEBUG: b2Assert((flags & b2ParticleGroupFlag.b2_particleGroupInternalMask) === 0);
flags |= this.m_groupFlags & exports.b2ParticleGroupFlag.b2_particleGroupInternalMask;
this.m_system.SetGroupFlags(this, flags);
};
b2ParticleGroup.prototype.GetMass = function () {
this.UpdateStatistics();
return this.m_mass;
};
b2ParticleGroup.prototype.GetInertia = function () {
this.UpdateStatistics();
return this.m_inertia;
};
b2ParticleGroup.prototype.GetCenter = function () {
this.UpdateStatistics();
return this.m_center;
};
b2ParticleGroup.prototype.GetLinearVelocity = function () {
this.UpdateStatistics();
return this.m_linearVelocity;
};
b2ParticleGroup.prototype.GetAngularVelocity = function () {
this.UpdateStatistics();
return this.m_angularVelocity;
};
b2ParticleGroup.prototype.GetTransform = function () {
return this.m_transform;
};
b2ParticleGroup.prototype.GetPosition = function () {
return this.m_transform.p;
};
b2ParticleGroup.prototype.GetAngle = function () {
return this.m_transform.q.GetAngle();
};
b2ParticleGroup.prototype.GetLinearVelocityFromWorldPoint = function (worldPoint, out) {
var s_t0 = b2ParticleGroup.GetLinearVelocityFromWorldPoint_s_t0;
this.UpdateStatistics();
/// return m_linearVelocity + b2Cross(m_angularVelocity, worldPoint - m_center);
return b2Vec2.AddVCrossSV(this.m_linearVelocity, this.m_angularVelocity, b2Vec2.SubVV(worldPoint, this.m_center, s_t0), out);
};
b2ParticleGroup.prototype.GetUserData = function () {
return this.m_userData;
};
b2ParticleGroup.prototype.SetUserData = function (data) {
this.m_userData = data;
};
b2ParticleGroup.prototype.ApplyForce = function (force) {
this.m_system.ApplyForce(this.m_firstIndex, this.m_lastIndex, force);
};
b2ParticleGroup.prototype.ApplyLinearImpulse = function (impulse) {
this.m_system.ApplyLinearImpulse(this.m_firstIndex, this.m_lastIndex, impulse);
};
b2ParticleGroup.prototype.DestroyParticles = function (callDestructionListener) {
if (this.m_system.m_world.IsLocked()) {
throw new Error();
}
for (var i = this.m_firstIndex; i < this.m_lastIndex; i++) {
this.m_system.DestroyParticle(i, callDestructionListener);
}
};
b2ParticleGroup.prototype.UpdateStatistics = function () {
if (!this.m_system.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_system.m_velocityBuffer.data) {
throw new Error();
}
var p = new b2Vec2();
var v = new b2Vec2();
if (this.m_timestamp !== this.m_system.m_timestamp) {
var m = this.m_system.GetParticleMass();
/// this.m_mass = 0;
this.m_mass = m * (this.m_lastIndex - this.m_firstIndex);
this.m_center.SetZero();
this.m_linearVelocity.SetZero();
for (var i = this.m_firstIndex; i < this.m_lastIndex; i++) {
/// this.m_mass += m;
/// this.m_center += m * this.m_system.m_positionBuffer.data[i];
this.m_center.SelfMulAdd(m, this.m_system.m_positionBuffer.data[i]);
/// this.m_linearVelocity += m * this.m_system.m_velocityBuffer.data[i];
this.m_linearVelocity.SelfMulAdd(m, this.m_system.m_velocityBuffer.data[i]);
}
if (this.m_mass > 0) {
var inv_mass = 1 / this.m_mass;
///this.m_center *= 1 / this.m_mass;
this.m_center.SelfMul(inv_mass);
///this.m_linearVelocity *= 1 / this.m_mass;
this.m_linearVelocity.SelfMul(inv_mass);
}
this.m_inertia = 0;
this.m_angularVelocity = 0;
for (var i = this.m_firstIndex; i < this.m_lastIndex; i++) {
///b2Vec2 p = this.m_system.m_positionBuffer.data[i] - this.m_center;
b2Vec2.SubVV(this.m_system.m_positionBuffer.data[i], this.m_center, p);
///b2Vec2 v = this.m_system.m_velocityBuffer.data[i] - this.m_linearVelocity;
b2Vec2.SubVV(this.m_system.m_velocityBuffer.data[i], this.m_linearVelocity, v);
this.m_inertia += m * b2Vec2.DotVV(p, p);
this.m_angularVelocity += m * b2Vec2.CrossVV(p, v);
}
if (this.m_inertia > 0) {
this.m_angularVelocity *= 1 / this.m_inertia;
}
this.m_timestamp = this.m_system.m_timestamp;
}
};
b2ParticleGroup.GetLinearVelocityFromWorldPoint_s_t0 = new b2Vec2();
return b2ParticleGroup;
}());
// #endif
/*
* Copyright (c) 2013 Google, Inc.
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
var b2StackQueue = /** @class */ (function () {
function b2StackQueue(capacity) {
this.m_front = 0;
this.m_back = 0;
this.m_capacity = 0;
this.m_buffer = b2MakeArray(capacity, function (index) { return null; });
this.m_capacity = capacity;
}
b2StackQueue.prototype.Push = function (item) {
if (this.m_back >= this.m_capacity) {
for (var i = this.m_front; i < this.m_back; i++) {
this.m_buffer[i - this.m_front] = this.m_buffer[i];
}
this.m_back -= this.m_front;
this.m_front = 0;
if (this.m_back >= this.m_capacity) {
if (this.m_capacity > 0) {
this.m_buffer.concat(b2MakeArray(this.m_capacity, function (index) { return null; }));
this.m_capacity *= 2;
}
else {
this.m_buffer.concat(b2MakeArray(1, function (index) { return null; }));
this.m_capacity = 1;
}
///m_buffer = (T*) m_allocator->Reallocate(m_buffer, sizeof(T) * m_capacity);
}
}
this.m_buffer[this.m_back] = item;
this.m_back++;
};
b2StackQueue.prototype.Pop = function () {
// DEBUG: b2Assert(this.m_front < this.m_back);
this.m_buffer[this.m_front] = null;
this.m_front++;
};
b2StackQueue.prototype.Empty = function () {
// DEBUG: b2Assert(this.m_front <= this.m_back);
return this.m_front === this.m_back;
};
b2StackQueue.prototype.Front = function () {
var item = this.m_buffer[this.m_front];
if (!item) {
throw new Error();
}
return item;
};
return b2StackQueue;
}());
// #endif
/*
* Copyright (c) 2013 Google, Inc.
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/**
* A field representing the nearest generator from each point.
*/
var b2VoronoiDiagram = /** @class */ (function () {
function b2VoronoiDiagram(generatorCapacity) {
this.m_generatorCapacity = 0;
this.m_generatorCount = 0;
this.m_countX = 0;
this.m_countY = 0;
this.m_diagram = [];
this.m_generatorBuffer = b2MakeArray(generatorCapacity, function (index) { return new b2VoronoiDiagram.Generator(); });
this.m_generatorCapacity = generatorCapacity;
}
/**
* Add a generator.
*
* @param center the position of the generator.
* @param tag a tag used to identify the generator in callback functions.
* @param necessary whether to callback for nodes associated with the generator.
*/
b2VoronoiDiagram.prototype.AddGenerator = function (center, tag, necessary) {
// DEBUG: b2Assert(this.m_generatorCount < this.m_generatorCapacity);
var g = this.m_generatorBuffer[this.m_generatorCount++];
g.center.Copy(center);
g.tag = tag;
g.necessary = necessary;
};
/**
* Generate the Voronoi diagram. It is rasterized with a given
* interval in the same range as the necessary generators exist.
*
* @param radius the interval of the diagram.
* @param margin margin for which the range of the diagram is extended.
*/
b2VoronoiDiagram.prototype.Generate = function (radius, margin) {
var inverseRadius = 1 / radius;
var lower = new b2Vec2(+b2_maxFloat, +b2_maxFloat);
var upper = new b2Vec2(-b2_maxFloat, -b2_maxFloat);
var necessary_count = 0;
for (var k = 0; k < this.m_generatorCount; k++) {
var g = this.m_generatorBuffer[k];
if (g.necessary) {
b2Vec2.MinV(lower, g.center, lower);
b2Vec2.MaxV(upper, g.center, upper);
++necessary_count;
}
}
if (necessary_count === 0) {
///debugger;
this.m_countX = 0;
this.m_countY = 0;
return;
}
lower.x -= margin;
lower.y -= margin;
upper.x += margin;
upper.y += margin;
this.m_countX = 1 + Math.floor(inverseRadius * (upper.x - lower.x));
this.m_countY = 1 + Math.floor(inverseRadius * (upper.y - lower.y));
/// m_diagram = (Generator**) m_allocator->Allocate(sizeof(Generator*) * m_countX * m_countY);
/// for (int32 i = 0; i < m_countX * m_countY; i++)
/// {
/// m_diagram[i] = NULL;
/// }
this.m_diagram = []; // b2MakeArray(this.m_countX * this.m_countY, (index) => null);
// (4 * m_countX * m_countY) is the queue capacity that is experimentally
// known to be necessary and sufficient for general particle distributions.
var queue = new b2StackQueue(4 * this.m_countX * this.m_countY);
for (var k = 0; k < this.m_generatorCount; k++) {
var g = this.m_generatorBuffer[k];
/// g.center = inverseRadius * (g.center - lower);
g.center.SelfSub(lower).SelfMul(inverseRadius);
var x = Math.floor(g.center.x);
var y = Math.floor(g.center.y);
if (x >= 0 && y >= 0 && x < this.m_countX && y < this.m_countY) {
queue.Push(new b2VoronoiDiagram.Task(x, y, x + y * this.m_countX, g));
}
}
while (!queue.Empty()) {
var task = queue.Front();
var x = task.m_x;
var y = task.m_y;
var i = task.m_i;
var g = task.m_generator;
queue.Pop();
if (!this.m_diagram[i]) {
this.m_diagram[i] = g;
if (x > 0) {
queue.Push(new b2VoronoiDiagram.Task(x - 1, y, i - 1, g));
}
if (y > 0) {
queue.Push(new b2VoronoiDiagram.Task(x, y - 1, i - this.m_countX, g));
}
if (x < this.m_countX - 1) {
queue.Push(new b2VoronoiDiagram.Task(x + 1, y, i + 1, g));
}
if (y < this.m_countY - 1) {
queue.Push(new b2VoronoiDiagram.Task(x, y + 1, i + this.m_countX, g));
}
}
}
for (var y = 0; y < this.m_countY; y++) {
for (var x = 0; x < this.m_countX - 1; x++) {
var i = x + y * this.m_countX;
var a = this.m_diagram[i];
var b = this.m_diagram[i + 1];
if (a !== b) {
queue.Push(new b2VoronoiDiagram.Task(x, y, i, b));
queue.Push(new b2VoronoiDiagram.Task(x + 1, y, i + 1, a));
}
}
}
for (var y = 0; y < this.m_countY - 1; y++) {
for (var x = 0; x < this.m_countX; x++) {
var i = x + y * this.m_countX;
var a = this.m_diagram[i];
var b = this.m_diagram[i + this.m_countX];
if (a !== b) {
queue.Push(new b2VoronoiDiagram.Task(x, y, i, b));
queue.Push(new b2VoronoiDiagram.Task(x, y + 1, i + this.m_countX, a));
}
}
}
while (!queue.Empty()) {
var task = queue.Front();
var x = task.m_x;
var y = task.m_y;
var i = task.m_i;
var k = task.m_generator;
queue.Pop();
var a = this.m_diagram[i];
var b = k;
if (a !== b) {
var ax = a.center.x - x;
var ay = a.center.y - y;
var bx = b.center.x - x;
var by = b.center.y - y;
var a2 = ax * ax + ay * ay;
var b2 = bx * bx + by * by;
if (a2 > b2) {
this.m_diagram[i] = b;
if (x > 0) {
queue.Push(new b2VoronoiDiagram.Task(x - 1, y, i - 1, b));
}
if (y > 0) {
queue.Push(new b2VoronoiDiagram.Task(x, y - 1, i - this.m_countX, b));
}
if (x < this.m_countX - 1) {
queue.Push(new b2VoronoiDiagram.Task(x + 1, y, i + 1, b));
}
if (y < this.m_countY - 1) {
queue.Push(new b2VoronoiDiagram.Task(x, y + 1, i + this.m_countX, b));
}
}
}
}
};
/**
* Enumerate all nodes that contain at least one necessary
* generator.
*/
b2VoronoiDiagram.prototype.GetNodes = function (callback) {
for (var y = 0; y < this.m_countY - 1; y++) {
for (var x = 0; x < this.m_countX - 1; x++) {
var i = x + y * this.m_countX;
var a = this.m_diagram[i];
var b = this.m_diagram[i + 1];
var c = this.m_diagram[i + this.m_countX];
var d = this.m_diagram[i + 1 + this.m_countX];
if (b !== c) {
if (a !== b && a !== c &&
(a.necessary || b.necessary || c.necessary)) {
callback(a.tag, b.tag, c.tag);
}
if (d !== b && d !== c &&
(a.necessary || b.necessary || c.necessary)) {
callback(b.tag, d.tag, c.tag);
}
}
}
}
};
return b2VoronoiDiagram;
}());
(function (b2VoronoiDiagram) {
var Generator = /** @class */ (function () {
function Generator() {
this.center = new b2Vec2();
this.tag = 0;
this.necessary = false;
}
return Generator;
}());
b2VoronoiDiagram.Generator = Generator;
var Task = /** @class */ (function () {
function Task(x, y, i, g) {
this.m_x = x;
this.m_y = y;
this.m_i = i;
this.m_generator = g;
}
return Task;
}());
b2VoronoiDiagram.Task = Task;
})(b2VoronoiDiagram || (b2VoronoiDiagram = {})); // namespace b2VoronoiDiagram
// #endif
/*
* Copyright (c) 2013 Google, Inc.
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
function std_iter_swap(array, a, b) {
var tmp = array[a];
array[a] = array[b];
array[b] = tmp;
}
function default_compare(a, b) { return a < b; }
function std_sort(array, first, len, cmp) {
if (first === void 0) { first = 0; }
if (len === void 0) { len = array.length - first; }
if (cmp === void 0) { cmp = default_compare; }
var left = first;
var stack = [];
var pos = 0;
for (;;) { /* outer loop */
for (; left + 1 < len; len++) { /* sort left to len-1 */
var pivot = array[left + Math.floor(Math.random() * (len - left))]; /* pick random pivot */
stack[pos++] = len; /* sort right part later */
for (var right = left - 1;;) { /* inner loop: partitioning */
while (cmp(array[++right], pivot)) { } /* look for greater element */
while (cmp(pivot, array[--len])) { } /* look for smaller element */
if (right >= len) {
break;
} /* partition point found? */
std_iter_swap(array, right, len); /* the only swap */
} /* partitioned, continue left part */
}
if (pos === 0) {
break;
} /* stack empty? */
left = len; /* left to right is sorted */
len = stack[--pos]; /* get next range to sort */
}
return array;
}
function std_stable_sort(array, first, len, cmp) {
if (first === void 0) { first = 0; }
if (len === void 0) { len = array.length - first; }
if (cmp === void 0) { cmp = default_compare; }
return std_sort(array, first, len, cmp);
}
function std_remove_if(array, predicate, length) {
if (length === void 0) { length = array.length; }
var l = 0;
for (var c = 0; c < length; ++c) {
// if we can be collapsed, keep l where it is.
if (predicate(array[c])) {
continue;
}
// this node can't be collapsed; push it back as far as we can.
if (c === l) {
++l;
continue; // quick exit if we're already in the right spot
}
// array[l++] = array[c];
std_iter_swap(array, l++, c);
}
return l;
}
function std_lower_bound(array, first, last, val, cmp) {
if (cmp === void 0) { cmp = default_compare; }
var count = last - first;
while (count > 0) {
var step = Math.floor(count / 2);
var it = first + step;
if (cmp(array[it], val)) {
first = ++it;
count -= step + 1;
}
else {
count = step;
}
}
return first;
}
function std_upper_bound(array, first, last, val, cmp) {
if (cmp === void 0) { cmp = default_compare; }
var count = last - first;
while (count > 0) {
var step = Math.floor(count / 2);
var it = first + step;
if (!cmp(val, array[it])) {
first = ++it;
count -= step + 1;
}
else {
count = step;
}
}
return first;
}
function std_rotate(array, first, n_first, last) {
var next = n_first;
while (first !== next) {
std_iter_swap(array, first++, next++);
if (next === last) {
next = n_first;
}
else if (first === n_first) {
n_first = next;
}
}
}
function std_unique(array, first, last, cmp) {
if (first === last) {
return last;
}
var result = first;
while (++first !== last) {
if (!cmp(array[result], array[first])) {
///array[++result] = array[first];
std_iter_swap(array, ++result, first);
}
}
return ++result;
}
var b2GrowableBuffer = /** @class */ (function () {
function b2GrowableBuffer(allocator) {
this.data = [];
this.count = 0;
this.capacity = 0;
this.allocator = allocator;
}
b2GrowableBuffer.prototype.Append = function () {
if (this.count >= this.capacity) {
this.Grow();
}
return this.count++;
};
b2GrowableBuffer.prototype.Reserve = function (newCapacity) {
if (this.capacity >= newCapacity) {
return;
}
// DEBUG: b2Assert(this.capacity === this.data.length);
for (var i = this.capacity; i < newCapacity; ++i) {
this.data[i] = this.allocator();
}
this.capacity = newCapacity;
};
b2GrowableBuffer.prototype.Grow = function () {
// Double the capacity.
var newCapacity = this.capacity ? 2 * this.capacity : b2_minParticleSystemBufferCapacity;
// DEBUG: b2Assert(newCapacity > this.capacity);
this.Reserve(newCapacity);
};
b2GrowableBuffer.prototype.Free = function () {
if (this.data.length === 0) {
return;
}
this.data = [];
this.capacity = 0;
this.count = 0;
};
b2GrowableBuffer.prototype.Shorten = function (newEnd) {
// DEBUG: b2Assert(false);
};
b2GrowableBuffer.prototype.Data = function () {
return this.data;
};
b2GrowableBuffer.prototype.GetCount = function () {
return this.count;
};
b2GrowableBuffer.prototype.SetCount = function (newCount) {
// DEBUG: b2Assert(0 <= newCount && newCount <= this.capacity);
this.count = newCount;
};
b2GrowableBuffer.prototype.GetCapacity = function () {
return this.capacity;
};
b2GrowableBuffer.prototype.RemoveIf = function (pred) {
// DEBUG: let count = 0;
// DEBUG: for (let i = 0; i < this.count; ++i) {
// DEBUG: if (!pred(this.data[i])) {
// DEBUG: count++;
// DEBUG: }
// DEBUG: }
this.count = std_remove_if(this.data, pred, this.count);
// DEBUG: b2Assert(count === this.count);
};
b2GrowableBuffer.prototype.Unique = function (pred) {
this.count = std_unique(this.data, 0, this.count, pred);
};
return b2GrowableBuffer;
}());
var b2FixtureParticleQueryCallback = /** @class */ (function (_super) {
__extends(b2FixtureParticleQueryCallback, _super);
function b2FixtureParticleQueryCallback(system) {
var _this = _super.call(this) || this;
_this.m_system = system;
return _this;
}
b2FixtureParticleQueryCallback.prototype.ShouldQueryParticleSystem = function (system) {
// Skip reporting particles.
return false;
};
b2FixtureParticleQueryCallback.prototype.ReportFixture = function (fixture) {
if (fixture.IsSensor()) {
return true;
}
var shape = fixture.GetShape();
var childCount = shape.GetChildCount();
for (var childIndex = 0; childIndex < childCount; childIndex++) {
var aabb = fixture.GetAABB(childIndex);
var enumerator = this.m_system.GetInsideBoundsEnumerator(aabb);
var index = void 0;
while ((index = enumerator.GetNext()) >= 0) {
this.ReportFixtureAndParticle(fixture, childIndex, index);
}
}
return true;
};
b2FixtureParticleQueryCallback.prototype.ReportParticle = function (system, index) {
return false;
};
b2FixtureParticleQueryCallback.prototype.ReportFixtureAndParticle = function (fixture, childIndex, index) {
// DEBUG: b2Assert(false); // pure virtual
};
return b2FixtureParticleQueryCallback;
}(b2QueryCallback));
var b2ParticleContact = /** @class */ (function () {
function b2ParticleContact() {
this.indexA = 0;
this.indexB = 0;
this.weight = 0;
this.normal = new b2Vec2();
this.flags = 0;
}
b2ParticleContact.prototype.SetIndices = function (a, b) {
// DEBUG: b2Assert(a <= b2_maxParticleIndex && b <= b2_maxParticleIndex);
this.indexA = a;
this.indexB = b;
};
b2ParticleContact.prototype.SetWeight = function (w) {
this.weight = w;
};
b2ParticleContact.prototype.SetNormal = function (n) {
this.normal.Copy(n);
};
b2ParticleContact.prototype.SetFlags = function (f) {
this.flags = f;
};
b2ParticleContact.prototype.GetIndexA = function () {
return this.indexA;
};
b2ParticleContact.prototype.GetIndexB = function () {
return this.indexB;
};
b2ParticleContact.prototype.GetWeight = function () {
return this.weight;
};
b2ParticleContact.prototype.GetNormal = function () {
return this.normal;
};
b2ParticleContact.prototype.GetFlags = function () {
return this.flags;
};
b2ParticleContact.prototype.IsEqual = function (rhs) {
return this.indexA === rhs.indexA && this.indexB === rhs.indexB && this.flags === rhs.flags && this.weight === rhs.weight && this.normal.x === rhs.normal.x && this.normal.y === rhs.normal.y;
};
b2ParticleContact.prototype.IsNotEqual = function (rhs) {
return !this.IsEqual(rhs);
};
b2ParticleContact.prototype.ApproximatelyEqual = function (rhs) {
var MAX_WEIGHT_DIFF = 0.01; // Weight 0 ~ 1, so about 1%
var MAX_NORMAL_DIFF_SQ = 0.01 * 0.01; // Normal length = 1, so 1%
return this.indexA === rhs.indexA && this.indexB === rhs.indexB && this.flags === rhs.flags && b2Abs(this.weight - rhs.weight) < MAX_WEIGHT_DIFF && b2Vec2.DistanceSquaredVV(this.normal, rhs.normal) < MAX_NORMAL_DIFF_SQ;
};
return b2ParticleContact;
}());
var b2ParticleBodyContact = /** @class */ (function () {
function b2ParticleBodyContact() {
this.index = 0; // Index of the particle making contact.
this.weight = 0.0; // Weight of the contact. A value between 0.0f and 1.0f.
this.normal = new b2Vec2(); // The normalized direction from the particle to the body.
this.mass = 0.0; // The effective mass used in calculating force.
}
return b2ParticleBodyContact;
}());
var b2ParticlePair = /** @class */ (function () {
function b2ParticlePair() {
this.indexA = 0; // Indices of the respective particles making pair.
this.indexB = 0;
this.flags = 0; // The logical sum of the particle flags. See the b2ParticleFlag enum.
this.strength = 0.0; // The strength of cohesion among the particles.
this.distance = 0.0; // The initial distance of the particles.
}
return b2ParticlePair;
}());
var b2ParticleTriad = /** @class */ (function () {
function b2ParticleTriad() {
this.indexA = 0; // Indices of the respective particles making triad.
this.indexB = 0;
this.indexC = 0;
this.flags = 0; // The logical sum of the particle flags. See the b2ParticleFlag enum.
this.strength = 0.0; // The strength of cohesion among the particles.
this.pa = new b2Vec2(0.0, 0.0); // Values used for calculation.
this.pb = new b2Vec2(0.0, 0.0);
this.pc = new b2Vec2(0.0, 0.0);
this.ka = 0.0;
this.kb = 0.0;
this.kc = 0.0;
this.s = 0.0;
}
return b2ParticleTriad;
}());
var b2ParticleSystemDef = /** @class */ (function () {
function b2ParticleSystemDef() {
// Initialize physical coefficients to the maximum values that
// maintain numerical stability.
/**
* Enable strict Particle/Body contact check.
* See SetStrictContactCheck for details.
*/
this.strictContactCheck = false;
/**
* Set the particle density.
* See SetDensity for details.
*/
this.density = 1.0;
/**
* Change the particle gravity scale. Adjusts the effect of the
* global gravity vector on particles. Default value is 1.0f.
*/
this.gravityScale = 1.0;
/**
* Particles behave as circles with this radius. In Box2D units.
*/
this.radius = 1.0;
/**
* Set the maximum number of particles.
* By default, there is no maximum. The particle buffers can
* continue to grow while b2World's block allocator still has
* memory.
* See SetMaxParticleCount for details.
*/
this.maxCount = 0;
/**
* Increases pressure in response to compression
* Smaller values allow more compression
*/
this.pressureStrength = 0.005;
/**
* Reduces velocity along the collision normal
* Smaller value reduces less
*/
this.dampingStrength = 1.0;
/**
* Restores shape of elastic particle groups
* Larger values increase elastic particle velocity
*/
this.elasticStrength = 0.25;
/**
* Restores length of spring particle groups
* Larger values increase spring particle velocity
*/
this.springStrength = 0.25;
/**
* Reduces relative velocity of viscous particles
* Larger values slow down viscous particles more
*/
this.viscousStrength = 0.25;
/**
* Produces pressure on tensile particles
* 0~0.2. Larger values increase the amount of surface tension.
*/
this.surfaceTensionPressureStrength = 0.2;
/**
* Smoothes outline of tensile particles
* 0~0.2. Larger values result in rounder, smoother,
* water-drop-like clusters of particles.
*/
this.surfaceTensionNormalStrength = 0.2;
/**
* Produces additional pressure on repulsive particles
* Larger values repulse more
* Negative values mean attraction. The range where particles
* behave stably is about -0.2 to 2.0.
*/
this.repulsiveStrength = 1.0;
/**
* Produces repulsion between powder particles
* Larger values repulse more
*/
this.powderStrength = 0.5;
/**
* Pushes particles out of solid particle group
* Larger values repulse more
*/
this.ejectionStrength = 0.5;
/**
* Produces static pressure
* Larger values increase the pressure on neighboring partilces
* For a description of static pressure, see
* http://en.wikipedia.org/wiki/Static_pressure#Static_pressure_in_fluid_dynamics
*/
this.staticPressureStrength = 0.2;
/**
* Reduces instability in static pressure calculation
* Larger values make stabilize static pressure with fewer
* iterations
*/
this.staticPressureRelaxation = 0.2;
/**
* Computes static pressure more precisely
* See SetStaticPressureIterations for details
*/
this.staticPressureIterations = 8;
/**
* Determines how fast colors are mixed
* 1.0f ==> mixed immediately
* 0.5f ==> mixed half way each simulation step (see
* b2World::Step())
*/
this.colorMixingStrength = 0.5;
/**
* Whether to destroy particles by age when no more particles
* can be created. See #b2ParticleSystem::SetDestructionByAge()
* for more information.
*/
this.destroyByAge = true;
/**
* Granularity of particle lifetimes in seconds. By default
* this is set to (1.0f / 60.0f) seconds. b2ParticleSystem uses
* a 32-bit signed value to track particle lifetimes so the
* maximum lifetime of a particle is (2^32 - 1) / (1.0f /
* lifetimeGranularity) seconds. With the value set to 1/60 the
* maximum lifetime or age of a particle is 2.27 years.
*/
this.lifetimeGranularity = 1.0 / 60.0;
}
b2ParticleSystemDef.prototype.Copy = function (def) {
this.strictContactCheck = def.strictContactCheck;
this.density = def.density;
this.gravityScale = def.gravityScale;
this.radius = def.radius;
this.maxCount = def.maxCount;
this.pressureStrength = def.pressureStrength;
this.dampingStrength = def.dampingStrength;
this.elasticStrength = def.elasticStrength;
this.springStrength = def.springStrength;
this.viscousStrength = def.viscousStrength;
this.surfaceTensionPressureStrength = def.surfaceTensionPressureStrength;
this.surfaceTensionNormalStrength = def.surfaceTensionNormalStrength;
this.repulsiveStrength = def.repulsiveStrength;
this.powderStrength = def.powderStrength;
this.ejectionStrength = def.ejectionStrength;
this.staticPressureStrength = def.staticPressureStrength;
this.staticPressureRelaxation = def.staticPressureRelaxation;
this.staticPressureIterations = def.staticPressureIterations;
this.colorMixingStrength = def.colorMixingStrength;
this.destroyByAge = def.destroyByAge;
this.lifetimeGranularity = def.lifetimeGranularity;
return this;
};
b2ParticleSystemDef.prototype.Clone = function () {
return new b2ParticleSystemDef().Copy(this);
};
return b2ParticleSystemDef;
}());
exports.b2ParticleSystem = /** @class */ (function () {
function b2ParticleSystem(def, world) {
this.m_paused = false;
this.m_timestamp = 0;
this.m_allParticleFlags = 0;
this.m_needsUpdateAllParticleFlags = false;
this.m_allGroupFlags = 0;
this.m_needsUpdateAllGroupFlags = false;
this.m_hasForce = false;
this.m_iterationIndex = 0;
this.m_inverseDensity = 0.0;
this.m_particleDiameter = 0.0;
this.m_inverseDiameter = 0.0;
this.m_squaredDiameter = 0.0;
this.m_count = 0;
this.m_internalAllocatedCapacity = 0;
/**
* Allocator for b2ParticleHandle instances.
*/
///m_handleAllocator: any = null;
/**
* Maps particle indicies to handles.
*/
this.m_handleIndexBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_flagsBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_positionBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_velocityBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_forceBuffer = [];
/**
* this.m_weightBuffer is populated in ComputeWeight and used in
* ComputeDepth(), SolveStaticPressure() and SolvePressure().
*/
this.m_weightBuffer = [];
/**
* When any particles have the flag b2_staticPressureParticle,
* this.m_staticPressureBuffer is first allocated and used in
* SolveStaticPressure() and SolvePressure(). It will be
* reallocated on subsequent CreateParticle() calls.
*/
this.m_staticPressureBuffer = [];
/**
* this.m_accumulationBuffer is used in many functions as a temporary
* buffer for scalar values.
*/
this.m_accumulationBuffer = [];
/**
* When any particles have the flag b2_tensileParticle,
* this.m_accumulation2Buffer is first allocated and used in
* SolveTensile() as a temporary buffer for vector values. It
* will be reallocated on subsequent CreateParticle() calls.
*/
this.m_accumulation2Buffer = [];
/**
* When any particle groups have the flag b2_solidParticleGroup,
* this.m_depthBuffer is first allocated and populated in
* ComputeDepth() and used in SolveSolid(). It will be
* reallocated on subsequent CreateParticle() calls.
*/
this.m_depthBuffer = [];
this.m_colorBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_groupBuffer = [];
this.m_userDataBuffer = new b2ParticleSystem.UserOverridableBuffer();
/**
* Stuck particle detection parameters and record keeping
*/
this.m_stuckThreshold = 0;
this.m_lastBodyContactStepBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_bodyContactCountBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_consecutiveContactStepsBuffer = new b2ParticleSystem.UserOverridableBuffer();
this.m_stuckParticleBuffer = new b2GrowableBuffer(function () { return 0; });
this.m_proxyBuffer = new b2GrowableBuffer(function () { return new b2ParticleSystem.Proxy(); });
this.m_contactBuffer = new b2GrowableBuffer(function () { return new b2ParticleContact(); });
this.m_bodyContactBuffer = new b2GrowableBuffer(function () { return new b2ParticleBodyContact(); });
this.m_pairBuffer = new b2GrowableBuffer(function () { return new b2ParticlePair(); });
this.m_triadBuffer = new b2GrowableBuffer(function () { return new b2ParticleTriad(); });
/**
* Time each particle should be destroyed relative to the last
* time this.m_timeElapsed was initialized. Each unit of time
* corresponds to b2ParticleSystemDef::lifetimeGranularity
* seconds.
*/
this.m_expirationTimeBuffer = new b2ParticleSystem.UserOverridableBuffer();
/**
* List of particle indices sorted by expiration time.
*/
this.m_indexByExpirationTimeBuffer = new b2ParticleSystem.UserOverridableBuffer();
/**
* Time elapsed in 32:32 fixed point. Each non-fractional unit
* of time corresponds to
* b2ParticleSystemDef::lifetimeGranularity seconds.
*/
this.m_timeElapsed = 0;
/**
* Whether the expiration time buffer has been modified and
* needs to be resorted.
*/
this.m_expirationTimeBufferRequiresSorting = false;
this.m_groupCount = 0;
this.m_groupList = null;
this.m_def = new b2ParticleSystemDef();
this.m_prev = null;
this.m_next = null;
this.SetStrictContactCheck(def.strictContactCheck);
this.SetDensity(def.density);
this.SetGravityScale(def.gravityScale);
this.SetRadius(def.radius);
this.SetMaxParticleCount(def.maxCount);
// DEBUG: b2Assert(def.lifetimeGranularity > 0.0);
this.m_def = def.Clone();
this.m_world = world;
this.SetDestructionByAge(this.m_def.destroyByAge);
}
b2ParticleSystem.computeTag = function (x, y) {
///return ((uint32)(y + yOffset) << yShift) + (uint32)(xScale * x + xOffset);
return ((((y + b2ParticleSystem.yOffset) >>> 0) << b2ParticleSystem.yShift) + ((b2ParticleSystem.xScale * x + b2ParticleSystem.xOffset) >>> 0)) >>> 0;
};
b2ParticleSystem.computeRelativeTag = function (tag, x, y) {
///return tag + (y << yShift) + (x << xShift);
return (tag + (y << b2ParticleSystem.yShift) + (x << b2ParticleSystem.xShift)) >>> 0;
};
b2ParticleSystem.prototype.Drop = function () {
while (this.m_groupList) {
this.DestroyParticleGroup(this.m_groupList);
}
this.FreeUserOverridableBuffer(this.m_handleIndexBuffer);
this.FreeUserOverridableBuffer(this.m_flagsBuffer);
this.FreeUserOverridableBuffer(this.m_lastBodyContactStepBuffer);
this.FreeUserOverridableBuffer(this.m_bodyContactCountBuffer);
this.FreeUserOverridableBuffer(this.m_consecutiveContactStepsBuffer);
this.FreeUserOverridableBuffer(this.m_positionBuffer);
this.FreeUserOverridableBuffer(this.m_velocityBuffer);
this.FreeUserOverridableBuffer(this.m_colorBuffer);
this.FreeUserOverridableBuffer(this.m_userDataBuffer);
this.FreeUserOverridableBuffer(this.m_expirationTimeBuffer);
this.FreeUserOverridableBuffer(this.m_indexByExpirationTimeBuffer);
this.FreeBuffer(this.m_forceBuffer, this.m_internalAllocatedCapacity);
this.FreeBuffer(this.m_weightBuffer, this.m_internalAllocatedCapacity);
this.FreeBuffer(this.m_staticPressureBuffer, this.m_internalAllocatedCapacity);
this.FreeBuffer(this.m_accumulationBuffer, this.m_internalAllocatedCapacity);
this.FreeBuffer(this.m_accumulation2Buffer, this.m_internalAllocatedCapacity);
this.FreeBuffer(this.m_depthBuffer, this.m_internalAllocatedCapacity);
this.FreeBuffer(this.m_groupBuffer, this.m_internalAllocatedCapacity);
};
/**
* Create a particle whose properties have been defined.
*
* No reference to the definition is retained.
*
* A simulation step must occur before it's possible to interact
* with a newly created particle. For example,
* DestroyParticleInShape() will not destroy a particle until
* b2World::Step() has been called.
*
* warning: This function is locked during callbacks.
*/
b2ParticleSystem.prototype.CreateParticle = function (def) {
if (this.m_world.IsLocked()) {
throw new Error();
}
if (this.m_count >= this.m_internalAllocatedCapacity) {
// Double the particle capacity.
var capacity = this.m_count ? 2 * this.m_count : b2_minParticleSystemBufferCapacity;
this.ReallocateInternalAllocatedBuffers(capacity);
}
if (this.m_count >= this.m_internalAllocatedCapacity) {
// If the oldest particle should be destroyed...
if (this.m_def.destroyByAge) {
this.DestroyOldestParticle(0, false);
// Need to destroy this particle *now* so that it's possible to
// create a new particle.
this.SolveZombie();
}
else {
return b2_invalidParticleIndex;
}
}
var index = this.m_count++;
if (!this.m_flagsBuffer.data) {
throw new Error();
}
this.m_flagsBuffer.data[index] = 0;
if (this.m_lastBodyContactStepBuffer.data) {
this.m_lastBodyContactStepBuffer.data[index] = 0;
}
if (this.m_bodyContactCountBuffer.data) {
this.m_bodyContactCountBuffer.data[index] = 0;
}
if (this.m_consecutiveContactStepsBuffer.data) {
this.m_consecutiveContactStepsBuffer.data[index] = 0;
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
this.m_positionBuffer.data[index] = (this.m_positionBuffer.data[index] || new b2Vec2()).Copy(b2Maybe(def.position, b2Vec2.ZERO));
this.m_velocityBuffer.data[index] = (this.m_velocityBuffer.data[index] || new b2Vec2()).Copy(b2Maybe(def.velocity, b2Vec2.ZERO));
this.m_weightBuffer[index] = 0;
this.m_forceBuffer[index] = (this.m_forceBuffer[index] || new b2Vec2()).SetZero();
if (this.m_staticPressureBuffer) {
this.m_staticPressureBuffer[index] = 0;
}
if (this.m_depthBuffer) {
this.m_depthBuffer[index] = 0;
}
var color = new b2Color().Copy(b2Maybe(def.color, b2Color.ZERO));
if (this.m_colorBuffer.data || !color.IsZero()) {
this.m_colorBuffer.data = this.RequestBuffer(this.m_colorBuffer.data);
this.m_colorBuffer.data[index] = (this.m_colorBuffer.data[index] || new b2Color()).Copy(color);
}
if (this.m_userDataBuffer.data || def.userData) {
this.m_userDataBuffer.data = this.RequestBuffer(this.m_userDataBuffer.data);
this.m_userDataBuffer.data[index] = def.userData;
}
if (this.m_handleIndexBuffer.data) {
this.m_handleIndexBuffer.data[index] = null;
}
///Proxy& proxy = m_proxyBuffer.Append();
var proxy = this.m_proxyBuffer.data[this.m_proxyBuffer.Append()];
// If particle lifetimes are enabled or the lifetime is set in the particle
// definition, initialize the lifetime.
var lifetime = b2Maybe(def.lifetime, 0.0);
var finiteLifetime = lifetime > 0.0;
if (this.m_expirationTimeBuffer.data || finiteLifetime) {
this.SetParticleLifetime(index, finiteLifetime ? lifetime :
this.ExpirationTimeToLifetime(-this.GetQuantizedTimeElapsed()));
// Add a reference to the newly added particle to the end of the
// queue.
if (!this.m_indexByExpirationTimeBuffer.data) {
throw new Error();
}
this.m_indexByExpirationTimeBuffer.data[index] = index;
}
proxy.index = index;
var group = b2Maybe(def.group, null);
this.m_groupBuffer[index] = group;
if (group) {
if (group.m_firstIndex < group.m_lastIndex) {
// Move particles in the group just before the new particle.
this.RotateBuffer(group.m_firstIndex, group.m_lastIndex, index);
// DEBUG: b2Assert(group.m_lastIndex === index);
// Update the index range of the group to contain the new particle.
group.m_lastIndex = index + 1;
}
else {
// If the group is empty, reset the index range to contain only the
// new particle.
group.m_firstIndex = index;
group.m_lastIndex = index + 1;
}
}
this.SetParticleFlags(index, b2Maybe(def.flags, 0));
return index;
};
/**
* Retrieve a handle to the particle at the specified index.
*
* Please see #b2ParticleHandle for why you might want a handle.
*/
b2ParticleSystem.prototype.GetParticleHandleFromIndex = function (index) {
// DEBUG: b2Assert(index >= 0 && index < this.GetParticleCount() && index !== b2_invalidParticleIndex);
this.m_handleIndexBuffer.data = this.RequestBuffer(this.m_handleIndexBuffer.data);
var handle = this.m_handleIndexBuffer.data[index];
if (handle) {
return handle;
}
// Create a handle.
///handle = m_handleAllocator.Allocate();
handle = new b2ParticleHandle();
// DEBUG: b2Assert(handle !== null);
handle.SetIndex(index);
this.m_handleIndexBuffer.data[index] = handle;
return handle;
};
/**
* Destroy a particle.
*
* The particle is removed after the next simulation step (see
* b2World::Step()).
*
* @param index Index of the particle to destroy.
* @param callDestructionListener Whether to call the
* destruction listener just before the particle is
* destroyed.
*/
b2ParticleSystem.prototype.DestroyParticle = function (index, callDestructionListener) {
if (callDestructionListener === void 0) { callDestructionListener = false; }
if (!this.m_flagsBuffer.data) {
throw new Error();
}
var flags = exports.b2ParticleFlag.b2_zombieParticle;
if (callDestructionListener) {
flags |= exports.b2ParticleFlag.b2_destructionListenerParticle;
}
this.SetParticleFlags(index, this.m_flagsBuffer.data[index] | flags);
};
/**
* Destroy the Nth oldest particle in the system.
*
* The particle is removed after the next b2World::Step().
*
* @param index Index of the Nth oldest particle to
* destroy, 0 will destroy the oldest particle in the
* system, 1 will destroy the next oldest particle etc.
* @param callDestructionListener Whether to call the
* destruction listener just before the particle is
* destroyed.
*/
b2ParticleSystem.prototype.DestroyOldestParticle = function (index, callDestructionListener) {
if (callDestructionListener === void 0) { callDestructionListener = false; }
var particleCount = this.GetParticleCount();
// DEBUG: b2Assert(index >= 0 && index < particleCount);
// Make sure particle lifetime tracking is enabled.
// DEBUG: b2Assert(this.m_indexByExpirationTimeBuffer.data !== null);
if (!this.m_indexByExpirationTimeBuffer.data) {
throw new Error();
}
if (!this.m_expirationTimeBuffer.data) {
throw new Error();
}
// Destroy the oldest particle (preferring to destroy finite
// lifetime particles first) to free a slot in the buffer.
var oldestFiniteLifetimeParticle = this.m_indexByExpirationTimeBuffer.data[particleCount - (index + 1)];
var oldestInfiniteLifetimeParticle = this.m_indexByExpirationTimeBuffer.data[index];
this.DestroyParticle(this.m_expirationTimeBuffer.data[oldestFiniteLifetimeParticle] > 0.0 ?
oldestFiniteLifetimeParticle : oldestInfiniteLifetimeParticle, callDestructionListener);
};
/**
* Destroy particles inside a shape.
*
* warning: This function is locked during callbacks.
*
* In addition, this function immediately destroys particles in
* the shape in constrast to DestroyParticle() which defers the
* destruction until the next simulation step.
*
* @return Number of particles destroyed.
* @param shape Shape which encloses particles
* that should be destroyed.
* @param xf Transform applied to the shape.
* @param callDestructionListener Whether to call the
* world b2DestructionListener for each particle
* destroyed.
*/
b2ParticleSystem.prototype.DestroyParticlesInShape = function (shape, xf, callDestructionListener) {
if (callDestructionListener === void 0) { callDestructionListener = false; }
var s_aabb = b2ParticleSystem.DestroyParticlesInShape_s_aabb;
if (this.m_world.IsLocked()) {
throw new Error();
}
var callback = new b2ParticleSystem.DestroyParticlesInShapeCallback(this, shape, xf, callDestructionListener);
var aabb = s_aabb;
shape.ComputeAABB(aabb, xf, 0);
this.m_world.QueryAABB(callback, aabb);
return callback.Destroyed();
};
/**
* Create a particle group whose properties have been defined.
*
* No reference to the definition is retained.
*
* warning: This function is locked during callbacks.
*/
b2ParticleSystem.prototype.CreateParticleGroup = function (groupDef) {
var s_transform = b2ParticleSystem.CreateParticleGroup_s_transform;
if (this.m_world.IsLocked()) {
throw new Error();
}
var transform = s_transform;
transform.SetPositionAngle(b2Maybe(groupDef.position, b2Vec2.ZERO), b2Maybe(groupDef.angle, 0));
var firstIndex = this.m_count;
if (groupDef.shape) {
this.CreateParticlesWithShapeForGroup(groupDef.shape, groupDef, transform);
}
if (groupDef.shapes) {
this.CreateParticlesWithShapesForGroup(groupDef.shapes, b2Maybe(groupDef.shapeCount, groupDef.shapes.length), groupDef, transform);
}
if (groupDef.positionData) {
var count = b2Maybe(groupDef.particleCount, groupDef.positionData.length);
for (var i = 0; i < count; i++) {
var p = groupDef.positionData[i];
this.CreateParticleForGroup(groupDef, transform, p);
}
}
var lastIndex = this.m_count;
var group = new b2ParticleGroup(this);
group.m_firstIndex = firstIndex;
group.m_lastIndex = lastIndex;
group.m_strength = b2Maybe(groupDef.strength, 1);
group.m_userData = groupDef.userData;
group.m_transform.Copy(transform);
group.m_prev = null;
group.m_next = this.m_groupList;
if (this.m_groupList) {
this.m_groupList.m_prev = group;
}
this.m_groupList = group;
++this.m_groupCount;
for (var i = firstIndex; i < lastIndex; i++) {
this.m_groupBuffer[i] = group;
}
this.SetGroupFlags(group, b2Maybe(groupDef.groupFlags, 0));
// Create pairs and triads between particles in the group.
var filter = new b2ParticleSystem.ConnectionFilter();
this.UpdateContacts(true);
this.UpdatePairsAndTriads(firstIndex, lastIndex, filter);
if (groupDef.group) {
this.JoinParticleGroups(groupDef.group, group);
group = groupDef.group;
}
return group;
};
/**
* Join two particle groups.
*
* warning: This function is locked during callbacks.
*
* @param groupA the first group. Expands to encompass the second group.
* @param groupB the second group. It is destroyed.
*/
b2ParticleSystem.prototype.JoinParticleGroups = function (groupA, groupB) {
if (this.m_world.IsLocked()) {
throw new Error();
}
// DEBUG: b2Assert(groupA !== groupB);
this.RotateBuffer(groupB.m_firstIndex, groupB.m_lastIndex, this.m_count);
// DEBUG: b2Assert(groupB.m_lastIndex === this.m_count);
this.RotateBuffer(groupA.m_firstIndex, groupA.m_lastIndex, groupB.m_firstIndex);
// DEBUG: b2Assert(groupA.m_lastIndex === groupB.m_firstIndex);
// Create pairs and triads connecting groupA and groupB.
var filter = new b2ParticleSystem.JoinParticleGroupsFilter(groupB.m_firstIndex);
this.UpdateContacts(true);
this.UpdatePairsAndTriads(groupA.m_firstIndex, groupB.m_lastIndex, filter);
for (var i = groupB.m_firstIndex; i < groupB.m_lastIndex; i++) {
this.m_groupBuffer[i] = groupA;
}
var groupFlags = groupA.m_groupFlags | groupB.m_groupFlags;
this.SetGroupFlags(groupA, groupFlags);
groupA.m_lastIndex = groupB.m_lastIndex;
groupB.m_firstIndex = groupB.m_lastIndex;
this.DestroyParticleGroup(groupB);
};
/**
* Split particle group into multiple disconnected groups.
*
* warning: This function is locked during callbacks.
*
* @param group the group to be split.
*/
b2ParticleSystem.prototype.SplitParticleGroup = function (group) {
this.UpdateContacts(true);
var particleCount = group.GetParticleCount();
// We create several linked lists. Each list represents a set of connected particles.
///ParticleListNode* nodeBuffer = (ParticleListNode*) m_world.m_stackAllocator.Allocate(sizeof(ParticleListNode) * particleCount);
var nodeBuffer = b2MakeArray(particleCount, function (index) { return new b2ParticleSystem.ParticleListNode(); });
b2ParticleSystem.InitializeParticleLists(group, nodeBuffer);
this.MergeParticleListsInContact(group, nodeBuffer);
var survivingList = b2ParticleSystem.FindLongestParticleList(group, nodeBuffer);
this.MergeZombieParticleListNodes(group, nodeBuffer, survivingList);
this.CreateParticleGroupsFromParticleList(group, nodeBuffer, survivingList);
this.UpdatePairsAndTriadsWithParticleList(group, nodeBuffer);
///this.m_world.m_stackAllocator.Free(nodeBuffer);
};
/**
* Get the world particle group list. With the returned group,
* use b2ParticleGroup::GetNext to get the next group in the
* world list.
*
* A null group indicates the end of the list.
*
* @return the head of the world particle group list.
*/
b2ParticleSystem.prototype.GetParticleGroupList = function () {
return this.m_groupList;
};
/**
* Get the number of particle groups.
*/
b2ParticleSystem.prototype.GetParticleGroupCount = function () {
return this.m_groupCount;
};
/**
* Get the number of particles.
*/
b2ParticleSystem.prototype.GetParticleCount = function () {
return this.m_count;
};
/**
* Get the maximum number of particles.
*/
b2ParticleSystem.prototype.GetMaxParticleCount = function () {
return this.m_def.maxCount;
};
/**
* Set the maximum number of particles.
*
* A value of 0 means there is no maximum. The particle buffers
* can continue to grow while b2World's block allocator still
* has memory.
*
* Note: If you try to CreateParticle() with more than this
* count, b2_invalidParticleIndex is returned unless
* SetDestructionByAge() is used to enable the destruction of
* the oldest particles in the system.
*/
b2ParticleSystem.prototype.SetMaxParticleCount = function (count) {
// DEBUG: b2Assert(this.m_count <= count);
this.m_def.maxCount = count;
};
/**
* Get all existing particle flags.
*/
b2ParticleSystem.prototype.GetAllParticleFlags = function () {
return this.m_allParticleFlags;
};
/**
* Get all existing particle group flags.
*/
b2ParticleSystem.prototype.GetAllGroupFlags = function () {
return this.m_allGroupFlags;
};
/**
* Pause or unpause the particle system. When paused,
* b2World::Step() skips over this particle system. All
* b2ParticleSystem function calls still work.
*
* @param paused paused is true to pause, false to un-pause.
*/
b2ParticleSystem.prototype.SetPaused = function (paused) {
this.m_paused = paused;
};
/**
* Initially, true, then, the last value passed into
* SetPaused().
*
* @return true if the particle system is being updated in b2World::Step().
*/
b2ParticleSystem.prototype.GetPaused = function () {
return this.m_paused;
};
/**
* Change the particle density.
*
* Particle density affects the mass of the particles, which in
* turn affects how the particles interact with b2Bodies. Note
* that the density does not affect how the particles interact
* with each other.
*/
b2ParticleSystem.prototype.SetDensity = function (density) {
this.m_def.density = density;
this.m_inverseDensity = 1 / this.m_def.density;
};
/**
* Get the particle density.
*/
b2ParticleSystem.prototype.GetDensity = function () {
return this.m_def.density;
};
/**
* Change the particle gravity scale. Adjusts the effect of the
* global gravity vector on particles.
*/
b2ParticleSystem.prototype.SetGravityScale = function (gravityScale) {
this.m_def.gravityScale = gravityScale;
};
/**
* Get the particle gravity scale.
*/
b2ParticleSystem.prototype.GetGravityScale = function () {
return this.m_def.gravityScale;
};
/**
* Damping is used to reduce the velocity of particles. The
* damping parameter can be larger than 1.0f but the damping
* effect becomes sensitive to the time step when the damping
* parameter is large.
*/
b2ParticleSystem.prototype.SetDamping = function (damping) {
this.m_def.dampingStrength = damping;
};
/**
* Get damping for particles
*/
b2ParticleSystem.prototype.GetDamping = function () {
return this.m_def.dampingStrength;
};
/**
* Change the number of iterations when calculating the static
* pressure of particles. By default, 8 iterations. You can
* reduce the number of iterations down to 1 in some situations,
* but this may cause instabilities when many particles come
* together. If you see particles popping away from each other
* like popcorn, you may have to increase the number of
* iterations.
*
* For a description of static pressure, see
* http://en.wikipedia.org/wiki/Static_pressure#Static_pressure_in_fluid_dynamics
*/
b2ParticleSystem.prototype.SetStaticPressureIterations = function (iterations) {
this.m_def.staticPressureIterations = iterations;
};
/**
* Get the number of iterations for static pressure of
* particles.
*/
b2ParticleSystem.prototype.GetStaticPressureIterations = function () {
return this.m_def.staticPressureIterations;
};
/**
* Change the particle radius.
*
* You should set this only once, on world start.
* If you change the radius during execution, existing particles
* may explode, shrink, or behave unexpectedly.
*/
b2ParticleSystem.prototype.SetRadius = function (radius) {
this.m_particleDiameter = 2 * radius;
this.m_squaredDiameter = this.m_particleDiameter * this.m_particleDiameter;
this.m_inverseDiameter = 1 / this.m_particleDiameter;
};
/**
* Get the particle radius.
*/
b2ParticleSystem.prototype.GetRadius = function () {
return this.m_particleDiameter / 2;
};
/**
* Get the position of each particle
*
* Array is length GetParticleCount()
*
* @return the pointer to the head of the particle positions array.
*/
b2ParticleSystem.prototype.GetPositionBuffer = function () {
if (!this.m_positionBuffer.data) {
throw new Error();
}
return this.m_positionBuffer.data;
};
/**
* Get the velocity of each particle
*
* Array is length GetParticleCount()
*
* @return the pointer to the head of the particle velocities array.
*/
b2ParticleSystem.prototype.GetVelocityBuffer = function () {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
return this.m_velocityBuffer.data;
};
/**
* Get the color of each particle
*
* Array is length GetParticleCount()
*
* @return the pointer to the head of the particle colors array.
*/
b2ParticleSystem.prototype.GetColorBuffer = function () {
this.m_colorBuffer.data = this.RequestBuffer(this.m_colorBuffer.data);
return this.m_colorBuffer.data;
};
/**
* Get the particle-group of each particle.
*
* Array is length GetParticleCount()
*
* @return the pointer to the head of the particle group array.
*/
b2ParticleSystem.prototype.GetGroupBuffer = function () {
return this.m_groupBuffer;
};
/**
* Get the weight of each particle
*
* Array is length GetParticleCount()
*
* @return the pointer to the head of the particle positions array.
*/
b2ParticleSystem.prototype.GetWeightBuffer = function () {
return this.m_weightBuffer;
};
/**
* Get the user-specified data of each particle.
*
* Array is length GetParticleCount()
*
* @return the pointer to the head of the particle user-data array.
*/
b2ParticleSystem.prototype.GetUserDataBuffer = function () {
this.m_userDataBuffer.data = this.RequestBuffer(this.m_userDataBuffer.data);
return this.m_userDataBuffer.data;
};
/**
* Get the flags for each particle. See the b2ParticleFlag enum.
*
* Array is length GetParticleCount()
*
* @return the pointer to the head of the particle-flags array.
*/
b2ParticleSystem.prototype.GetFlagsBuffer = function () {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
return this.m_flagsBuffer.data;
};
/**
* Set flags for a particle. See the b2ParticleFlag enum.
*/
b2ParticleSystem.prototype.SetParticleFlags = function (index, newFlags) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
var oldFlags = this.m_flagsBuffer.data[index];
if (oldFlags & ~newFlags) {
// If any flags might be removed
this.m_needsUpdateAllParticleFlags = true;
}
if (~this.m_allParticleFlags & newFlags) {
// If any flags were added
if (newFlags & exports.b2ParticleFlag.b2_tensileParticle) {
this.m_accumulation2Buffer = this.RequestBuffer(this.m_accumulation2Buffer);
}
if (newFlags & exports.b2ParticleFlag.b2_colorMixingParticle) {
this.m_colorBuffer.data = this.RequestBuffer(this.m_colorBuffer.data);
}
this.m_allParticleFlags |= newFlags;
}
this.m_flagsBuffer.data[index] = newFlags;
};
/**
* Get flags for a particle. See the b2ParticleFlag enum.
*/
b2ParticleSystem.prototype.GetParticleFlags = function (index) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
return this.m_flagsBuffer.data[index];
};
/**
* Set an external buffer for particle data.
*
* Normally, the b2World's block allocator is used for particle
* data. However, sometimes you may have an OpenGL or Java
* buffer for particle data. To avoid data duplication, you may
* supply this external buffer.
*
* Note that, when b2World's block allocator is used, the
* particle data buffers can grow as required. However, when
* external buffers are used, the maximum number of particles is
* clamped to the size of the smallest external buffer.
*
* @param buffer a pointer to a block of memory.
* @param capacity the number of values in the block.
*/
b2ParticleSystem.prototype.SetFlagsBuffer = function (buffer, capacity) {
this.SetUserOverridableBuffer(this.m_flagsBuffer, buffer, capacity);
};
b2ParticleSystem.prototype.SetPositionBuffer = function (buffer, capacity) {
///if (buffer instanceof Float32Array) {
///let array = [];
///for (let i = 0; i < capacity; ++i) {
/// array[i] = new b2Vec2(buffer.subarray(i * 2, i * 2 + 2));
///}
///this.SetUserOverridableBuffer(this.m_positionBuffer, array, capacity);
///} else {
this.SetUserOverridableBuffer(this.m_positionBuffer, buffer, capacity);
///}
};
b2ParticleSystem.prototype.SetVelocityBuffer = function (buffer, capacity) {
///if (buffer instanceof Float32Array) {
///let array = [];
///for (let i = 0; i < capacity; ++i) {
/// array[i] = new b2Vec2(buffer.subarray(i * 2, i * 2 + 2));
///}
///this.SetUserOverridableBuffer(this.m_velocityBuffer, array, capacity);
///} else {
this.SetUserOverridableBuffer(this.m_velocityBuffer, buffer, capacity);
///}
};
b2ParticleSystem.prototype.SetColorBuffer = function (buffer, capacity) {
///if (buffer instanceof Uint8Array) {
///let array: b2Color[] = [];
///for (let i = 0; i < capacity; ++i) {
/// array[i] = new b2Color(buffer.subarray(i * 4, i * 4 + 4));
///}
///this.SetUserOverridableBuffer(this.m_colorBuffer, array, capacity);
///} else {
this.SetUserOverridableBuffer(this.m_colorBuffer, buffer, capacity);
///}
};
b2ParticleSystem.prototype.SetUserDataBuffer = function (buffer, capacity) {
this.SetUserOverridableBuffer(this.m_userDataBuffer, buffer, capacity);
};
/**
* Get contacts between particles
* Contact data can be used for many reasons, for example to
* trigger rendering or audio effects.
*/
b2ParticleSystem.prototype.GetContacts = function () {
return this.m_contactBuffer.data;
};
b2ParticleSystem.prototype.GetContactCount = function () {
return this.m_contactBuffer.count;
};
/**
* Get contacts between particles and bodies
*
* Contact data can be used for many reasons, for example to
* trigger rendering or audio effects.
*/
b2ParticleSystem.prototype.GetBodyContacts = function () {
return this.m_bodyContactBuffer.data;
};
b2ParticleSystem.prototype.GetBodyContactCount = function () {
return this.m_bodyContactBuffer.count;
};
/**
* Get array of particle pairs. The particles in a pair:
* (1) are contacting,
* (2) are in the same particle group,
* (3) are part of a rigid particle group, or are spring, elastic,
* or wall particles.
* (4) have at least one particle that is a spring or barrier
* particle (i.e. one of the types in k_pairFlags),
* (5) have at least one particle that returns true for
* ConnectionFilter::IsNecessary,
* (6) are not zombie particles.
*
* Essentially, this is an array of spring or barrier particles
* that are interacting. The array is sorted by b2ParticlePair's
* indexA, and then indexB. There are no duplicate entries.
*/
b2ParticleSystem.prototype.GetPairs = function () {
return this.m_pairBuffer.data;
};
b2ParticleSystem.prototype.GetPairCount = function () {
return this.m_pairBuffer.count;
};
/**
* Get array of particle triads. The particles in a triad:
* (1) are in the same particle group,
* (2) are in a Voronoi triangle together,
* (3) are within b2_maxTriadDistance particle diameters of each
* other,
* (4) return true for ConnectionFilter::ShouldCreateTriad
* (5) have at least one particle of type elastic (i.e. one of the
* types in k_triadFlags),
* (6) are part of a rigid particle group, or are spring, elastic,
* or wall particles.
* (7) are not zombie particles.
*
* Essentially, this is an array of elastic particles that are
* interacting. The array is sorted by b2ParticleTriad's indexA,
* then indexB, then indexC. There are no duplicate entries.
*/
b2ParticleSystem.prototype.GetTriads = function () {
return this.m_triadBuffer.data;
};
b2ParticleSystem.prototype.GetTriadCount = function () {
return this.m_triadBuffer.count;
};
/**
* Set an optional threshold for the maximum number of
* consecutive particle iterations that a particle may contact
* multiple bodies before it is considered a candidate for being
* "stuck". Setting to zero or less disables.
*/
b2ParticleSystem.prototype.SetStuckThreshold = function (steps) {
this.m_stuckThreshold = steps;
if (steps > 0) {
this.m_lastBodyContactStepBuffer.data = this.RequestBuffer(this.m_lastBodyContactStepBuffer.data);
this.m_bodyContactCountBuffer.data = this.RequestBuffer(this.m_bodyContactCountBuffer.data);
this.m_consecutiveContactStepsBuffer.data = this.RequestBuffer(this.m_consecutiveContactStepsBuffer.data);
}
};
/**
* Get potentially stuck particles from the last step; the user
* must decide if they are stuck or not, and if so, delete or
* move them
*/
b2ParticleSystem.prototype.GetStuckCandidates = function () {
///return m_stuckParticleBuffer.Data();
return this.m_stuckParticleBuffer.Data();
};
/**
* Get the number of stuck particle candidates from the last
* step.
*/
b2ParticleSystem.prototype.GetStuckCandidateCount = function () {
///return m_stuckParticleBuffer.GetCount();
return this.m_stuckParticleBuffer.GetCount();
};
/**
* Compute the kinetic energy that can be lost by damping force
*/
b2ParticleSystem.prototype.ComputeCollisionEnergy = function () {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var s_v = b2ParticleSystem.ComputeCollisionEnergy_s_v;
var vel_data = this.m_velocityBuffer.data;
var sum_v2 = 0;
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
var n = contact.normal;
///b2Vec2 v = m_velocityBuffer.data[b] - m_velocityBuffer.data[a];
var v = b2Vec2.SubVV(vel_data[b], vel_data[a], s_v);
var vn = b2Vec2.DotVV(v, n);
if (vn < 0) {
sum_v2 += vn * vn;
}
}
return 0.5 * this.GetParticleMass() * sum_v2;
};
/**
* Set strict Particle/Body contact check.
*
* This is an option that will help ensure correct behavior if
* there are corners in the world model where Particle/Body
* contact is ambiguous. This option scales at n*log(n) of the
* number of Particle/Body contacts, so it is best to only
* enable if it is necessary for your geometry. Enable if you
* see strange particle behavior around b2Body intersections.
*/
b2ParticleSystem.prototype.SetStrictContactCheck = function (enabled) {
this.m_def.strictContactCheck = enabled;
};
/**
* Get the status of the strict contact check.
*/
b2ParticleSystem.prototype.GetStrictContactCheck = function () {
return this.m_def.strictContactCheck;
};
/**
* Set the lifetime (in seconds) of a particle relative to the
* current time. A lifetime of less than or equal to 0.0f
* results in the particle living forever until it's manually
* destroyed by the application.
*/
b2ParticleSystem.prototype.SetParticleLifetime = function (index, lifetime) {
// DEBUG: b2Assert(this.ValidateParticleIndex(index));
var initializeExpirationTimes = this.m_indexByExpirationTimeBuffer.data === null;
this.m_expirationTimeBuffer.data = this.RequestBuffer(this.m_expirationTimeBuffer.data);
this.m_indexByExpirationTimeBuffer.data = this.RequestBuffer(this.m_indexByExpirationTimeBuffer.data);
// Initialize the inverse mapping buffer.
if (initializeExpirationTimes) {
var particleCount = this.GetParticleCount();
for (var i = 0; i < particleCount; ++i) {
this.m_indexByExpirationTimeBuffer.data[i] = i;
}
}
///const int32 quantizedLifetime = (int32)(lifetime / m_def.lifetimeGranularity);
var quantizedLifetime = lifetime / this.m_def.lifetimeGranularity;
// Use a negative lifetime so that it's possible to track which
// of the infinite lifetime particles are older.
var newExpirationTime = quantizedLifetime > 0.0 ? this.GetQuantizedTimeElapsed() + quantizedLifetime : quantizedLifetime;
if (newExpirationTime !== this.m_expirationTimeBuffer.data[index]) {
this.m_expirationTimeBuffer.data[index] = newExpirationTime;
this.m_expirationTimeBufferRequiresSorting = true;
}
};
/**
* Get the lifetime (in seconds) of a particle relative to the
* current time. A value > 0.0f is returned if the particle is
* scheduled to be destroyed in the future, values <= 0.0f
* indicate the particle has an infinite lifetime.
*/
b2ParticleSystem.prototype.GetParticleLifetime = function (index) {
// DEBUG: b2Assert(this.ValidateParticleIndex(index));
return this.ExpirationTimeToLifetime(this.GetExpirationTimeBuffer()[index]);
};
/**
* Enable / disable destruction of particles in CreateParticle()
* when no more particles can be created due to a prior call to
* SetMaxParticleCount(). When this is enabled, the oldest
* particle is destroyed in CreateParticle() favoring the
* destruction of particles with a finite lifetime over
* particles with infinite lifetimes. This feature is enabled by
* default when particle lifetimes are tracked. Explicitly
* enabling this feature using this function enables particle
* lifetime tracking.
*/
b2ParticleSystem.prototype.SetDestructionByAge = function (enable) {
if (enable) {
this.GetExpirationTimeBuffer();
}
this.m_def.destroyByAge = enable;
};
/**
* Get whether the oldest particle will be destroyed in
* CreateParticle() when the maximum number of particles are
* present in the system.
*/
b2ParticleSystem.prototype.GetDestructionByAge = function () {
return this.m_def.destroyByAge;
};
/**
* Get the array of particle expiration times indexed by
* particle index.
*
* GetParticleCount() items are in the returned array.
*/
b2ParticleSystem.prototype.GetExpirationTimeBuffer = function () {
this.m_expirationTimeBuffer.data = this.RequestBuffer(this.m_expirationTimeBuffer.data);
return this.m_expirationTimeBuffer.data;
};
/**
* Convert a expiration time value in returned by
* GetExpirationTimeBuffer() to a time in seconds relative to
* the current simulation time.
*/
b2ParticleSystem.prototype.ExpirationTimeToLifetime = function (expirationTime) {
return (expirationTime > 0 ?
expirationTime - this.GetQuantizedTimeElapsed() :
expirationTime) * this.m_def.lifetimeGranularity;
};
/**
* Get the array of particle indices ordered by reverse
* lifetime. The oldest particle indexes are at the end of the
* array with the newest at the start. Particles with infinite
* lifetimes (i.e expiration times less than or equal to 0) are
* placed at the start of the array.
* ExpirationTimeToLifetime(GetExpirationTimeBuffer()[index]) is
* equivalent to GetParticleLifetime(index).
*
* GetParticleCount() items are in the returned array.
*/
b2ParticleSystem.prototype.GetIndexByExpirationTimeBuffer = function () {
// If particles are present, initialize / reinitialize the lifetime buffer.
if (this.GetParticleCount()) {
this.SetParticleLifetime(0, this.GetParticleLifetime(0));
}
else {
this.m_indexByExpirationTimeBuffer.data = this.RequestBuffer(this.m_indexByExpirationTimeBuffer.data);
}
if (!this.m_indexByExpirationTimeBuffer.data) {
throw new Error();
}
return this.m_indexByExpirationTimeBuffer.data;
};
/**
* Apply an impulse to one particle. This immediately modifies
* the velocity. Similar to b2Body::ApplyLinearImpulse.
*
* @param index the particle that will be modified.
* @param impulse impulse the world impulse vector, usually in N-seconds or kg-m/s.
*/
b2ParticleSystem.prototype.ParticleApplyLinearImpulse = function (index, impulse) {
this.ApplyLinearImpulse(index, index + 1, impulse);
};
/**
* Apply an impulse to all particles between 'firstIndex' and
* 'lastIndex'. This immediately modifies the velocity. Note
* that the impulse is applied to the total mass of all
* particles. So, calling ParticleApplyLinearImpulse(0, impulse)
* and ParticleApplyLinearImpulse(1, impulse) will impart twice
* as much velocity as calling just ApplyLinearImpulse(0, 1,
* impulse).
*
* @param firstIndex the first particle to be modified.
* @param lastIndex the last particle to be modified.
* @param impulse the world impulse vector, usually in N-seconds or kg-m/s.
*/
b2ParticleSystem.prototype.ApplyLinearImpulse = function (firstIndex, lastIndex, impulse) {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
var numParticles = (lastIndex - firstIndex);
var totalMass = numParticles * this.GetParticleMass();
///const b2Vec2 velocityDelta = impulse / totalMass;
var velocityDelta = new b2Vec2().Copy(impulse).SelfMul(1 / totalMass);
for (var i = firstIndex; i < lastIndex; i++) {
///m_velocityBuffer.data[i] += velocityDelta;
vel_data[i].SelfAdd(velocityDelta);
}
};
b2ParticleSystem.IsSignificantForce = function (force) {
return force.x !== 0 || force.y !== 0;
};
/**
* Apply a force to the center of a particle.
*
* @param index the particle that will be modified.
* @param force the world force vector, usually in Newtons (N).
*/
b2ParticleSystem.prototype.ParticleApplyForce = function (index, force) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (b2ParticleSystem.IsSignificantForce(force) &&
this.ForceCanBeApplied(this.m_flagsBuffer.data[index])) {
this.PrepareForceBuffer();
///m_forceBuffer[index] += force;
this.m_forceBuffer[index].SelfAdd(force);
}
};
/**
* Distribute a force across several particles. The particles
* must not be wall particles. Note that the force is
* distributed across all the particles, so calling this
* function for indices 0..N is not the same as calling
* ParticleApplyForce(i, force) for i in 0..N.
*
* @param firstIndex the first particle to be modified.
* @param lastIndex the last particle to be modified.
* @param force the world force vector, usually in Newtons (N).
*/
b2ParticleSystem.prototype.ApplyForce = function (firstIndex, lastIndex, force) {
// Ensure we're not trying to apply force to particles that can't move,
// such as wall particles.
// DEBUG: if (!this.m_flagsBuffer.data) { throw new Error(); }
// DEBUG: let flags = 0;
// DEBUG: for (let i = firstIndex; i < lastIndex; i++) {
// DEBUG: flags |= this.m_flagsBuffer.data[i];
// DEBUG: }
// DEBUG: b2Assert(this.ForceCanBeApplied(flags));
// Early out if force does nothing (optimization).
///const b2Vec2 distributedForce = force / (float32)(lastIndex - firstIndex);
var distributedForce = new b2Vec2().Copy(force).SelfMul(1 / (lastIndex - firstIndex));
if (b2ParticleSystem.IsSignificantForce(distributedForce)) {
this.PrepareForceBuffer();
// Distribute the force over all the particles.
for (var i = firstIndex; i < lastIndex; i++) {
///m_forceBuffer[i] += distributedForce;
this.m_forceBuffer[i].SelfAdd(distributedForce);
}
}
};
/**
* Get the next particle-system in the world's particle-system
* list.
*/
b2ParticleSystem.prototype.GetNext = function () {
return this.m_next;
};
/**
* Query the particle system for all particles that potentially
* overlap the provided AABB.
* b2QueryCallback::ShouldQueryParticleSystem is ignored.
*
* @param callback a user implemented callback class.
* @param aabb the query box.
*/
b2ParticleSystem.prototype.QueryAABB = function (callback, aabb) {
if (this.m_proxyBuffer.count === 0) {
return;
}
var beginProxy = 0;
var endProxy = this.m_proxyBuffer.count;
var firstProxy = std_lower_bound(this.m_proxyBuffer.data, beginProxy, endProxy, b2ParticleSystem.computeTag(this.m_inverseDiameter * aabb.lowerBound.x, this.m_inverseDiameter * aabb.lowerBound.y), b2ParticleSystem.Proxy.CompareProxyTag);
var lastProxy = std_upper_bound(this.m_proxyBuffer.data, firstProxy, endProxy, b2ParticleSystem.computeTag(this.m_inverseDiameter * aabb.upperBound.x, this.m_inverseDiameter * aabb.upperBound.y), b2ParticleSystem.Proxy.CompareTagProxy);
if (!this.m_positionBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
for (var k = firstProxy; k < lastProxy; ++k) {
var proxy = this.m_proxyBuffer.data[k];
var i = proxy.index;
var p = pos_data[i];
if (aabb.lowerBound.x < p.x && p.x < aabb.upperBound.x &&
aabb.lowerBound.y < p.y && p.y < aabb.upperBound.y) {
if (!callback.ReportParticle(this, i)) {
break;
}
}
}
};
/**
* Query the particle system for all particles that potentially
* overlap the provided shape's AABB. Calls QueryAABB
* internally. b2QueryCallback::ShouldQueryParticleSystem is
* ignored.
*
* @param callback a user implemented callback class.
* @param shape the query shape
* @param xf the transform of the AABB
* @param childIndex
*/
b2ParticleSystem.prototype.QueryShapeAABB = function (callback, shape, xf, childIndex) {
if (childIndex === void 0) { childIndex = 0; }
var s_aabb = b2ParticleSystem.QueryShapeAABB_s_aabb;
var aabb = s_aabb;
shape.ComputeAABB(aabb, xf, childIndex);
this.QueryAABB(callback, aabb);
};
b2ParticleSystem.prototype.QueryPointAABB = function (callback, point, slop) {
if (slop === void 0) { slop = b2_linearSlop; }
var s_aabb = b2ParticleSystem.QueryPointAABB_s_aabb;
var aabb = s_aabb;
aabb.lowerBound.Set(point.x - slop, point.y - slop);
aabb.upperBound.Set(point.x + slop, point.y + slop);
this.QueryAABB(callback, aabb);
};
/**
* Ray-cast the particle system for all particles in the path of
* the ray. Your callback controls whether you get the closest
* point, any point, or n-points. The ray-cast ignores particles
* that contain the starting point.
* b2RayCastCallback::ShouldQueryParticleSystem is ignored.
*
* @param callback a user implemented callback class.
* @param point1 the ray starting point
* @param point2 the ray ending point
*/
b2ParticleSystem.prototype.RayCast = function (callback, point1, point2) {
var s_aabb = b2ParticleSystem.RayCast_s_aabb;
var s_p = b2ParticleSystem.RayCast_s_p;
var s_v = b2ParticleSystem.RayCast_s_v;
var s_n = b2ParticleSystem.RayCast_s_n;
var s_point = b2ParticleSystem.RayCast_s_point;
if (this.m_proxyBuffer.count === 0) {
return;
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var aabb = s_aabb;
b2Vec2.MinV(point1, point2, aabb.lowerBound);
b2Vec2.MaxV(point1, point2, aabb.upperBound);
var fraction = 1;
// solving the following equation:
// ((1-t)*point1+t*point2-position)^2=diameter^2
// where t is a potential fraction
///b2Vec2 v = point2 - point1;
var v = b2Vec2.SubVV(point2, point1, s_v);
var v2 = b2Vec2.DotVV(v, v);
var enumerator = this.GetInsideBoundsEnumerator(aabb);
var i;
while ((i = enumerator.GetNext()) >= 0) {
///b2Vec2 p = point1 - m_positionBuffer.data[i];
var p = b2Vec2.SubVV(point1, pos_data[i], s_p);
var pv = b2Vec2.DotVV(p, v);
var p2 = b2Vec2.DotVV(p, p);
var determinant = pv * pv - v2 * (p2 - this.m_squaredDiameter);
if (determinant >= 0) {
var sqrtDeterminant = b2Sqrt(determinant);
// find a solution between 0 and fraction
var t = (-pv - sqrtDeterminant) / v2;
if (t > fraction) {
continue;
}
if (t < 0) {
t = (-pv + sqrtDeterminant) / v2;
if (t < 0 || t > fraction) {
continue;
}
}
///b2Vec2 n = p + t * v;
var n = b2Vec2.AddVMulSV(p, t, v, s_n);
n.Normalize();
///float32 f = callback.ReportParticle(this, i, point1 + t * v, n, t);
var f = callback.ReportParticle(this, i, b2Vec2.AddVMulSV(point1, t, v, s_point), n, t);
fraction = b2Min(fraction, f);
if (fraction <= 0) {
break;
}
}
}
};
/**
* Compute the axis-aligned bounding box for all particles
* contained within this particle system.
* @param aabb Returns the axis-aligned bounding box of the system.
*/
b2ParticleSystem.prototype.ComputeAABB = function (aabb) {
var particleCount = this.GetParticleCount();
// DEBUG: b2Assert(aabb !== null);
aabb.lowerBound.x = +b2_maxFloat;
aabb.lowerBound.y = +b2_maxFloat;
aabb.upperBound.x = -b2_maxFloat;
aabb.upperBound.y = -b2_maxFloat;
if (!this.m_positionBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
for (var i = 0; i < particleCount; i++) {
var p = pos_data[i];
b2Vec2.MinV(aabb.lowerBound, p, aabb.lowerBound);
b2Vec2.MaxV(aabb.upperBound, p, aabb.upperBound);
}
aabb.lowerBound.x -= this.m_particleDiameter;
aabb.lowerBound.y -= this.m_particleDiameter;
aabb.upperBound.x += this.m_particleDiameter;
aabb.upperBound.y += this.m_particleDiameter;
};
b2ParticleSystem.prototype.FreeBuffer = function (b, capacity) {
if (b === null) {
return;
}
b.length = 0;
};
b2ParticleSystem.prototype.FreeUserOverridableBuffer = function (b) {
if (b.userSuppliedCapacity === 0) {
this.FreeBuffer(b.data, this.m_internalAllocatedCapacity);
}
};
/**
* Reallocate a buffer
*/
b2ParticleSystem.prototype.ReallocateBuffer3 = function (oldBuffer, oldCapacity, newCapacity) {
// b2Assert(newCapacity > oldCapacity);
if (newCapacity <= oldCapacity) {
throw new Error();
}
var newBuffer = (oldBuffer) ? oldBuffer.slice() : [];
newBuffer.length = newCapacity;
return newBuffer;
};
/**
* Reallocate a buffer
*/
b2ParticleSystem.prototype.ReallocateBuffer5 = function (buffer, userSuppliedCapacity, oldCapacity, newCapacity, deferred) {
// b2Assert(newCapacity > oldCapacity);
if (newCapacity <= oldCapacity) {
throw new Error();
}
// A 'deferred' buffer is reallocated only if it is not NULL.
// If 'userSuppliedCapacity' is not zero, buffer is user supplied and must
// be kept.
// b2Assert(!userSuppliedCapacity || newCapacity <= userSuppliedCapacity);
if (!(!userSuppliedCapacity || newCapacity <= userSuppliedCapacity)) {
throw new Error();
}
if ((!deferred || buffer) && !userSuppliedCapacity) {
buffer = this.ReallocateBuffer3(buffer, oldCapacity, newCapacity);
}
return buffer; // TODO: fix this
};
/**
* Reallocate a buffer
*/
b2ParticleSystem.prototype.ReallocateBuffer4 = function (buffer, oldCapacity, newCapacity, deferred) {
// DEBUG: b2Assert(newCapacity > oldCapacity);
return this.ReallocateBuffer5(buffer.data, buffer.userSuppliedCapacity, oldCapacity, newCapacity, deferred);
};
b2ParticleSystem.prototype.RequestBuffer = function (buffer) {
if (!buffer) {
if (this.m_internalAllocatedCapacity === 0) {
this.ReallocateInternalAllocatedBuffers(b2_minParticleSystemBufferCapacity);
}
buffer = [];
buffer.length = this.m_internalAllocatedCapacity;
}
return buffer;
};
/**
* Reallocate the handle / index map and schedule the allocation
* of a new pool for handle allocation.
*/
b2ParticleSystem.prototype.ReallocateHandleBuffers = function (newCapacity) {
// DEBUG: b2Assert(newCapacity > this.m_internalAllocatedCapacity);
// Reallocate a new handle / index map buffer, copying old handle pointers
// is fine since they're kept around.
this.m_handleIndexBuffer.data = this.ReallocateBuffer4(this.m_handleIndexBuffer, this.m_internalAllocatedCapacity, newCapacity, true);
// Set the size of the next handle allocation.
///this.m_handleAllocator.SetItemsPerSlab(newCapacity - this.m_internalAllocatedCapacity);
};
b2ParticleSystem.prototype.ReallocateInternalAllocatedBuffers = function (capacity) {
function LimitCapacity(capacity, maxCount) {
return maxCount && capacity > maxCount ? maxCount : capacity;
}
// Don't increase capacity beyond the smallest user-supplied buffer size.
capacity = LimitCapacity(capacity, this.m_def.maxCount);
capacity = LimitCapacity(capacity, this.m_flagsBuffer.userSuppliedCapacity);
capacity = LimitCapacity(capacity, this.m_positionBuffer.userSuppliedCapacity);
capacity = LimitCapacity(capacity, this.m_velocityBuffer.userSuppliedCapacity);
capacity = LimitCapacity(capacity, this.m_colorBuffer.userSuppliedCapacity);
capacity = LimitCapacity(capacity, this.m_userDataBuffer.userSuppliedCapacity);
if (this.m_internalAllocatedCapacity < capacity) {
this.ReallocateHandleBuffers(capacity);
this.m_flagsBuffer.data = this.ReallocateBuffer4(this.m_flagsBuffer, this.m_internalAllocatedCapacity, capacity, false);
// Conditionally defer these as they are optional if the feature is
// not enabled.
var stuck = this.m_stuckThreshold > 0;
this.m_lastBodyContactStepBuffer.data = this.ReallocateBuffer4(this.m_lastBodyContactStepBuffer, this.m_internalAllocatedCapacity, capacity, stuck);
this.m_bodyContactCountBuffer.data = this.ReallocateBuffer4(this.m_bodyContactCountBuffer, this.m_internalAllocatedCapacity, capacity, stuck);
this.m_consecutiveContactStepsBuffer.data = this.ReallocateBuffer4(this.m_consecutiveContactStepsBuffer, this.m_internalAllocatedCapacity, capacity, stuck);
this.m_positionBuffer.data = this.ReallocateBuffer4(this.m_positionBuffer, this.m_internalAllocatedCapacity, capacity, false);
this.m_velocityBuffer.data = this.ReallocateBuffer4(this.m_velocityBuffer, this.m_internalAllocatedCapacity, capacity, false);
this.m_forceBuffer = this.ReallocateBuffer5(this.m_forceBuffer, 0, this.m_internalAllocatedCapacity, capacity, false);
this.m_weightBuffer = this.ReallocateBuffer5(this.m_weightBuffer, 0, this.m_internalAllocatedCapacity, capacity, false);
this.m_staticPressureBuffer = this.ReallocateBuffer5(this.m_staticPressureBuffer, 0, this.m_internalAllocatedCapacity, capacity, true);
this.m_accumulationBuffer = this.ReallocateBuffer5(this.m_accumulationBuffer, 0, this.m_internalAllocatedCapacity, capacity, false);
this.m_accumulation2Buffer = this.ReallocateBuffer5(this.m_accumulation2Buffer, 0, this.m_internalAllocatedCapacity, capacity, true);
this.m_depthBuffer = this.ReallocateBuffer5(this.m_depthBuffer, 0, this.m_internalAllocatedCapacity, capacity, true);
this.m_colorBuffer.data = this.ReallocateBuffer4(this.m_colorBuffer, this.m_internalAllocatedCapacity, capacity, true);
this.m_groupBuffer = this.ReallocateBuffer5(this.m_groupBuffer, 0, this.m_internalAllocatedCapacity, capacity, false);
this.m_userDataBuffer.data = this.ReallocateBuffer4(this.m_userDataBuffer, this.m_internalAllocatedCapacity, capacity, true);
this.m_expirationTimeBuffer.data = this.ReallocateBuffer4(this.m_expirationTimeBuffer, this.m_internalAllocatedCapacity, capacity, true);
this.m_indexByExpirationTimeBuffer.data = this.ReallocateBuffer4(this.m_indexByExpirationTimeBuffer, this.m_internalAllocatedCapacity, capacity, false);
this.m_internalAllocatedCapacity = capacity;
}
};
b2ParticleSystem.prototype.CreateParticleForGroup = function (groupDef, xf, p) {
var particleDef = new b2ParticleDef();
particleDef.flags = b2Maybe(groupDef.flags, 0);
///particleDef.position = b2Mul(xf, p);
b2Transform.MulXV(xf, p, particleDef.position);
///particleDef.velocity =
/// groupDef.linearVelocity +
/// b2Cross(groupDef.angularVelocity,
/// particleDef.position - groupDef.position);
b2Vec2.AddVV(b2Maybe(groupDef.linearVelocity, b2Vec2.ZERO), b2Vec2.CrossSV(b2Maybe(groupDef.angularVelocity, 0), b2Vec2.SubVV(particleDef.position, b2Maybe(groupDef.position, b2Vec2.ZERO), b2Vec2.s_t0), b2Vec2.s_t0), particleDef.velocity);
particleDef.color.Copy(b2Maybe(groupDef.color, b2Color.ZERO));
particleDef.lifetime = b2Maybe(groupDef.lifetime, 0);
particleDef.userData = groupDef.userData;
this.CreateParticle(particleDef);
};
b2ParticleSystem.prototype.CreateParticlesStrokeShapeForGroup = function (shape, groupDef, xf) {
var s_edge = b2ParticleSystem.CreateParticlesStrokeShapeForGroup_s_edge;
var s_d = b2ParticleSystem.CreateParticlesStrokeShapeForGroup_s_d;
var s_p = b2ParticleSystem.CreateParticlesStrokeShapeForGroup_s_p;
var stride = b2Maybe(groupDef.stride, 0);
if (stride === 0) {
stride = this.GetParticleStride();
}
var positionOnEdge = 0;
var childCount = shape.GetChildCount();
for (var childIndex = 0; childIndex < childCount; childIndex++) {
var edge = null;
if (shape.GetType() === exports.b2ShapeType.e_edgeShape) {
edge = shape;
}
else {
// DEBUG: b2Assert(shape.GetType() === b2ShapeType.e_chainShape);
edge = s_edge;
shape.GetChildEdge(edge, childIndex);
}
var d = b2Vec2.SubVV(edge.m_vertex2, edge.m_vertex1, s_d);
var edgeLength = d.Length();
while (positionOnEdge < edgeLength) {
///b2Vec2 p = edge.m_vertex1 + positionOnEdge / edgeLength * d;
var p = b2Vec2.AddVMulSV(edge.m_vertex1, positionOnEdge / edgeLength, d, s_p);
this.CreateParticleForGroup(groupDef, xf, p);
positionOnEdge += stride;
}
positionOnEdge -= edgeLength;
}
};
b2ParticleSystem.prototype.CreateParticlesFillShapeForGroup = function (shape, groupDef, xf) {
var s_aabb = b2ParticleSystem.CreateParticlesFillShapeForGroup_s_aabb;
var s_p = b2ParticleSystem.CreateParticlesFillShapeForGroup_s_p;
var stride = b2Maybe(groupDef.stride, 0);
if (stride === 0) {
stride = this.GetParticleStride();
}
///b2Transform identity;
/// identity.SetIdentity();
var identity = b2Transform.IDENTITY;
var aabb = s_aabb;
// DEBUG: b2Assert(shape.GetChildCount() === 1);
shape.ComputeAABB(aabb, identity, 0);
for (var y = Math.floor(aabb.lowerBound.y / stride) * stride; y < aabb.upperBound.y; y += stride) {
for (var x = Math.floor(aabb.lowerBound.x / stride) * stride; x < aabb.upperBound.x; x += stride) {
var p = s_p.Set(x, y);
if (shape.TestPoint(identity, p)) {
this.CreateParticleForGroup(groupDef, xf, p);
}
}
}
};
b2ParticleSystem.prototype.CreateParticlesWithShapeForGroup = function (shape, groupDef, xf) {
switch (shape.GetType()) {
case exports.b2ShapeType.e_edgeShape:
case exports.b2ShapeType.e_chainShape:
this.CreateParticlesStrokeShapeForGroup(shape, groupDef, xf);
break;
case exports.b2ShapeType.e_polygonShape:
case exports.b2ShapeType.e_circleShape:
this.CreateParticlesFillShapeForGroup(shape, groupDef, xf);
break;
default:
// DEBUG: b2Assert(false);
break;
}
};
b2ParticleSystem.prototype.CreateParticlesWithShapesForGroup = function (shapes, shapeCount, groupDef, xf) {
var compositeShape = new b2ParticleSystem.CompositeShape(shapes, shapeCount);
this.CreateParticlesFillShapeForGroup(compositeShape, groupDef, xf);
};
b2ParticleSystem.prototype.CloneParticle = function (oldIndex, group) {
var def = new b2ParticleDef();
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
def.flags = this.m_flagsBuffer.data[oldIndex];
def.position.Copy(this.m_positionBuffer.data[oldIndex]);
def.velocity.Copy(this.m_velocityBuffer.data[oldIndex]);
if (this.m_colorBuffer.data) {
def.color.Copy(this.m_colorBuffer.data[oldIndex]);
}
if (this.m_userDataBuffer.data) {
def.userData = this.m_userDataBuffer.data[oldIndex];
}
def.group = group;
var newIndex = this.CreateParticle(def);
if (this.m_handleIndexBuffer.data) {
var handle = this.m_handleIndexBuffer.data[oldIndex];
if (handle) {
handle.SetIndex(newIndex);
}
this.m_handleIndexBuffer.data[newIndex] = handle;
this.m_handleIndexBuffer.data[oldIndex] = null;
}
if (this.m_lastBodyContactStepBuffer.data) {
this.m_lastBodyContactStepBuffer.data[newIndex] =
this.m_lastBodyContactStepBuffer.data[oldIndex];
}
if (this.m_bodyContactCountBuffer.data) {
this.m_bodyContactCountBuffer.data[newIndex] =
this.m_bodyContactCountBuffer.data[oldIndex];
}
if (this.m_consecutiveContactStepsBuffer.data) {
this.m_consecutiveContactStepsBuffer.data[newIndex] =
this.m_consecutiveContactStepsBuffer.data[oldIndex];
}
if (this.m_hasForce) {
this.m_forceBuffer[newIndex].Copy(this.m_forceBuffer[oldIndex]);
}
if (this.m_staticPressureBuffer) {
this.m_staticPressureBuffer[newIndex] = this.m_staticPressureBuffer[oldIndex];
}
if (this.m_depthBuffer) {
this.m_depthBuffer[newIndex] = this.m_depthBuffer[oldIndex];
}
if (this.m_expirationTimeBuffer.data) {
this.m_expirationTimeBuffer.data[newIndex] =
this.m_expirationTimeBuffer.data[oldIndex];
}
return newIndex;
};
b2ParticleSystem.prototype.DestroyParticlesInGroup = function (group, callDestructionListener) {
if (callDestructionListener === void 0) { callDestructionListener = false; }
for (var i = group.m_firstIndex; i < group.m_lastIndex; i++) {
this.DestroyParticle(i, callDestructionListener);
}
};
b2ParticleSystem.prototype.DestroyParticleGroup = function (group) {
// DEBUG: b2Assert(this.m_groupCount > 0);
// DEBUG: b2Assert(group !== null);
if (this.m_world.m_destructionListener) {
this.m_world.m_destructionListener.SayGoodbyeParticleGroup(group);
}
this.SetGroupFlags(group, 0);
for (var i = group.m_firstIndex; i < group.m_lastIndex; i++) {
this.m_groupBuffer[i] = null;
}
if (group.m_prev) {
group.m_prev.m_next = group.m_next;
}
if (group.m_next) {
group.m_next.m_prev = group.m_prev;
}
if (group === this.m_groupList) {
this.m_groupList = group.m_next;
}
--this.m_groupCount;
};
b2ParticleSystem.ParticleCanBeConnected = function (flags, group) {
return ((flags & (exports.b2ParticleFlag.b2_wallParticle | exports.b2ParticleFlag.b2_springParticle | exports.b2ParticleFlag.b2_elasticParticle)) !== 0) ||
((group !== null) && ((group.GetGroupFlags() & exports.b2ParticleGroupFlag.b2_rigidParticleGroup) !== 0));
};
b2ParticleSystem.prototype.UpdatePairsAndTriads = function (firstIndex, lastIndex, filter) {
var s_dab = b2ParticleSystem.UpdatePairsAndTriads_s_dab;
var s_dbc = b2ParticleSystem.UpdatePairsAndTriads_s_dbc;
var s_dca = b2ParticleSystem.UpdatePairsAndTriads_s_dca;
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
// Create pairs or triads.
// All particles in each pair/triad should satisfy the following:
// * firstIndex <= index < lastIndex
// * don't have b2_zombieParticle
// * ParticleCanBeConnected returns true
// * ShouldCreatePair/ShouldCreateTriad returns true
// Any particles in each pair/triad should satisfy the following:
// * filter.IsNeeded returns true
// * have one of k_pairFlags/k_triadsFlags
// DEBUG: b2Assert(firstIndex <= lastIndex);
var particleFlags = 0;
for (var i = firstIndex; i < lastIndex; i++) {
particleFlags |= this.m_flagsBuffer.data[i];
}
if (particleFlags & b2ParticleSystem.k_pairFlags) {
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
var af = this.m_flagsBuffer.data[a];
var bf = this.m_flagsBuffer.data[b];
var groupA = this.m_groupBuffer[a];
var groupB = this.m_groupBuffer[b];
if (a >= firstIndex && a < lastIndex &&
b >= firstIndex && b < lastIndex &&
!((af | bf) & exports.b2ParticleFlag.b2_zombieParticle) &&
((af | bf) & b2ParticleSystem.k_pairFlags) &&
(filter.IsNecessary(a) || filter.IsNecessary(b)) &&
b2ParticleSystem.ParticleCanBeConnected(af, groupA) &&
b2ParticleSystem.ParticleCanBeConnected(bf, groupB) &&
filter.ShouldCreatePair(a, b)) {
///b2ParticlePair& pair = m_pairBuffer.Append();
var pair = this.m_pairBuffer.data[this.m_pairBuffer.Append()];
pair.indexA = a;
pair.indexB = b;
pair.flags = contact.flags;
pair.strength = b2Min(groupA ? groupA.m_strength : 1, groupB ? groupB.m_strength : 1);
///pair.distance = b2Distance(pos_data[a], pos_data[b]); // TODO: this was wrong!
pair.distance = b2Vec2.DistanceVV(pos_data[a], pos_data[b]);
}
///std::stable_sort(m_pairBuffer.Begin(), m_pairBuffer.End(), ComparePairIndices);
std_stable_sort(this.m_pairBuffer.data, 0, this.m_pairBuffer.count, b2ParticleSystem.ComparePairIndices);
///m_pairBuffer.Unique(MatchPairIndices);
this.m_pairBuffer.Unique(b2ParticleSystem.MatchPairIndices);
}
}
if (particleFlags & b2ParticleSystem.k_triadFlags) {
var diagram = new b2VoronoiDiagram(lastIndex - firstIndex);
///let necessary_count = 0;
for (var i = firstIndex; i < lastIndex; i++) {
var flags = this.m_flagsBuffer.data[i];
var group = this.m_groupBuffer[i];
if (!(flags & exports.b2ParticleFlag.b2_zombieParticle) &&
b2ParticleSystem.ParticleCanBeConnected(flags, group)) {
///if (filter.IsNecessary(i)) {
///++necessary_count;
///}
diagram.AddGenerator(pos_data[i], i, filter.IsNecessary(i));
}
}
///if (necessary_count === 0) {
/////debugger;
///for (let i = firstIndex; i < lastIndex; i++) {
/// filter.IsNecessary(i);
///}
///}
var stride = this.GetParticleStride();
diagram.Generate(stride / 2, stride * 2);
var system_1 = this;
var callback = /*UpdateTriadsCallback*/ function (a, b, c) {
if (!system_1.m_flagsBuffer.data) {
throw new Error();
}
var af = system_1.m_flagsBuffer.data[a];
var bf = system_1.m_flagsBuffer.data[b];
var cf = system_1.m_flagsBuffer.data[c];
if (((af | bf | cf) & b2ParticleSystem.k_triadFlags) &&
filter.ShouldCreateTriad(a, b, c)) {
var pa = pos_data[a];
var pb = pos_data[b];
var pc = pos_data[c];
var dab = b2Vec2.SubVV(pa, pb, s_dab);
var dbc = b2Vec2.SubVV(pb, pc, s_dbc);
var dca = b2Vec2.SubVV(pc, pa, s_dca);
var maxDistanceSquared = b2_maxTriadDistanceSquared * system_1.m_squaredDiameter;
if (b2Vec2.DotVV(dab, dab) > maxDistanceSquared ||
b2Vec2.DotVV(dbc, dbc) > maxDistanceSquared ||
b2Vec2.DotVV(dca, dca) > maxDistanceSquared) {
return;
}
var groupA = system_1.m_groupBuffer[a];
var groupB = system_1.m_groupBuffer[b];
var groupC = system_1.m_groupBuffer[c];
///b2ParticleTriad& triad = m_system.m_triadBuffer.Append();
var triad = system_1.m_triadBuffer.data[system_1.m_triadBuffer.Append()];
triad.indexA = a;
triad.indexB = b;
triad.indexC = c;
triad.flags = af | bf | cf;
triad.strength = b2Min(b2Min(groupA ? groupA.m_strength : 1, groupB ? groupB.m_strength : 1), groupC ? groupC.m_strength : 1);
///let midPoint = b2Vec2.MulSV(1.0 / 3.0, b2Vec2.AddVV(pa, b2Vec2.AddVV(pb, pc, new b2Vec2()), new b2Vec2()), new b2Vec2());
var midPoint_x = (pa.x + pb.x + pc.x) / 3.0;
var midPoint_y = (pa.y + pb.y + pc.y) / 3.0;
///triad.pa = b2Vec2.SubVV(pa, midPoint, new b2Vec2());
triad.pa.x = pa.x - midPoint_x;
triad.pa.y = pa.y - midPoint_y;
///triad.pb = b2Vec2.SubVV(pb, midPoint, new b2Vec2());
triad.pb.x = pb.x - midPoint_x;
triad.pb.y = pb.y - midPoint_y;
///triad.pc = b2Vec2.SubVV(pc, midPoint, new b2Vec2());
triad.pc.x = pc.x - midPoint_x;
triad.pc.y = pc.y - midPoint_y;
triad.ka = -b2Vec2.DotVV(dca, dab);
triad.kb = -b2Vec2.DotVV(dab, dbc);
triad.kc = -b2Vec2.DotVV(dbc, dca);
triad.s = b2Vec2.CrossVV(pa, pb) + b2Vec2.CrossVV(pb, pc) + b2Vec2.CrossVV(pc, pa);
}
};
diagram.GetNodes(callback);
///std::stable_sort(m_triadBuffer.Begin(), m_triadBuffer.End(), CompareTriadIndices);
std_stable_sort(this.m_triadBuffer.data, 0, this.m_triadBuffer.count, b2ParticleSystem.CompareTriadIndices);
///m_triadBuffer.Unique(MatchTriadIndices);
this.m_triadBuffer.Unique(b2ParticleSystem.MatchTriadIndices);
}
};
b2ParticleSystem.prototype.UpdatePairsAndTriadsWithReactiveParticles = function () {
var filter = new b2ParticleSystem.ReactiveFilter(this.m_flagsBuffer);
this.UpdatePairsAndTriads(0, this.m_count, filter);
if (!this.m_flagsBuffer.data) {
throw new Error();
}
for (var i = 0; i < this.m_count; i++) {
this.m_flagsBuffer.data[i] &= ~exports.b2ParticleFlag.b2_reactiveParticle;
}
this.m_allParticleFlags &= ~exports.b2ParticleFlag.b2_reactiveParticle;
};
b2ParticleSystem.ComparePairIndices = function (a, b) {
var diffA = a.indexA - b.indexA;
if (diffA !== 0) {
return diffA < 0;
}
return a.indexB < b.indexB;
};
b2ParticleSystem.MatchPairIndices = function (a, b) {
return a.indexA === b.indexA && a.indexB === b.indexB;
};
b2ParticleSystem.CompareTriadIndices = function (a, b) {
var diffA = a.indexA - b.indexA;
if (diffA !== 0) {
return diffA < 0;
}
var diffB = a.indexB - b.indexB;
if (diffB !== 0) {
return diffB < 0;
}
return a.indexC < b.indexC;
};
b2ParticleSystem.MatchTriadIndices = function (a, b) {
return a.indexA === b.indexA && a.indexB === b.indexB && a.indexC === b.indexC;
};
b2ParticleSystem.InitializeParticleLists = function (group, nodeBuffer) {
var bufferIndex = group.GetBufferIndex();
var particleCount = group.GetParticleCount();
for (var i = 0; i < particleCount; i++) {
var node = nodeBuffer[i];
node.list = node;
node.next = null;
node.count = 1;
node.index = i + bufferIndex;
}
};
b2ParticleSystem.prototype.MergeParticleListsInContact = function (group, nodeBuffer) {
var bufferIndex = group.GetBufferIndex();
for (var k = 0; k < this.m_contactBuffer.count; k++) {
/*const b2ParticleContact&*/
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
if (!group.ContainsParticle(a) || !group.ContainsParticle(b)) {
continue;
}
var listA = nodeBuffer[a - bufferIndex].list;
var listB = nodeBuffer[b - bufferIndex].list;
if (listA === listB) {
continue;
}
// To minimize the cost of insertion, make sure listA is longer than
// listB.
if (listA.count < listB.count) {
var _tmp = listA;
listA = listB;
listB = _tmp; ///b2Swap(listA, listB);
}
// DEBUG: b2Assert(listA.count >= listB.count);
b2ParticleSystem.MergeParticleLists(listA, listB);
}
};
b2ParticleSystem.MergeParticleLists = function (listA, listB) {
// Insert listB between index 0 and 1 of listA
// Example:
// listA => a1 => a2 => a3 => null
// listB => b1 => b2 => null
// to
// listA => listB => b1 => b2 => a1 => a2 => a3 => null
// DEBUG: b2Assert(listA !== listB);
for (var b = listB;;) {
b.list = listA;
var nextB = b.next;
if (nextB) {
b = nextB;
}
else {
b.next = listA.next;
break;
}
}
listA.next = listB;
listA.count += listB.count;
listB.count = 0;
};
b2ParticleSystem.FindLongestParticleList = function (group, nodeBuffer) {
var particleCount = group.GetParticleCount();
var result = nodeBuffer[0];
for (var i = 0; i < particleCount; i++) {
var node = nodeBuffer[i];
if (result.count < node.count) {
result = node;
}
}
return result;
};
b2ParticleSystem.prototype.MergeZombieParticleListNodes = function (group, nodeBuffer, survivingList) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
var particleCount = group.GetParticleCount();
for (var i = 0; i < particleCount; i++) {
var node = nodeBuffer[i];
if (node !== survivingList &&
(this.m_flagsBuffer.data[node.index] & exports.b2ParticleFlag.b2_zombieParticle)) {
b2ParticleSystem.MergeParticleListAndNode(survivingList, node);
}
}
};
b2ParticleSystem.MergeParticleListAndNode = function (list, node) {
// Insert node between index 0 and 1 of list
// Example:
// list => a1 => a2 => a3 => null
// node => null
// to
// list => node => a1 => a2 => a3 => null
// DEBUG: b2Assert(node !== list);
// DEBUG: b2Assert(node.list === node);
// DEBUG: b2Assert(node.count === 1);
node.list = list;
node.next = list.next;
list.next = node;
list.count++;
node.count = 0;
};
b2ParticleSystem.prototype.CreateParticleGroupsFromParticleList = function (group, nodeBuffer, survivingList) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
var particleCount = group.GetParticleCount();
var def = new b2ParticleGroupDef();
def.groupFlags = group.GetGroupFlags();
def.userData = group.GetUserData();
for (var i = 0; i < particleCount; i++) {
var list = nodeBuffer[i];
if (!list.count || list === survivingList) {
continue;
}
// DEBUG: b2Assert(list.list === list);
var newGroup = this.CreateParticleGroup(def);
for (var node = list; node; node = node.next) {
var oldIndex = node.index;
// DEBUG: const flags = this.m_flagsBuffer.data[oldIndex];
// DEBUG: b2Assert(!(flags & b2ParticleFlag.b2_zombieParticle));
var newIndex = this.CloneParticle(oldIndex, newGroup);
this.m_flagsBuffer.data[oldIndex] |= exports.b2ParticleFlag.b2_zombieParticle;
node.index = newIndex;
}
}
};
b2ParticleSystem.prototype.UpdatePairsAndTriadsWithParticleList = function (group, nodeBuffer) {
var bufferIndex = group.GetBufferIndex();
// Update indices in pairs and triads. If an index belongs to the group,
// replace it with the corresponding value in nodeBuffer.
// Note that nodeBuffer is allocated only for the group and the index should
// be shifted by bufferIndex.
for (var k = 0; k < this.m_pairBuffer.count; k++) {
var pair = this.m_pairBuffer.data[k];
var a = pair.indexA;
var b = pair.indexB;
if (group.ContainsParticle(a)) {
pair.indexA = nodeBuffer[a - bufferIndex].index;
}
if (group.ContainsParticle(b)) {
pair.indexB = nodeBuffer[b - bufferIndex].index;
}
}
for (var k = 0; k < this.m_triadBuffer.count; k++) {
var triad = this.m_triadBuffer.data[k];
var a = triad.indexA;
var b = triad.indexB;
var c = triad.indexC;
if (group.ContainsParticle(a)) {
triad.indexA = nodeBuffer[a - bufferIndex].index;
}
if (group.ContainsParticle(b)) {
triad.indexB = nodeBuffer[b - bufferIndex].index;
}
if (group.ContainsParticle(c)) {
triad.indexC = nodeBuffer[c - bufferIndex].index;
}
}
};
b2ParticleSystem.prototype.ComputeDepth = function () {
///b2ParticleContact* contactGroups = (b2ParticleContact*) this.m_world.m_stackAllocator.Allocate(sizeof(b2ParticleContact) * this.m_contactBuffer.GetCount());
var contactGroups = []; // TODO: static
var contactGroupsCount = 0;
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
var groupA = this.m_groupBuffer[a];
var groupB = this.m_groupBuffer[b];
if (groupA && groupA === groupB &&
(groupA.m_groupFlags & exports.b2ParticleGroupFlag.b2_particleGroupNeedsUpdateDepth)) {
contactGroups[contactGroupsCount++] = contact;
}
}
///b2ParticleGroup** groupsToUpdate = (b2ParticleGroup**) this.m_world.m_stackAllocator.Allocate(sizeof(b2ParticleGroup*) * this.m_groupCount);
var groupsToUpdate = []; // TODO: static
var groupsToUpdateCount = 0;
for (var group = this.m_groupList; group; group = group.GetNext()) {
if (group.m_groupFlags & exports.b2ParticleGroupFlag.b2_particleGroupNeedsUpdateDepth) {
groupsToUpdate[groupsToUpdateCount++] = group;
this.SetGroupFlags(group, group.m_groupFlags &
~exports.b2ParticleGroupFlag.b2_particleGroupNeedsUpdateDepth);
for (var i = group.m_firstIndex; i < group.m_lastIndex; i++) {
this.m_accumulationBuffer[i] = 0;
}
}
}
// Compute sum of weight of contacts except between different groups.
for (var k = 0; k < contactGroupsCount; k++) {
var contact = contactGroups[k];
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
this.m_accumulationBuffer[a] += w;
this.m_accumulationBuffer[b] += w;
}
// DEBUG: b2Assert(this.m_depthBuffer !== null);
for (var i = 0; i < groupsToUpdateCount; i++) {
var group = groupsToUpdate[i];
for (var i_1 = group.m_firstIndex; i_1 < group.m_lastIndex; i_1++) {
var w = this.m_accumulationBuffer[i_1];
this.m_depthBuffer[i_1] = w < 0.8 ? 0 : b2_maxFloat;
}
}
// The number of iterations is equal to particle number from the deepest
// particle to the nearest surface particle, and in general it is smaller
// than sqrt of total particle number.
///int32 iterationCount = (int32)b2Sqrt((float)m_count);
var iterationCount = b2Sqrt(this.m_count) >> 0;
for (var t = 0; t < iterationCount; t++) {
var updated = false;
for (var k = 0; k < contactGroupsCount; k++) {
var contact = contactGroups[k];
var a = contact.indexA;
var b = contact.indexB;
var r = 1 - contact.weight;
///float32& ap0 = m_depthBuffer[a];
var ap0 = this.m_depthBuffer[a];
///float32& bp0 = m_depthBuffer[b];
var bp0 = this.m_depthBuffer[b];
var ap1 = bp0 + r;
var bp1 = ap0 + r;
if (ap0 > ap1) {
///ap0 = ap1;
this.m_depthBuffer[a] = ap1;
updated = true;
}
if (bp0 > bp1) {
///bp0 = bp1;
this.m_depthBuffer[b] = bp1;
updated = true;
}
}
if (!updated) {
break;
}
}
for (var i = 0; i < groupsToUpdateCount; i++) {
var group = groupsToUpdate[i];
for (var i_2 = group.m_firstIndex; i_2 < group.m_lastIndex; i_2++) {
if (this.m_depthBuffer[i_2] < b2_maxFloat) {
this.m_depthBuffer[i_2] *= this.m_particleDiameter;
}
else {
this.m_depthBuffer[i_2] = 0;
}
}
}
///this.m_world.m_stackAllocator.Free(groupsToUpdate);
///this.m_world.m_stackAllocator.Free(contactGroups);
};
b2ParticleSystem.prototype.GetInsideBoundsEnumerator = function (aabb) {
var lowerTag = b2ParticleSystem.computeTag(this.m_inverseDiameter * aabb.lowerBound.x - 1, this.m_inverseDiameter * aabb.lowerBound.y - 1);
var upperTag = b2ParticleSystem.computeTag(this.m_inverseDiameter * aabb.upperBound.x + 1, this.m_inverseDiameter * aabb.upperBound.y + 1);
///const Proxy* beginProxy = m_proxyBuffer.Begin();
var beginProxy = 0;
///const Proxy* endProxy = m_proxyBuffer.End();
var endProxy = this.m_proxyBuffer.count;
///const Proxy* firstProxy = std::lower_bound(beginProxy, endProxy, lowerTag);
var firstProxy = std_lower_bound(this.m_proxyBuffer.data, beginProxy, endProxy, lowerTag, b2ParticleSystem.Proxy.CompareProxyTag);
///const Proxy* lastProxy = std::upper_bound(firstProxy, endProxy, upperTag);
var lastProxy = std_upper_bound(this.m_proxyBuffer.data, beginProxy, endProxy, upperTag, b2ParticleSystem.Proxy.CompareTagProxy);
// DEBUG: b2Assert(beginProxy <= firstProxy);
// DEBUG: b2Assert(firstProxy <= lastProxy);
// DEBUG: b2Assert(lastProxy <= endProxy);
return new b2ParticleSystem.InsideBoundsEnumerator(this, lowerTag, upperTag, firstProxy, lastProxy);
};
b2ParticleSystem.prototype.UpdateAllParticleFlags = function () {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
this.m_allParticleFlags = 0;
for (var i = 0; i < this.m_count; i++) {
this.m_allParticleFlags |= this.m_flagsBuffer.data[i];
}
this.m_needsUpdateAllParticleFlags = false;
};
b2ParticleSystem.prototype.UpdateAllGroupFlags = function () {
this.m_allGroupFlags = 0;
for (var group = this.m_groupList; group; group = group.GetNext()) {
this.m_allGroupFlags |= group.m_groupFlags;
}
this.m_needsUpdateAllGroupFlags = false;
};
b2ParticleSystem.prototype.AddContact = function (a, b, contacts) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
var s_d = b2ParticleSystem.AddContact_s_d;
var pos_data = this.m_positionBuffer.data;
// DEBUG: b2Assert(contacts === this.m_contactBuffer);
///b2Vec2 d = m_positionBuffer.data[b] - m_positionBuffer.data[a];
var d = b2Vec2.SubVV(pos_data[b], pos_data[a], s_d);
var distBtParticlesSq = b2Vec2.DotVV(d, d);
if (distBtParticlesSq < this.m_squaredDiameter) {
var invD = b2InvSqrt(distBtParticlesSq);
if (!isFinite(invD)) {
invD = 1.98177537e+019;
}
///b2ParticleContact& contact = contacts.Append();
var contact = this.m_contactBuffer.data[this.m_contactBuffer.Append()];
contact.indexA = a;
contact.indexB = b;
contact.flags = this.m_flagsBuffer.data[a] | this.m_flagsBuffer.data[b];
contact.weight = 1 - distBtParticlesSq * invD * this.m_inverseDiameter;
///contact.SetNormal(invD * d);
b2Vec2.MulSV(invD, d, contact.normal);
}
};
b2ParticleSystem.prototype.FindContacts_Reference = function (contacts) {
// DEBUG: b2Assert(contacts === this.m_contactBuffer);
var beginProxy = 0;
var endProxy = this.m_proxyBuffer.count;
this.m_contactBuffer.count = 0;
for (var a = beginProxy, c = beginProxy; a < endProxy; a++) {
var rightTag = b2ParticleSystem.computeRelativeTag(this.m_proxyBuffer.data[a].tag, 1, 0);
for (var b = a + 1; b < endProxy; b++) {
if (rightTag < this.m_proxyBuffer.data[b].tag) {
break;
}
this.AddContact(this.m_proxyBuffer.data[a].index, this.m_proxyBuffer.data[b].index, this.m_contactBuffer);
}
var bottomLeftTag = b2ParticleSystem.computeRelativeTag(this.m_proxyBuffer.data[a].tag, -1, 1);
for (; c < endProxy; c++) {
if (bottomLeftTag <= this.m_proxyBuffer.data[c].tag) {
break;
}
}
var bottomRightTag = b2ParticleSystem.computeRelativeTag(this.m_proxyBuffer.data[a].tag, 1, 1);
for (var b = c; b < endProxy; b++) {
if (bottomRightTag < this.m_proxyBuffer.data[b].tag) {
break;
}
this.AddContact(this.m_proxyBuffer.data[a].index, this.m_proxyBuffer.data[b].index, this.m_contactBuffer);
}
}
};
///void ReorderForFindContact(FindContactInput* reordered, int alignedCount) const;
///void GatherChecksOneParticle(const uint32 bound, const int startIndex, const int particleIndex, int* nextUncheckedIndex, b2GrowableBuffer<FindContactCheck>& checks) const;
///void GatherChecks(b2GrowableBuffer<FindContactCheck>& checks) const;
///void FindContacts_Simd(b2GrowableBuffer<b2ParticleContact>& contacts) const;
b2ParticleSystem.prototype.FindContacts = function (contacts) {
this.FindContacts_Reference(contacts);
};
///static void UpdateProxyTags(const uint32* const tags, b2GrowableBuffer<Proxy>& proxies);
///static bool ProxyBufferHasIndex(int32 index, const Proxy* const a, int count);
///static int NumProxiesWithSameTag(const Proxy* const a, const Proxy* const b, int count);
///static bool AreProxyBuffersTheSame(const b2GrowableBuffer<Proxy>& a, const b2GrowableBuffer<Proxy>& b);
b2ParticleSystem.prototype.UpdateProxies_Reference = function (proxies) {
// DEBUG: b2Assert(proxies === this.m_proxyBuffer);
if (!this.m_positionBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var inv_diam = this.m_inverseDiameter;
for (var k = 0; k < this.m_proxyBuffer.count; ++k) {
var proxy = this.m_proxyBuffer.data[k];
var i = proxy.index;
var p = pos_data[i];
proxy.tag = b2ParticleSystem.computeTag(inv_diam * p.x, inv_diam * p.y);
}
};
///void UpdateProxies_Simd(b2GrowableBuffer<Proxy>& proxies) const;
b2ParticleSystem.prototype.UpdateProxies = function (proxies) {
this.UpdateProxies_Reference(proxies);
};
b2ParticleSystem.prototype.SortProxies = function (proxies) {
// DEBUG: b2Assert(proxies === this.m_proxyBuffer);
///std::sort(proxies.Begin(), proxies.End());
std_sort(this.m_proxyBuffer.data, 0, this.m_proxyBuffer.count, b2ParticleSystem.Proxy.CompareProxyProxy);
};
b2ParticleSystem.prototype.FilterContacts = function (contacts) {
// Optionally filter the contact.
var contactFilter = this.GetParticleContactFilter();
if (contactFilter === null) {
return;
}
/// contacts.RemoveIf(b2ParticleContactRemovePredicate(this, contactFilter));
// DEBUG: b2Assert(contacts === this.m_contactBuffer);
var system = this;
var predicate = function (contact) {
return ((contact.flags & exports.b2ParticleFlag.b2_particleContactFilterParticle) !== 0) && !contactFilter.ShouldCollideParticleParticle(system, contact.indexA, contact.indexB);
};
this.m_contactBuffer.RemoveIf(predicate);
};
b2ParticleSystem.prototype.NotifyContactListenerPreContact = function (particlePairs) {
var contactListener = this.GetParticleContactListener();
if (contactListener === null) {
return;
}
///particlePairs.Initialize(m_contactBuffer.Begin(), m_contactBuffer.GetCount(), GetFlagsBuffer());
particlePairs.Initialize(this.m_contactBuffer, this.m_flagsBuffer);
throw new Error(); // TODO: notify
};
b2ParticleSystem.prototype.NotifyContactListenerPostContact = function (particlePairs) {
var contactListener = this.GetParticleContactListener();
if (contactListener === null) {
return;
}
// Loop through all new contacts, reporting any new ones, and
// "invalidating" the ones that still exist.
///const b2ParticleContact* const endContact = m_contactBuffer.End();
///for (b2ParticleContact* contact = m_contactBuffer.Begin(); contact < endContact; ++contact)
for (var k = 0; k < this.m_contactBuffer.count; ++k) {
var contact = this.m_contactBuffer.data[k];
{
// Just started touching, inform the listener.
contactListener.BeginContactParticleParticle(this, contact);
}
}
// Report particles that are no longer touching.
// That is, any pairs that were not invalidated above.
///const int32 pairCount = particlePairs.GetCount();
///const ParticlePair* const pairs = particlePairs.GetBuffer();
///const int8* const valid = particlePairs.GetValidBuffer();
///for (int32 i = 0; i < pairCount; ++i)
///{
/// if (valid[i])
/// {
/// contactListener.EndContactParticleParticle(this, pairs[i].first, pairs[i].second);
/// }
///}
throw new Error(); // TODO: notify
};
b2ParticleSystem.b2ParticleContactIsZombie = function (contact) {
return (contact.flags & exports.b2ParticleFlag.b2_zombieParticle) === exports.b2ParticleFlag.b2_zombieParticle;
};
b2ParticleSystem.prototype.UpdateContacts = function (exceptZombie) {
this.UpdateProxies(this.m_proxyBuffer);
this.SortProxies(this.m_proxyBuffer);
///b2ParticlePairSet particlePairs(&this.m_world.m_stackAllocator);
var particlePairs = new b2ParticleSystem.b2ParticlePairSet(); // TODO: static
this.NotifyContactListenerPreContact(particlePairs);
this.FindContacts(this.m_contactBuffer);
this.FilterContacts(this.m_contactBuffer);
this.NotifyContactListenerPostContact(particlePairs);
if (exceptZombie) {
this.m_contactBuffer.RemoveIf(b2ParticleSystem.b2ParticleContactIsZombie);
}
};
b2ParticleSystem.prototype.NotifyBodyContactListenerPreContact = function (fixtureSet) {
var contactListener = this.GetFixtureContactListener();
if (contactListener === null) {
return;
}
///fixtureSet.Initialize(m_bodyContactBuffer.Begin(), m_bodyContactBuffer.GetCount(), GetFlagsBuffer());
fixtureSet.Initialize(this.m_bodyContactBuffer, this.m_flagsBuffer);
throw new Error(); // TODO: notify
};
b2ParticleSystem.prototype.NotifyBodyContactListenerPostContact = function (fixtureSet) {
var contactListener = this.GetFixtureContactListener();
if (contactListener === null) {
return;
}
// Loop through all new contacts, reporting any new ones, and
// "invalidating" the ones that still exist.
///for (b2ParticleBodyContact* contact = m_bodyContactBuffer.Begin(); contact !== m_bodyContactBuffer.End(); ++contact)
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
{
// Just started touching, report it!
contactListener.BeginContactFixtureParticle(this, contact);
}
}
// If the contact listener is enabled, report all fixtures that are no
// longer in contact with particles.
///const FixtureParticle* const fixtureParticles = fixtureSet.GetBuffer();
///const int8* const fixtureParticlesValid = fixtureSet.GetValidBuffer();
///const int32 fixtureParticleCount = fixtureSet.GetCount();
///for (int32 i = 0; i < fixtureParticleCount; ++i)
///{
/// if (fixtureParticlesValid[i])
/// {
/// const FixtureParticle* const fixtureParticle = &fixtureParticles[i];
/// contactListener.EndContactFixtureParticle(fixtureParticle.first, this, fixtureParticle.second);
/// }
///}
throw new Error(); // TODO: notify
};
b2ParticleSystem.prototype.UpdateBodyContacts = function () {
var s_aabb = b2ParticleSystem.UpdateBodyContacts_s_aabb;
// If the particle contact listener is enabled, generate a set of
// fixture / particle contacts.
///FixtureParticleSet fixtureSet(&m_world.m_stackAllocator);
var fixtureSet = new b2ParticleSystem.FixtureParticleSet(); // TODO: static
this.NotifyBodyContactListenerPreContact(fixtureSet);
if (this.m_stuckThreshold > 0) {
if (!this.m_bodyContactCountBuffer.data) {
throw new Error();
}
if (!this.m_lastBodyContactStepBuffer.data) {
throw new Error();
}
if (!this.m_consecutiveContactStepsBuffer.data) {
throw new Error();
}
var particleCount = this.GetParticleCount();
for (var i = 0; i < particleCount; i++) {
// Detect stuck particles, see comment in
// b2ParticleSystem::DetectStuckParticle()
this.m_bodyContactCountBuffer.data[i] = 0;
if (this.m_timestamp > (this.m_lastBodyContactStepBuffer.data[i] + 1)) {
this.m_consecutiveContactStepsBuffer.data[i] = 0;
}
}
}
this.m_bodyContactBuffer.SetCount(0);
this.m_stuckParticleBuffer.SetCount(0);
var aabb = s_aabb;
this.ComputeAABB(aabb);
var callback = new b2ParticleSystem.UpdateBodyContactsCallback(this, this.GetFixtureContactFilter());
this.m_world.QueryAABB(callback, aabb);
if (this.m_def.strictContactCheck) {
this.RemoveSpuriousBodyContacts();
}
this.NotifyBodyContactListenerPostContact(fixtureSet);
};
b2ParticleSystem.prototype.Solve = function (step) {
var s_subStep = b2ParticleSystem.Solve_s_subStep;
if (this.m_count === 0) {
return;
}
// If particle lifetimes are enabled, destroy particles that are too old.
if (this.m_expirationTimeBuffer.data) {
this.SolveLifetimes(step);
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_zombieParticle) {
this.SolveZombie();
}
if (this.m_needsUpdateAllParticleFlags) {
this.UpdateAllParticleFlags();
}
if (this.m_needsUpdateAllGroupFlags) {
this.UpdateAllGroupFlags();
}
if (this.m_paused) {
return;
}
for (this.m_iterationIndex = 0; this.m_iterationIndex < step.particleIterations; this.m_iterationIndex++) {
++this.m_timestamp;
var subStep = s_subStep.Copy(step);
subStep.dt /= step.particleIterations;
subStep.inv_dt *= step.particleIterations;
this.UpdateContacts(false);
this.UpdateBodyContacts();
this.ComputeWeight();
if (this.m_allGroupFlags & exports.b2ParticleGroupFlag.b2_particleGroupNeedsUpdateDepth) {
this.ComputeDepth();
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_reactiveParticle) {
this.UpdatePairsAndTriadsWithReactiveParticles();
}
if (this.m_hasForce) {
this.SolveForce(subStep);
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_viscousParticle) {
this.SolveViscous();
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_repulsiveParticle) {
this.SolveRepulsive(subStep);
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_powderParticle) {
this.SolvePowder(subStep);
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_tensileParticle) {
this.SolveTensile(subStep);
}
if (this.m_allGroupFlags & exports.b2ParticleGroupFlag.b2_solidParticleGroup) {
this.SolveSolid(subStep);
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_colorMixingParticle) {
this.SolveColorMixing();
}
this.SolveGravity(subStep);
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_staticPressureParticle) {
this.SolveStaticPressure(subStep);
}
this.SolvePressure(subStep);
this.SolveDamping(subStep);
if (this.m_allParticleFlags & b2ParticleSystem.k_extraDampingFlags) {
this.SolveExtraDamping();
}
// SolveElastic and SolveSpring refer the current velocities for
// numerical stability, they should be called as late as possible.
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_elasticParticle) {
this.SolveElastic(subStep);
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_springParticle) {
this.SolveSpring(subStep);
}
this.LimitVelocity(subStep);
if (this.m_allGroupFlags & exports.b2ParticleGroupFlag.b2_rigidParticleGroup) {
this.SolveRigidDamping();
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_barrierParticle) {
this.SolveBarrier(subStep);
}
// SolveCollision, SolveRigid and SolveWall should be called after
// other force functions because they may require particles to have
// specific velocities.
this.SolveCollision(subStep);
if (this.m_allGroupFlags & exports.b2ParticleGroupFlag.b2_rigidParticleGroup) {
this.SolveRigid(subStep);
}
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_wallParticle) {
this.SolveWall();
}
// The particle positions can be updated only at the end of substep.
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
for (var i = 0; i < this.m_count; i++) {
///m_positionBuffer.data[i] += subStep.dt * m_velocityBuffer.data[i];
this.m_positionBuffer.data[i].SelfMulAdd(subStep.dt, this.m_velocityBuffer.data[i]);
}
}
};
b2ParticleSystem.prototype.SolveCollision = function (step) {
var s_aabb = b2ParticleSystem.SolveCollision_s_aabb;
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
// This function detects particles which are crossing boundary of bodies
// and modifies velocities of them so that they will move just in front of
// boundary. This function function also applies the reaction force to
// bodies as precisely as the numerical stability is kept.
var aabb = s_aabb;
aabb.lowerBound.x = +b2_maxFloat;
aabb.lowerBound.y = +b2_maxFloat;
aabb.upperBound.x = -b2_maxFloat;
aabb.upperBound.y = -b2_maxFloat;
for (var i = 0; i < this.m_count; i++) {
var v = vel_data[i];
var p1 = pos_data[i];
///let p2 = p1 + step.dt * v;
var p2_x = p1.x + step.dt * v.x;
var p2_y = p1.y + step.dt * v.y;
///aabb.lowerBound = b2Min(aabb.lowerBound, b2Min(p1, p2));
aabb.lowerBound.x = b2Min(aabb.lowerBound.x, b2Min(p1.x, p2_x));
aabb.lowerBound.y = b2Min(aabb.lowerBound.y, b2Min(p1.y, p2_y));
///aabb.upperBound = b2Max(aabb.upperBound, b2Max(p1, p2));
aabb.upperBound.x = b2Max(aabb.upperBound.x, b2Max(p1.x, p2_x));
aabb.upperBound.y = b2Max(aabb.upperBound.y, b2Max(p1.y, p2_y));
}
var callback = new b2ParticleSystem.SolveCollisionCallback(this, step);
this.m_world.QueryAABB(callback, aabb);
};
b2ParticleSystem.prototype.LimitVelocity = function (step) {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
var criticalVelocitySquared = this.GetCriticalVelocitySquared(step);
for (var i = 0; i < this.m_count; i++) {
var v = vel_data[i];
var v2 = b2Vec2.DotVV(v, v);
if (v2 > criticalVelocitySquared) {
///v *= b2Sqrt(criticalVelocitySquared / v2);
v.SelfMul(b2Sqrt(criticalVelocitySquared / v2));
}
}
};
b2ParticleSystem.prototype.SolveGravity = function (step) {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var s_gravity = b2ParticleSystem.SolveGravity_s_gravity;
var vel_data = this.m_velocityBuffer.data;
///b2Vec2 gravity = step.dt * m_def.gravityScale * m_world.GetGravity();
var gravity = b2Vec2.MulSV(step.dt * this.m_def.gravityScale, this.m_world.GetGravity(), s_gravity);
for (var i = 0; i < this.m_count; i++) {
vel_data[i].SelfAdd(gravity);
}
};
b2ParticleSystem.prototype.SolveBarrier = function (step) {
var s_aabb = b2ParticleSystem.SolveBarrier_s_aabb;
var s_va = b2ParticleSystem.SolveBarrier_s_va;
var s_vb = b2ParticleSystem.SolveBarrier_s_vb;
var s_pba = b2ParticleSystem.SolveBarrier_s_pba;
var s_vba = b2ParticleSystem.SolveBarrier_s_vba;
var s_vc = b2ParticleSystem.SolveBarrier_s_vc;
var s_pca = b2ParticleSystem.SolveBarrier_s_pca;
var s_vca = b2ParticleSystem.SolveBarrier_s_vca;
var s_qba = b2ParticleSystem.SolveBarrier_s_qba;
var s_qca = b2ParticleSystem.SolveBarrier_s_qca;
var s_dv = b2ParticleSystem.SolveBarrier_s_dv;
var s_f = b2ParticleSystem.SolveBarrier_s_f;
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
// If a particle is passing between paired barrier particles,
// its velocity will be decelerated to avoid passing.
for (var i = 0; i < this.m_count; i++) {
var flags = this.m_flagsBuffer.data[i];
///if ((flags & b2ParticleSystem.k_barrierWallFlags) === b2ParticleSystem.k_barrierWallFlags)
if ((flags & b2ParticleSystem.k_barrierWallFlags) !== 0) {
vel_data[i].SetZero();
}
}
var tmax = b2_barrierCollisionTime * step.dt;
var mass = this.GetParticleMass();
for (var k = 0; k < this.m_pairBuffer.count; k++) {
var pair = this.m_pairBuffer.data[k];
if (pair.flags & exports.b2ParticleFlag.b2_barrierParticle) {
var a = pair.indexA;
var b = pair.indexB;
var pa = pos_data[a];
var pb = pos_data[b];
/// b2AABB aabb;
var aabb = s_aabb;
///aabb.lowerBound = b2Min(pa, pb);
b2Vec2.MinV(pa, pb, aabb.lowerBound);
///aabb.upperBound = b2Max(pa, pb);
b2Vec2.MaxV(pa, pb, aabb.upperBound);
var aGroup = this.m_groupBuffer[a];
var bGroup = this.m_groupBuffer[b];
///b2Vec2 va = GetLinearVelocity(aGroup, a, pa);
var va = this.GetLinearVelocity(aGroup, a, pa, s_va);
///b2Vec2 vb = GetLinearVelocity(bGroup, b, pb);
var vb = this.GetLinearVelocity(bGroup, b, pb, s_vb);
///b2Vec2 pba = pb - pa;
var pba = b2Vec2.SubVV(pb, pa, s_pba);
///b2Vec2 vba = vb - va;
var vba = b2Vec2.SubVV(vb, va, s_vba);
///InsideBoundsEnumerator enumerator = GetInsideBoundsEnumerator(aabb);
var enumerator = this.GetInsideBoundsEnumerator(aabb);
var c = void 0;
while ((c = enumerator.GetNext()) >= 0) {
var pc = pos_data[c];
var cGroup = this.m_groupBuffer[c];
if (aGroup !== cGroup && bGroup !== cGroup) {
///b2Vec2 vc = GetLinearVelocity(cGroup, c, pc);
var vc = this.GetLinearVelocity(cGroup, c, pc, s_vc);
// Solve the equation below:
// (1-s)*(pa+t*va)+s*(pb+t*vb) = pc+t*vc
// which expresses that the particle c will pass a line
// connecting the particles a and b at the time of t.
// if s is between 0 and 1, c will pass between a and b.
///b2Vec2 pca = pc - pa;
var pca = b2Vec2.SubVV(pc, pa, s_pca);
///b2Vec2 vca = vc - va;
var vca = b2Vec2.SubVV(vc, va, s_vca);
var e2 = b2Vec2.CrossVV(vba, vca);
var e1 = b2Vec2.CrossVV(pba, vca) - b2Vec2.CrossVV(pca, vba);
var e0 = b2Vec2.CrossVV(pba, pca);
var s = void 0, t = void 0;
///b2Vec2 qba, qca;
var qba = s_qba, qca = s_qca;
if (e2 === 0) {
if (e1 === 0) {
continue;
}
t = -e0 / e1;
if (!(t >= 0 && t < tmax)) {
continue;
}
///qba = pba + t * vba;
b2Vec2.AddVMulSV(pba, t, vba, qba);
///qca = pca + t * vca;
b2Vec2.AddVMulSV(pca, t, vca, qca);
s = b2Vec2.DotVV(qba, qca) / b2Vec2.DotVV(qba, qba);
if (!(s >= 0 && s <= 1)) {
continue;
}
}
else {
var det = e1 * e1 - 4 * e0 * e2;
if (det < 0) {
continue;
}
var sqrtDet = b2Sqrt(det);
var t1 = (-e1 - sqrtDet) / (2 * e2);
var t2 = (-e1 + sqrtDet) / (2 * e2);
///if (t1 > t2) b2Swap(t1, t2);
if (t1 > t2) {
var tmp = t1;
t1 = t2;
t2 = tmp;
}
t = t1;
///qba = pba + t * vba;
b2Vec2.AddVMulSV(pba, t, vba, qba);
///qca = pca + t * vca;
b2Vec2.AddVMulSV(pca, t, vca, qca);
///s = b2Dot(qba, qca) / b2Dot(qba, qba);
s = b2Vec2.DotVV(qba, qca) / b2Vec2.DotVV(qba, qba);
if (!(t >= 0 && t < tmax && s >= 0 && s <= 1)) {
t = t2;
if (!(t >= 0 && t < tmax)) {
continue;
}
///qba = pba + t * vba;
b2Vec2.AddVMulSV(pba, t, vba, qba);
///qca = pca + t * vca;
b2Vec2.AddVMulSV(pca, t, vca, qca);
///s = b2Dot(qba, qca) / b2Dot(qba, qba);
s = b2Vec2.DotVV(qba, qca) / b2Vec2.DotVV(qba, qba);
if (!(s >= 0 && s <= 1)) {
continue;
}
}
}
// Apply a force to particle c so that it will have the
// interpolated velocity at the collision point on line ab.
///b2Vec2 dv = va + s * vba - vc;
var dv = s_dv;
dv.x = va.x + s * vba.x - vc.x;
dv.y = va.y + s * vba.y - vc.y;
///b2Vec2 f = GetParticleMass() * dv;
var f = b2Vec2.MulSV(mass, dv, s_f);
if (cGroup && this.IsRigidGroup(cGroup)) {
// If c belongs to a rigid group, the force will be
// distributed in the group.
var mass_1 = cGroup.GetMass();
var inertia = cGroup.GetInertia();
if (mass_1 > 0) {
///cGroup.m_linearVelocity += 1 / mass * f;
cGroup.m_linearVelocity.SelfMulAdd(1 / mass_1, f);
}
if (inertia > 0) {
///cGroup.m_angularVelocity += b2Cross(pc - cGroup.GetCenter(), f) / inertia;
cGroup.m_angularVelocity += b2Vec2.CrossVV(b2Vec2.SubVV(pc, cGroup.GetCenter(), b2Vec2.s_t0), f) / inertia;
}
}
else {
///m_velocityBuffer.data[c] += dv;
vel_data[c].SelfAdd(dv);
}
// Apply a reversed force to particle c after particle
// movement so that momentum will be preserved.
///ParticleApplyForce(c, -step.inv_dt * f);
this.ParticleApplyForce(c, f.SelfMul(-step.inv_dt));
}
}
}
}
};
b2ParticleSystem.prototype.SolveStaticPressure = function (step) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
this.m_staticPressureBuffer = this.RequestBuffer(this.m_staticPressureBuffer);
var criticalPressure = this.GetCriticalPressure(step);
var pressurePerWeight = this.m_def.staticPressureStrength * criticalPressure;
var maxPressure = b2_maxParticlePressure * criticalPressure;
var relaxation = this.m_def.staticPressureRelaxation;
/// Compute pressure satisfying the modified Poisson equation:
/// Sum_for_j((p_i - p_j) * w_ij) + relaxation * p_i =
/// pressurePerWeight * (w_i - b2_minParticleWeight)
/// by iterating the calculation:
/// p_i = (Sum_for_j(p_j * w_ij) + pressurePerWeight *
/// (w_i - b2_minParticleWeight)) / (w_i + relaxation)
/// where
/// p_i and p_j are static pressure of particle i and j
/// w_ij is contact weight between particle i and j
/// w_i is sum of contact weight of particle i
for (var t = 0; t < this.m_def.staticPressureIterations; t++) {
///memset(m_accumulationBuffer, 0, sizeof(*m_accumulationBuffer) * m_count);
for (var i = 0; i < this.m_count; i++) {
this.m_accumulationBuffer[i] = 0;
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
if (contact.flags & exports.b2ParticleFlag.b2_staticPressureParticle) {
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
this.m_accumulationBuffer[a] += w * this.m_staticPressureBuffer[b]; // a <- b
this.m_accumulationBuffer[b] += w * this.m_staticPressureBuffer[a]; // b <- a
}
}
for (var i = 0; i < this.m_count; i++) {
var w = this.m_weightBuffer[i];
if (this.m_flagsBuffer.data[i] & exports.b2ParticleFlag.b2_staticPressureParticle) {
var wh = this.m_accumulationBuffer[i];
var h = (wh + pressurePerWeight * (w - b2_minParticleWeight)) /
(w + relaxation);
this.m_staticPressureBuffer[i] = b2Clamp(h, 0.0, maxPressure);
}
else {
this.m_staticPressureBuffer[i] = 0;
}
}
}
};
b2ParticleSystem.prototype.ComputeWeight = function () {
// calculates the sum of contact-weights for each particle
// that means dimensionless density
///memset(m_weightBuffer, 0, sizeof(*m_weightBuffer) * m_count);
for (var k = 0; k < this.m_count; k++) {
this.m_weightBuffer[k] = 0;
}
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
var a = contact.index;
var w = contact.weight;
this.m_weightBuffer[a] += w;
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
this.m_weightBuffer[a] += w;
this.m_weightBuffer[b] += w;
}
};
b2ParticleSystem.prototype.SolvePressure = function (step) {
var s_f = b2ParticleSystem.SolvePressure_s_f;
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
// calculates pressure as a linear function of density
var criticalPressure = this.GetCriticalPressure(step);
var pressurePerWeight = this.m_def.pressureStrength * criticalPressure;
var maxPressure = b2_maxParticlePressure * criticalPressure;
for (var i = 0; i < this.m_count; i++) {
var w = this.m_weightBuffer[i];
var h = pressurePerWeight * b2Max(0.0, w - b2_minParticleWeight);
this.m_accumulationBuffer[i] = b2Min(h, maxPressure);
}
// ignores particles which have their own repulsive force
if (this.m_allParticleFlags & b2ParticleSystem.k_noPressureFlags) {
for (var i = 0; i < this.m_count; i++) {
if (this.m_flagsBuffer.data[i] & b2ParticleSystem.k_noPressureFlags) {
this.m_accumulationBuffer[i] = 0;
}
}
}
// static pressure
if (this.m_allParticleFlags & exports.b2ParticleFlag.b2_staticPressureParticle) {
// DEBUG: b2Assert(this.m_staticPressureBuffer !== null);
for (var i = 0; i < this.m_count; i++) {
if (this.m_flagsBuffer.data[i] & exports.b2ParticleFlag.b2_staticPressureParticle) {
this.m_accumulationBuffer[i] += this.m_staticPressureBuffer[i];
}
}
}
// applies pressure between each particles in contact
var velocityPerPressure = step.dt / (this.m_def.density * this.m_particleDiameter);
var inv_mass = this.GetParticleInvMass();
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
var a = contact.index;
var b = contact.body;
var w = contact.weight;
var m = contact.mass;
var n = contact.normal;
var p = pos_data[a];
var h = this.m_accumulationBuffer[a] + pressurePerWeight * w;
///b2Vec2 f = velocityPerPressure * w * m * h * n;
var f = b2Vec2.MulSV(velocityPerPressure * w * m * h, n, s_f);
///m_velocityBuffer.data[a] -= GetParticleInvMass() * f;
vel_data[a].SelfMulSub(inv_mass, f);
b.ApplyLinearImpulse(f, p, true);
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
var n = contact.normal;
var h = this.m_accumulationBuffer[a] + this.m_accumulationBuffer[b];
///b2Vec2 f = velocityPerPressure * w * h * n;
var f = b2Vec2.MulSV(velocityPerPressure * w * h, n, s_f);
///m_velocityBuffer.data[a] -= f;
vel_data[a].SelfSub(f);
///m_velocityBuffer.data[b] += f;
vel_data[b].SelfAdd(f);
}
};
b2ParticleSystem.prototype.SolveDamping = function (step) {
var s_v = b2ParticleSystem.SolveDamping_s_v;
var s_f = b2ParticleSystem.SolveDamping_s_f;
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
// reduces normal velocity of each contact
var linearDamping = this.m_def.dampingStrength;
var quadraticDamping = 1 / this.GetCriticalVelocity(step);
var inv_mass = this.GetParticleInvMass();
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
var a = contact.index;
var b = contact.body;
var w = contact.weight;
var m = contact.mass;
var n = contact.normal;
var p = pos_data[a];
///b2Vec2 v = b.GetLinearVelocityFromWorldPoint(p) - m_velocityBuffer.data[a];
var v = b2Vec2.SubVV(b.GetLinearVelocityFromWorldPoint(p, b2Vec2.s_t0), vel_data[a], s_v);
var vn = b2Vec2.DotVV(v, n);
if (vn < 0) {
var damping = b2Max(linearDamping * w, b2Min(-quadraticDamping * vn, 0.5));
///b2Vec2 f = damping * m * vn * n;
var f = b2Vec2.MulSV(damping * m * vn, n, s_f);
///m_velocityBuffer.data[a] += GetParticleInvMass() * f;
vel_data[a].SelfMulAdd(inv_mass, f);
///b.ApplyLinearImpulse(-f, p, true);
b.ApplyLinearImpulse(f.SelfNeg(), p, true);
}
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
var n = contact.normal;
///b2Vec2 v = m_velocityBuffer.data[b] - m_velocityBuffer.data[a];
var v = b2Vec2.SubVV(vel_data[b], vel_data[a], s_v);
var vn = b2Vec2.DotVV(v, n);
if (vn < 0) {
///float32 damping = b2Max(linearDamping * w, b2Min(- quadraticDamping * vn, 0.5f));
var damping = b2Max(linearDamping * w, b2Min(-quadraticDamping * vn, 0.5));
///b2Vec2 f = damping * vn * n;
var f = b2Vec2.MulSV(damping * vn, n, s_f);
///this.m_velocityBuffer.data[a] += f;
vel_data[a].SelfAdd(f);
///this.m_velocityBuffer.data[b] -= f;
vel_data[b].SelfSub(f);
}
}
};
b2ParticleSystem.prototype.SolveRigidDamping = function () {
var s_t0 = b2ParticleSystem.SolveRigidDamping_s_t0;
var s_t1 = b2ParticleSystem.SolveRigidDamping_s_t1;
var s_p = b2ParticleSystem.SolveRigidDamping_s_p;
var s_v = b2ParticleSystem.SolveRigidDamping_s_v;
var invMassA = [0.0], invInertiaA = [0.0], tangentDistanceA = [0.0]; // TODO: static
var invMassB = [0.0], invInertiaB = [0.0], tangentDistanceB = [0.0]; // TODO: static
// Apply impulse to rigid particle groups colliding with other objects
// to reduce relative velocity at the colliding point.
if (!this.m_positionBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var damping = this.m_def.dampingStrength;
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
var a = contact.index;
var aGroup = this.m_groupBuffer[a];
if (aGroup && this.IsRigidGroup(aGroup)) {
var b = contact.body;
var n = contact.normal;
var w = contact.weight;
var p = pos_data[a];
///b2Vec2 v = b.GetLinearVelocityFromWorldPoint(p) - aGroup.GetLinearVelocityFromWorldPoint(p);
var v = b2Vec2.SubVV(b.GetLinearVelocityFromWorldPoint(p, s_t0), aGroup.GetLinearVelocityFromWorldPoint(p, s_t1), s_v);
var vn = b2Vec2.DotVV(v, n);
if (vn < 0) {
// The group's average velocity at particle position 'p' is pushing
// the particle into the body.
///this.InitDampingParameterWithRigidGroupOrParticle(&invMassA, &invInertiaA, &tangentDistanceA, true, aGroup, a, p, n);
this.InitDampingParameterWithRigidGroupOrParticle(invMassA, invInertiaA, tangentDistanceA, true, aGroup, a, p, n);
// Calculate b.m_I from public functions of b2Body.
///this.InitDampingParameter(&invMassB, &invInertiaB, &tangentDistanceB, b.GetMass(), b.GetInertia() - b.GetMass() * b.GetLocalCenter().LengthSquared(), b.GetWorldCenter(), p, n);
this.InitDampingParameter(invMassB, invInertiaB, tangentDistanceB, b.GetMass(), b.GetInertia() - b.GetMass() * b.GetLocalCenter().LengthSquared(), b.GetWorldCenter(), p, n);
///float32 f = damping * b2Min(w, 1.0) * this.ComputeDampingImpulse(invMassA, invInertiaA, tangentDistanceA, invMassB, invInertiaB, tangentDistanceB, vn);
var f = damping * b2Min(w, 1.0) * this.ComputeDampingImpulse(invMassA[0], invInertiaA[0], tangentDistanceA[0], invMassB[0], invInertiaB[0], tangentDistanceB[0], vn);
///this.ApplyDamping(invMassA, invInertiaA, tangentDistanceA, true, aGroup, a, f, n);
this.ApplyDamping(invMassA[0], invInertiaA[0], tangentDistanceA[0], true, aGroup, a, f, n);
///b.ApplyLinearImpulse(-f * n, p, true);
b.ApplyLinearImpulse(b2Vec2.MulSV(-f, n, b2Vec2.s_t0), p, true);
}
}
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
var n = contact.normal;
var w = contact.weight;
var aGroup = this.m_groupBuffer[a];
var bGroup = this.m_groupBuffer[b];
var aRigid = this.IsRigidGroup(aGroup);
var bRigid = this.IsRigidGroup(bGroup);
if (aGroup !== bGroup && (aRigid || bRigid)) {
///b2Vec2 p = 0.5f * (this.m_positionBuffer.data[a] + this.m_positionBuffer.data[b]);
var p = b2Vec2.MidVV(pos_data[a], pos_data[b], s_p);
///b2Vec2 v = GetLinearVelocity(bGroup, b, p) - GetLinearVelocity(aGroup, a, p);
var v = b2Vec2.SubVV(this.GetLinearVelocity(bGroup, b, p, s_t0), this.GetLinearVelocity(aGroup, a, p, s_t1), s_v);
var vn = b2Vec2.DotVV(v, n);
if (vn < 0) {
///this.InitDampingParameterWithRigidGroupOrParticle(&invMassA, &invInertiaA, &tangentDistanceA, aRigid, aGroup, a, p, n);
this.InitDampingParameterWithRigidGroupOrParticle(invMassA, invInertiaA, tangentDistanceA, aRigid, aGroup, a, p, n);
///this.InitDampingParameterWithRigidGroupOrParticle(&invMassB, &invInertiaB, &tangentDistanceB, bRigid, bGroup, b, p, n);
this.InitDampingParameterWithRigidGroupOrParticle(invMassB, invInertiaB, tangentDistanceB, bRigid, bGroup, b, p, n);
///float32 f = damping * w * this.ComputeDampingImpulse(invMassA, invInertiaA, tangentDistanceA, invMassB, invInertiaB, tangentDistanceB, vn);
var f = damping * w * this.ComputeDampingImpulse(invMassA[0], invInertiaA[0], tangentDistanceA[0], invMassB[0], invInertiaB[0], tangentDistanceB[0], vn);
///this.ApplyDamping(invMassA, invInertiaA, tangentDistanceA, aRigid, aGroup, a, f, n);
this.ApplyDamping(invMassA[0], invInertiaA[0], tangentDistanceA[0], aRigid, aGroup, a, f, n);
///this.ApplyDamping(invMassB, invInertiaB, tangentDistanceB, bRigid, bGroup, b, -f, n);
this.ApplyDamping(invMassB[0], invInertiaB[0], tangentDistanceB[0], bRigid, bGroup, b, -f, n);
}
}
}
};
b2ParticleSystem.prototype.SolveExtraDamping = function () {
var s_v = b2ParticleSystem.SolveExtraDamping_s_v;
var s_f = b2ParticleSystem.SolveExtraDamping_s_f;
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
// Applies additional damping force between bodies and particles which can
// produce strong repulsive force. Applying damping force multiple times
// is effective in suppressing vibration.
var pos_data = this.m_positionBuffer.data;
var inv_mass = this.GetParticleInvMass();
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
var a = contact.index;
if (this.m_flagsBuffer.data[a] & b2ParticleSystem.k_extraDampingFlags) {
var b = contact.body;
var m = contact.mass;
var n = contact.normal;
var p = pos_data[a];
///b2Vec2 v = b.GetLinearVelocityFromWorldPoint(p) - m_velocityBuffer.data[a];
var v = b2Vec2.SubVV(b.GetLinearVelocityFromWorldPoint(p, b2Vec2.s_t0), vel_data[a], s_v);
///float32 vn = b2Dot(v, n);
var vn = b2Vec2.DotVV(v, n);
if (vn < 0) {
///b2Vec2 f = 0.5f * m * vn * n;
var f = b2Vec2.MulSV(0.5 * m * vn, n, s_f);
///m_velocityBuffer.data[a] += GetParticleInvMass() * f;
vel_data[a].SelfMulAdd(inv_mass, f);
///b.ApplyLinearImpulse(-f, p, true);
b.ApplyLinearImpulse(f.SelfNeg(), p, true);
}
}
}
};
b2ParticleSystem.prototype.SolveWall = function () {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
for (var i = 0; i < this.m_count; i++) {
if (this.m_flagsBuffer.data[i] & exports.b2ParticleFlag.b2_wallParticle) {
vel_data[i].SetZero();
}
}
};
b2ParticleSystem.prototype.SolveRigid = function (step) {
var s_position = b2ParticleSystem.SolveRigid_s_position;
var s_rotation = b2ParticleSystem.SolveRigid_s_rotation;
var s_transform = b2ParticleSystem.SolveRigid_s_transform;
var s_velocityTransform = b2ParticleSystem.SolveRigid_s_velocityTransform;
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
for (var group = this.m_groupList; group; group = group.GetNext()) {
if (group.m_groupFlags & exports.b2ParticleGroupFlag.b2_rigidParticleGroup) {
group.UpdateStatistics();
///b2Rot rotation(step.dt * group.m_angularVelocity);
var rotation = s_rotation;
rotation.SetAngle(step.dt * group.m_angularVelocity);
///b2Transform transform(group.m_center + step.dt * group.m_linearVelocity - b2Mul(rotation, group.m_center), rotation);
var position = b2Vec2.AddVV(group.m_center, b2Vec2.SubVV(b2Vec2.MulSV(step.dt, group.m_linearVelocity, b2Vec2.s_t0), b2Rot.MulRV(rotation, group.m_center, b2Vec2.s_t1), b2Vec2.s_t0), s_position);
var transform = s_transform;
transform.SetPositionRotation(position, rotation);
///group.m_transform = b2Mul(transform, group.m_transform);
b2Transform.MulXX(transform, group.m_transform, group.m_transform);
var velocityTransform = s_velocityTransform;
velocityTransform.p.x = step.inv_dt * transform.p.x;
velocityTransform.p.y = step.inv_dt * transform.p.y;
velocityTransform.q.s = step.inv_dt * transform.q.s;
velocityTransform.q.c = step.inv_dt * (transform.q.c - 1);
for (var i = group.m_firstIndex; i < group.m_lastIndex; i++) {
///m_velocityBuffer.data[i] = b2Mul(velocityTransform, m_positionBuffer.data[i]);
b2Transform.MulXV(velocityTransform, pos_data[i], vel_data[i]);
}
}
}
};
b2ParticleSystem.prototype.SolveElastic = function (step) {
var s_pa = b2ParticleSystem.SolveElastic_s_pa;
var s_pb = b2ParticleSystem.SolveElastic_s_pb;
var s_pc = b2ParticleSystem.SolveElastic_s_pc;
var s_r = b2ParticleSystem.SolveElastic_s_r;
var s_t0 = b2ParticleSystem.SolveElastic_s_t0;
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
var elasticStrength = step.inv_dt * this.m_def.elasticStrength;
for (var k = 0; k < this.m_triadBuffer.count; k++) {
var triad = this.m_triadBuffer.data[k];
if (triad.flags & exports.b2ParticleFlag.b2_elasticParticle) {
var a = triad.indexA;
var b = triad.indexB;
var c = triad.indexC;
var oa = triad.pa;
var ob = triad.pb;
var oc = triad.pc;
///b2Vec2 pa = m_positionBuffer.data[a];
var pa = s_pa.Copy(pos_data[a]);
///b2Vec2 pb = m_positionBuffer.data[b];
var pb = s_pb.Copy(pos_data[b]);
///b2Vec2 pc = m_positionBuffer.data[c];
var pc = s_pc.Copy(pos_data[c]);
var va = vel_data[a];
var vb = vel_data[b];
var vc = vel_data[c];
///pa += step.dt * va;
pa.SelfMulAdd(step.dt, va);
///pb += step.dt * vb;
pb.SelfMulAdd(step.dt, vb);
///pc += step.dt * vc;
pc.SelfMulAdd(step.dt, vc);
///b2Vec2 midPoint = (float32) 1 / 3 * (pa + pb + pc);
var midPoint_x = (pa.x + pb.x + pc.x) / 3.0;
var midPoint_y = (pa.y + pb.y + pc.y) / 3.0;
///pa -= midPoint;
pa.x -= midPoint_x;
pa.y -= midPoint_y;
///pb -= midPoint;
pb.x -= midPoint_x;
pb.y -= midPoint_y;
///pc -= midPoint;
pc.x -= midPoint_x;
pc.y -= midPoint_y;
///b2Rot r;
var r = s_r;
r.s = b2Vec2.CrossVV(oa, pa) + b2Vec2.CrossVV(ob, pb) + b2Vec2.CrossVV(oc, pc);
r.c = b2Vec2.DotVV(oa, pa) + b2Vec2.DotVV(ob, pb) + b2Vec2.DotVV(oc, pc);
var r2 = r.s * r.s + r.c * r.c;
var invR = b2InvSqrt(r2);
if (!isFinite(invR)) {
invR = 1.98177537e+019;
}
r.s *= invR;
r.c *= invR;
///r.angle = Math.atan2(r.s, r.c); // TODO: optimize
var strength = elasticStrength * triad.strength;
///va += strength * (b2Mul(r, oa) - pa);
b2Rot.MulRV(r, oa, s_t0);
b2Vec2.SubVV(s_t0, pa, s_t0);
b2Vec2.MulSV(strength, s_t0, s_t0);
va.SelfAdd(s_t0);
///vb += strength * (b2Mul(r, ob) - pb);
b2Rot.MulRV(r, ob, s_t0);
b2Vec2.SubVV(s_t0, pb, s_t0);
b2Vec2.MulSV(strength, s_t0, s_t0);
vb.SelfAdd(s_t0);
///vc += strength * (b2Mul(r, oc) - pc);
b2Rot.MulRV(r, oc, s_t0);
b2Vec2.SubVV(s_t0, pc, s_t0);
b2Vec2.MulSV(strength, s_t0, s_t0);
vc.SelfAdd(s_t0);
}
}
};
b2ParticleSystem.prototype.SolveSpring = function (step) {
var s_pa = b2ParticleSystem.SolveSpring_s_pa;
var s_pb = b2ParticleSystem.SolveSpring_s_pb;
var s_d = b2ParticleSystem.SolveSpring_s_d;
var s_f = b2ParticleSystem.SolveSpring_s_f;
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
var springStrength = step.inv_dt * this.m_def.springStrength;
for (var k = 0; k < this.m_pairBuffer.count; k++) {
var pair = this.m_pairBuffer.data[k];
if (pair.flags & exports.b2ParticleFlag.b2_springParticle) {
///int32 a = pair.indexA;
var a = pair.indexA;
///int32 b = pair.indexB;
var b = pair.indexB;
///b2Vec2 pa = m_positionBuffer.data[a];
var pa = s_pa.Copy(pos_data[a]);
///b2Vec2 pb = m_positionBuffer.data[b];
var pb = s_pb.Copy(pos_data[b]);
///b2Vec2& va = m_velocityBuffer.data[a];
var va = vel_data[a];
///b2Vec2& vb = m_velocityBuffer.data[b];
var vb = vel_data[b];
///pa += step.dt * va;
pa.SelfMulAdd(step.dt, va);
///pb += step.dt * vb;
pb.SelfMulAdd(step.dt, vb);
///b2Vec2 d = pb - pa;
var d = b2Vec2.SubVV(pb, pa, s_d);
///float32 r0 = pair.distance;
var r0 = pair.distance;
///float32 r1 = d.Length();
var r1 = d.Length();
///float32 strength = springStrength * pair.strength;
var strength = springStrength * pair.strength;
///b2Vec2 f = strength * (r0 - r1) / r1 * d;
var f = b2Vec2.MulSV(strength * (r0 - r1) / r1, d, s_f);
///va -= f;
va.SelfSub(f);
///vb += f;
vb.SelfAdd(f);
}
}
};
b2ParticleSystem.prototype.SolveTensile = function (step) {
var s_weightedNormal = b2ParticleSystem.SolveTensile_s_weightedNormal;
var s_s = b2ParticleSystem.SolveTensile_s_s;
var s_f = b2ParticleSystem.SolveTensile_s_f;
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
// DEBUG: b2Assert(this.m_accumulation2Buffer !== null);
for (var i = 0; i < this.m_count; i++) {
this.m_accumulation2Buffer[i] = new b2Vec2();
this.m_accumulation2Buffer[i].SetZero();
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
if (contact.flags & exports.b2ParticleFlag.b2_tensileParticle) {
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
var n = contact.normal;
///b2Vec2 weightedNormal = (1 - w) * w * n;
var weightedNormal = b2Vec2.MulSV((1 - w) * w, n, s_weightedNormal);
///m_accumulation2Buffer[a] -= weightedNormal;
this.m_accumulation2Buffer[a].SelfSub(weightedNormal);
///m_accumulation2Buffer[b] += weightedNormal;
this.m_accumulation2Buffer[b].SelfAdd(weightedNormal);
}
}
var criticalVelocity = this.GetCriticalVelocity(step);
var pressureStrength = this.m_def.surfaceTensionPressureStrength * criticalVelocity;
var normalStrength = this.m_def.surfaceTensionNormalStrength * criticalVelocity;
var maxVelocityVariation = b2_maxParticleForce * criticalVelocity;
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
if (contact.flags & exports.b2ParticleFlag.b2_tensileParticle) {
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
var n = contact.normal;
var h = this.m_weightBuffer[a] + this.m_weightBuffer[b];
///b2Vec2 s = m_accumulation2Buffer[b] - m_accumulation2Buffer[a];
var s = b2Vec2.SubVV(this.m_accumulation2Buffer[b], this.m_accumulation2Buffer[a], s_s);
var fn = b2Min(pressureStrength * (h - 2) + normalStrength * b2Vec2.DotVV(s, n), maxVelocityVariation) * w;
///b2Vec2 f = fn * n;
var f = b2Vec2.MulSV(fn, n, s_f);
///m_velocityBuffer.data[a] -= f;
vel_data[a].SelfSub(f);
///m_velocityBuffer.data[b] += f;
vel_data[b].SelfAdd(f);
}
}
};
b2ParticleSystem.prototype.SolveViscous = function () {
var s_v = b2ParticleSystem.SolveViscous_s_v;
var s_f = b2ParticleSystem.SolveViscous_s_f;
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
var viscousStrength = this.m_def.viscousStrength;
var inv_mass = this.GetParticleInvMass();
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
var a = contact.index;
if (this.m_flagsBuffer.data[a] & exports.b2ParticleFlag.b2_viscousParticle) {
var b = contact.body;
var w = contact.weight;
var m = contact.mass;
var p = pos_data[a];
///b2Vec2 v = b.GetLinearVelocityFromWorldPoint(p) - m_velocityBuffer.data[a];
var v = b2Vec2.SubVV(b.GetLinearVelocityFromWorldPoint(p, b2Vec2.s_t0), vel_data[a], s_v);
///b2Vec2 f = viscousStrength * m * w * v;
var f = b2Vec2.MulSV(viscousStrength * m * w, v, s_f);
///m_velocityBuffer.data[a] += GetParticleInvMass() * f;
vel_data[a].SelfMulAdd(inv_mass, f);
///b.ApplyLinearImpulse(-f, p, true);
b.ApplyLinearImpulse(f.SelfNeg(), p, true);
}
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
if (contact.flags & exports.b2ParticleFlag.b2_viscousParticle) {
var a = contact.indexA;
var b = contact.indexB;
var w = contact.weight;
///b2Vec2 v = m_velocityBuffer.data[b] - m_velocityBuffer.data[a];
var v = b2Vec2.SubVV(vel_data[b], vel_data[a], s_v);
///b2Vec2 f = viscousStrength * w * v;
var f = b2Vec2.MulSV(viscousStrength * w, v, s_f);
///m_velocityBuffer.data[a] += f;
vel_data[a].SelfAdd(f);
///m_velocityBuffer.data[b] -= f;
vel_data[b].SelfSub(f);
}
}
};
b2ParticleSystem.prototype.SolveRepulsive = function (step) {
var s_f = b2ParticleSystem.SolveRepulsive_s_f;
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
var repulsiveStrength = this.m_def.repulsiveStrength * this.GetCriticalVelocity(step);
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
if (contact.flags & exports.b2ParticleFlag.b2_repulsiveParticle) {
var a = contact.indexA;
var b = contact.indexB;
if (this.m_groupBuffer[a] !== this.m_groupBuffer[b]) {
var w = contact.weight;
var n = contact.normal;
///b2Vec2 f = repulsiveStrength * w * n;
var f = b2Vec2.MulSV(repulsiveStrength * w, n, s_f);
///m_velocityBuffer.data[a] -= f;
vel_data[a].SelfSub(f);
///m_velocityBuffer.data[b] += f;
vel_data[b].SelfAdd(f);
}
}
}
};
b2ParticleSystem.prototype.SolvePowder = function (step) {
var s_f = b2ParticleSystem.SolvePowder_s_f;
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var pos_data = this.m_positionBuffer.data;
var vel_data = this.m_velocityBuffer.data;
var powderStrength = this.m_def.powderStrength * this.GetCriticalVelocity(step);
var minWeight = 1.0 - b2_particleStride;
var inv_mass = this.GetParticleInvMass();
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
var a = contact.index;
if (this.m_flagsBuffer.data[a] & exports.b2ParticleFlag.b2_powderParticle) {
var w = contact.weight;
if (w > minWeight) {
var b = contact.body;
var m = contact.mass;
var p = pos_data[a];
var n = contact.normal;
var f = b2Vec2.MulSV(powderStrength * m * (w - minWeight), n, s_f);
vel_data[a].SelfMulSub(inv_mass, f);
b.ApplyLinearImpulse(f, p, true);
}
}
}
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
if (contact.flags & exports.b2ParticleFlag.b2_powderParticle) {
var w = contact.weight;
if (w > minWeight) {
var a = contact.indexA;
var b = contact.indexB;
var n = contact.normal;
var f = b2Vec2.MulSV(powderStrength * (w - minWeight), n, s_f);
vel_data[a].SelfSub(f);
vel_data[b].SelfAdd(f);
}
}
}
};
b2ParticleSystem.prototype.SolveSolid = function (step) {
var s_f = b2ParticleSystem.SolveSolid_s_f;
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
// applies extra repulsive force from solid particle groups
this.m_depthBuffer = this.RequestBuffer(this.m_depthBuffer);
var ejectionStrength = step.inv_dt * this.m_def.ejectionStrength;
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
if (this.m_groupBuffer[a] !== this.m_groupBuffer[b]) {
var w = contact.weight;
var n = contact.normal;
var h = this.m_depthBuffer[a] + this.m_depthBuffer[b];
var f = b2Vec2.MulSV(ejectionStrength * h * w, n, s_f);
vel_data[a].SelfSub(f);
vel_data[b].SelfAdd(f);
}
}
};
b2ParticleSystem.prototype.SolveForce = function (step) {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
var vel_data = this.m_velocityBuffer.data;
var velocityPerForce = step.dt * this.GetParticleInvMass();
for (var i = 0; i < this.m_count; i++) {
///m_velocityBuffer.data[i] += velocityPerForce * m_forceBuffer[i];
vel_data[i].SelfMulAdd(velocityPerForce, this.m_forceBuffer[i]);
}
this.m_hasForce = false;
};
b2ParticleSystem.prototype.SolveColorMixing = function () {
// mixes color between contacting particles
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_colorBuffer.data) {
throw new Error();
}
var colorMixing = 0.5 * this.m_def.colorMixingStrength;
if (colorMixing) {
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
var a = contact.indexA;
var b = contact.indexB;
if (this.m_flagsBuffer.data[a] & this.m_flagsBuffer.data[b] &
exports.b2ParticleFlag.b2_colorMixingParticle) {
var colorA = this.m_colorBuffer.data[a];
var colorB = this.m_colorBuffer.data[b];
// Use the static method to ensure certain compilers inline
// this correctly.
b2Color.MixColors(colorA, colorB, colorMixing);
}
}
}
};
b2ParticleSystem.prototype.SolveZombie = function () {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
// removes particles with zombie flag
var newCount = 0;
///int32* newIndices = (int32*) this.m_world.m_stackAllocator.Allocate(sizeof(int32) * this.m_count);
var newIndices = []; // TODO: static
for (var i = 0; i < this.m_count; i++) {
newIndices[i] = b2_invalidParticleIndex;
}
// DEBUG: b2Assert(newIndices.length === this.m_count);
var allParticleFlags = 0;
for (var i = 0; i < this.m_count; i++) {
var flags = this.m_flagsBuffer.data[i];
if (flags & exports.b2ParticleFlag.b2_zombieParticle) {
var destructionListener = this.m_world.m_destructionListener;
if ((flags & exports.b2ParticleFlag.b2_destructionListenerParticle) && destructionListener) {
destructionListener.SayGoodbyeParticle(this, i);
}
// Destroy particle handle.
if (this.m_handleIndexBuffer.data) {
var handle = this.m_handleIndexBuffer.data[i];
if (handle) {
handle.SetIndex(b2_invalidParticleIndex);
this.m_handleIndexBuffer.data[i] = null;
///m_handleAllocator.Free(handle);
}
}
newIndices[i] = b2_invalidParticleIndex;
}
else {
newIndices[i] = newCount;
if (i !== newCount) {
// Update handle to reference new particle index.
if (this.m_handleIndexBuffer.data) {
var handle = this.m_handleIndexBuffer.data[i];
if (handle) {
handle.SetIndex(newCount);
}
this.m_handleIndexBuffer.data[newCount] = handle;
}
this.m_flagsBuffer.data[newCount] = this.m_flagsBuffer.data[i];
if (this.m_lastBodyContactStepBuffer.data) {
this.m_lastBodyContactStepBuffer.data[newCount] = this.m_lastBodyContactStepBuffer.data[i];
}
if (this.m_bodyContactCountBuffer.data) {
this.m_bodyContactCountBuffer.data[newCount] = this.m_bodyContactCountBuffer.data[i];
}
if (this.m_consecutiveContactStepsBuffer.data) {
this.m_consecutiveContactStepsBuffer.data[newCount] = this.m_consecutiveContactStepsBuffer.data[i];
}
this.m_positionBuffer.data[newCount].Copy(this.m_positionBuffer.data[i]);
this.m_velocityBuffer.data[newCount].Copy(this.m_velocityBuffer.data[i]);
this.m_groupBuffer[newCount] = this.m_groupBuffer[i];
if (this.m_hasForce) {
this.m_forceBuffer[newCount].Copy(this.m_forceBuffer[i]);
}
if (this.m_staticPressureBuffer) {
this.m_staticPressureBuffer[newCount] = this.m_staticPressureBuffer[i];
}
if (this.m_depthBuffer) {
this.m_depthBuffer[newCount] = this.m_depthBuffer[i];
}
if (this.m_colorBuffer.data) {
this.m_colorBuffer.data[newCount].Copy(this.m_colorBuffer.data[i]);
}
if (this.m_userDataBuffer.data) {
this.m_userDataBuffer.data[newCount] = this.m_userDataBuffer.data[i];
}
if (this.m_expirationTimeBuffer.data) {
this.m_expirationTimeBuffer.data[newCount] = this.m_expirationTimeBuffer.data[i];
}
}
newCount++;
allParticleFlags |= flags;
}
}
// predicate functions
var Test = {
///static bool IsProxyInvalid(const Proxy& proxy)
IsProxyInvalid: function (proxy) {
return proxy.index < 0;
},
///static bool IsContactInvalid(const b2ParticleContact& contact)
IsContactInvalid: function (contact) {
return contact.indexA < 0 || contact.indexB < 0;
},
///static bool IsBodyContactInvalid(const b2ParticleBodyContact& contact)
IsBodyContactInvalid: function (contact) {
return contact.index < 0;
},
///static bool IsPairInvalid(const b2ParticlePair& pair)
IsPairInvalid: function (pair) {
return pair.indexA < 0 || pair.indexB < 0;
},
///static bool IsTriadInvalid(const b2ParticleTriad& triad)
IsTriadInvalid: function (triad) {
return triad.indexA < 0 || triad.indexB < 0 || triad.indexC < 0;
},
};
// update proxies
for (var k = 0; k < this.m_proxyBuffer.count; k++) {
var proxy = this.m_proxyBuffer.data[k];
proxy.index = newIndices[proxy.index];
}
this.m_proxyBuffer.RemoveIf(Test.IsProxyInvalid);
// update contacts
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
contact.indexA = newIndices[contact.indexA];
contact.indexB = newIndices[contact.indexB];
}
this.m_contactBuffer.RemoveIf(Test.IsContactInvalid);
// update particle-body contacts
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
contact.index = newIndices[contact.index];
}
this.m_bodyContactBuffer.RemoveIf(Test.IsBodyContactInvalid);
// update pairs
for (var k = 0; k < this.m_pairBuffer.count; k++) {
var pair = this.m_pairBuffer.data[k];
pair.indexA = newIndices[pair.indexA];
pair.indexB = newIndices[pair.indexB];
}
this.m_pairBuffer.RemoveIf(Test.IsPairInvalid);
// update triads
for (var k = 0; k < this.m_triadBuffer.count; k++) {
var triad = this.m_triadBuffer.data[k];
triad.indexA = newIndices[triad.indexA];
triad.indexB = newIndices[triad.indexB];
triad.indexC = newIndices[triad.indexC];
}
this.m_triadBuffer.RemoveIf(Test.IsTriadInvalid);
// Update lifetime indices.
if (this.m_indexByExpirationTimeBuffer.data) {
var writeOffset = 0;
for (var readOffset = 0; readOffset < this.m_count; readOffset++) {
var newIndex = newIndices[this.m_indexByExpirationTimeBuffer.data[readOffset]];
if (newIndex !== b2_invalidParticleIndex) {
this.m_indexByExpirationTimeBuffer.data[writeOffset++] = newIndex;
}
}
}
// update groups
for (var group = this.m_groupList; group; group = group.GetNext()) {
var firstIndex = newCount;
var lastIndex = 0;
var modified = false;
for (var i = group.m_firstIndex; i < group.m_lastIndex; i++) {
var j = newIndices[i];
if (j >= 0) {
firstIndex = b2Min(firstIndex, j);
lastIndex = b2Max(lastIndex, j + 1);
}
else {
modified = true;
}
}
if (firstIndex < lastIndex) {
group.m_firstIndex = firstIndex;
group.m_lastIndex = lastIndex;
if (modified) {
if (group.m_groupFlags & exports.b2ParticleGroupFlag.b2_solidParticleGroup) {
this.SetGroupFlags(group, group.m_groupFlags | exports.b2ParticleGroupFlag.b2_particleGroupNeedsUpdateDepth);
}
}
}
else {
group.m_firstIndex = 0;
group.m_lastIndex = 0;
if (!(group.m_groupFlags & exports.b2ParticleGroupFlag.b2_particleGroupCanBeEmpty)) {
this.SetGroupFlags(group, group.m_groupFlags | exports.b2ParticleGroupFlag.b2_particleGroupWillBeDestroyed);
}
}
}
// update particle count
this.m_count = newCount;
///m_world.m_stackAllocator.Free(newIndices);
this.m_allParticleFlags = allParticleFlags;
this.m_needsUpdateAllParticleFlags = false;
// destroy bodies with no particles
for (var group = this.m_groupList; group;) {
var next = group.GetNext();
if (group.m_groupFlags & exports.b2ParticleGroupFlag.b2_particleGroupWillBeDestroyed) {
this.DestroyParticleGroup(group);
}
group = next;
}
};
/**
* Destroy all particles which have outlived their lifetimes set
* by SetParticleLifetime().
*/
b2ParticleSystem.prototype.SolveLifetimes = function (step) {
if (!this.m_expirationTimeBuffer.data) {
throw new Error();
}
if (!this.m_indexByExpirationTimeBuffer.data) {
throw new Error();
}
// Update the time elapsed.
this.m_timeElapsed = this.LifetimeToExpirationTime(step.dt);
// Get the floor (non-fractional component) of the elapsed time.
var quantizedTimeElapsed = this.GetQuantizedTimeElapsed();
var expirationTimes = this.m_expirationTimeBuffer.data;
var expirationTimeIndices = this.m_indexByExpirationTimeBuffer.data;
var particleCount = this.GetParticleCount();
// Sort the lifetime buffer if it's required.
if (this.m_expirationTimeBufferRequiresSorting) {
///const ExpirationTimeComparator expirationTimeComparator(expirationTimes);
///std::sort(expirationTimeIndices, expirationTimeIndices + particleCount, expirationTimeComparator);
/**
* Compare the lifetime of particleIndexA and particleIndexB
* returning true if the lifetime of A is greater than B for
* particles that will expire. If either particle's lifetime is
* infinite (<= 0.0f) this function return true if the lifetime
* of A is lesser than B. When used with std::sort() this
* results in an array of particle indicies sorted in reverse
* order by particle lifetime.
*
* For example, the set of lifetimes
* (1.0, 0.7, 0.3, 0.0, -1.0, 2.0)
* would be sorted as
* (0.0, 1.0, -2.0, 1.0, 0.7, 0.3)
*/
var ExpirationTimeComparator = function (particleIndexA, particleIndexB) {
var expirationTimeA = expirationTimes[particleIndexA];
var expirationTimeB = expirationTimes[particleIndexB];
var infiniteExpirationTimeA = expirationTimeA <= 0.0;
var infiniteExpirationTimeB = expirationTimeB <= 0.0;
return infiniteExpirationTimeA === infiniteExpirationTimeB ?
expirationTimeA > expirationTimeB : infiniteExpirationTimeA;
};
std_sort(expirationTimeIndices, 0, particleCount, ExpirationTimeComparator);
this.m_expirationTimeBufferRequiresSorting = false;
}
// Destroy particles which have expired.
for (var i = particleCount - 1; i >= 0; --i) {
var particleIndex = expirationTimeIndices[i];
var expirationTime = expirationTimes[particleIndex];
// If no particles need to be destroyed, skip this.
if (quantizedTimeElapsed < expirationTime || expirationTime <= 0) {
break;
}
// Destroy this particle.
this.DestroyParticle(particleIndex);
}
};
b2ParticleSystem.prototype.RotateBuffer = function (start, mid, end) {
// move the particles assigned to the given group toward the end of array
if (start === mid || mid === end) {
return;
}
// DEBUG: b2Assert(mid >= start && mid <= end);
function newIndices(i) {
if (i < start) {
return i;
}
else if (i < mid) {
return i + end - mid;
}
else if (i < end) {
return i + start - mid;
}
else {
return i;
}
}
if (!this.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_velocityBuffer.data) {
throw new Error();
}
///std::rotate(m_flagsBuffer.data + start, m_flagsBuffer.data + mid, m_flagsBuffer.data + end);
std_rotate(this.m_flagsBuffer.data, start, mid, end);
if (this.m_lastBodyContactStepBuffer.data) {
///std::rotate(m_lastBodyContactStepBuffer.data + start, m_lastBodyContactStepBuffer.data + mid, m_lastBodyContactStepBuffer.data + end);
std_rotate(this.m_lastBodyContactStepBuffer.data, start, mid, end);
}
if (this.m_bodyContactCountBuffer.data) {
///std::rotate(m_bodyContactCountBuffer.data + start, m_bodyContactCountBuffer.data + mid, m_bodyContactCountBuffer.data + end);
std_rotate(this.m_bodyContactCountBuffer.data, start, mid, end);
}
if (this.m_consecutiveContactStepsBuffer.data) {
///std::rotate(m_consecutiveContactStepsBuffer.data + start, m_consecutiveContactStepsBuffer.data + mid, m_consecutiveContactStepsBuffer.data + end);
std_rotate(this.m_consecutiveContactStepsBuffer.data, start, mid, end);
}
///std::rotate(m_positionBuffer.data + start, m_positionBuffer.data + mid, m_positionBuffer.data + end);
std_rotate(this.m_positionBuffer.data, start, mid, end);
///std::rotate(m_velocityBuffer.data + start, m_velocityBuffer.data + mid, m_velocityBuffer.data + end);
std_rotate(this.m_velocityBuffer.data, start, mid, end);
///std::rotate(m_groupBuffer + start, m_groupBuffer + mid, m_groupBuffer + end);
std_rotate(this.m_groupBuffer, start, mid, end);
if (this.m_hasForce) {
///std::rotate(m_forceBuffer + start, m_forceBuffer + mid, m_forceBuffer + end);
std_rotate(this.m_forceBuffer, start, mid, end);
}
if (this.m_staticPressureBuffer) {
///std::rotate(m_staticPressureBuffer + start, m_staticPressureBuffer + mid, m_staticPressureBuffer + end);
std_rotate(this.m_staticPressureBuffer, start, mid, end);
}
if (this.m_depthBuffer) {
///std::rotate(m_depthBuffer + start, m_depthBuffer + mid, m_depthBuffer + end);
std_rotate(this.m_depthBuffer, start, mid, end);
}
if (this.m_colorBuffer.data) {
///std::rotate(m_colorBuffer.data + start, m_colorBuffer.data + mid, m_colorBuffer.data + end);
std_rotate(this.m_colorBuffer.data, start, mid, end);
}
if (this.m_userDataBuffer.data) {
///std::rotate(m_userDataBuffer.data + start, m_userDataBuffer.data + mid, m_userDataBuffer.data + end);
std_rotate(this.m_userDataBuffer.data, start, mid, end);
}
// Update handle indices.
if (this.m_handleIndexBuffer.data) {
///std::rotate(m_handleIndexBuffer.data + start, m_handleIndexBuffer.data + mid, m_handleIndexBuffer.data + end);
std_rotate(this.m_handleIndexBuffer.data, start, mid, end);
for (var i = start; i < end; ++i) {
var handle = this.m_handleIndexBuffer.data[i];
if (handle) {
handle.SetIndex(newIndices(handle.GetIndex()));
}
}
}
if (this.m_expirationTimeBuffer.data) {
///std::rotate(m_expirationTimeBuffer.data + start, m_expirationTimeBuffer.data + mid, m_expirationTimeBuffer.data + end);
std_rotate(this.m_expirationTimeBuffer.data, start, mid, end);
// Update expiration time buffer indices.
var particleCount = this.GetParticleCount();
if (!this.m_indexByExpirationTimeBuffer.data) {
throw new Error();
}
var indexByExpirationTime = this.m_indexByExpirationTimeBuffer.data;
for (var i = 0; i < particleCount; ++i) {
indexByExpirationTime[i] = newIndices(indexByExpirationTime[i]);
}
}
// update proxies
for (var k = 0; k < this.m_proxyBuffer.count; k++) {
var proxy = this.m_proxyBuffer.data[k];
proxy.index = newIndices(proxy.index);
}
// update contacts
for (var k = 0; k < this.m_contactBuffer.count; k++) {
var contact = this.m_contactBuffer.data[k];
contact.indexA = newIndices(contact.indexA);
contact.indexB = newIndices(contact.indexB);
}
// update particle-body contacts
for (var k = 0; k < this.m_bodyContactBuffer.count; k++) {
var contact = this.m_bodyContactBuffer.data[k];
contact.index = newIndices(contact.index);
}
// update pairs
for (var k = 0; k < this.m_pairBuffer.count; k++) {
var pair = this.m_pairBuffer.data[k];
pair.indexA = newIndices(pair.indexA);
pair.indexB = newIndices(pair.indexB);
}
// update triads
for (var k = 0; k < this.m_triadBuffer.count; k++) {
var triad = this.m_triadBuffer.data[k];
triad.indexA = newIndices(triad.indexA);
triad.indexB = newIndices(triad.indexB);
triad.indexC = newIndices(triad.indexC);
}
// update groups
for (var group = this.m_groupList; group; group = group.GetNext()) {
group.m_firstIndex = newIndices(group.m_firstIndex);
group.m_lastIndex = newIndices(group.m_lastIndex - 1) + 1;
}
};
b2ParticleSystem.prototype.GetCriticalVelocity = function (step) {
return this.m_particleDiameter * step.inv_dt;
};
b2ParticleSystem.prototype.GetCriticalVelocitySquared = function (step) {
var velocity = this.GetCriticalVelocity(step);
return velocity * velocity;
};
b2ParticleSystem.prototype.GetCriticalPressure = function (step) {
return this.m_def.density * this.GetCriticalVelocitySquared(step);
};
b2ParticleSystem.prototype.GetParticleStride = function () {
return b2_particleStride * this.m_particleDiameter;
};
b2ParticleSystem.prototype.GetParticleMass = function () {
var stride = this.GetParticleStride();
return this.m_def.density * stride * stride;
};
b2ParticleSystem.prototype.GetParticleInvMass = function () {
///return 1.777777 * this.m_inverseDensity * this.m_inverseDiameter * this.m_inverseDiameter;
// mass = density * stride^2, so we take the inverse of this.
var inverseStride = this.m_inverseDiameter * (1.0 / b2_particleStride);
return this.m_inverseDensity * inverseStride * inverseStride;
};
/**
* Get the world's contact filter if any particles with the
* b2_contactFilterParticle flag are present in the system.
*/
b2ParticleSystem.prototype.GetFixtureContactFilter = function () {
return (this.m_allParticleFlags & exports.b2ParticleFlag.b2_fixtureContactFilterParticle) ?
this.m_world.m_contactManager.m_contactFilter : null;
};
/**
* Get the world's contact filter if any particles with the
* b2_particleContactFilterParticle flag are present in the
* system.
*/
b2ParticleSystem.prototype.GetParticleContactFilter = function () {
return (this.m_allParticleFlags & exports.b2ParticleFlag.b2_particleContactFilterParticle) ?
this.m_world.m_contactManager.m_contactFilter : null;
};
/**
* Get the world's contact listener if any particles with the
* b2_fixtureContactListenerParticle flag are present in the
* system.
*/
b2ParticleSystem.prototype.GetFixtureContactListener = function () {
return (this.m_allParticleFlags & exports.b2ParticleFlag.b2_fixtureContactListenerParticle) ?
this.m_world.m_contactManager.m_contactListener : null;
};
/**
* Get the world's contact listener if any particles with the
* b2_particleContactListenerParticle flag are present in the
* system.
*/
b2ParticleSystem.prototype.GetParticleContactListener = function () {
return (this.m_allParticleFlags & exports.b2ParticleFlag.b2_particleContactListenerParticle) ?
this.m_world.m_contactManager.m_contactListener : null;
};
b2ParticleSystem.prototype.SetUserOverridableBuffer = function (buffer, newData, newCapacity) {
// DEBUG: b2Assert(((newData !== null) && (newCapacity > 0)) || ((newData === null) && (newCapacity === 0)));
///if (!buffer.userSuppliedCapacity)
///{
///this.m_world.m_blockAllocator.Free(buffer.data, sizeof(T) * m_internalAllocatedCapacity);
///}
buffer.data = newData;
buffer.userSuppliedCapacity = newCapacity;
};
b2ParticleSystem.prototype.SetGroupFlags = function (group, newFlags) {
var oldFlags = group.m_groupFlags;
if ((oldFlags ^ newFlags) & exports.b2ParticleGroupFlag.b2_solidParticleGroup) {
// If the b2_solidParticleGroup flag changed schedule depth update.
newFlags |= exports.b2ParticleGroupFlag.b2_particleGroupNeedsUpdateDepth;
}
if (oldFlags & ~newFlags) {
// If any flags might be removed
this.m_needsUpdateAllGroupFlags = true;
}
if (~this.m_allGroupFlags & newFlags) {
// If any flags were added
if (newFlags & exports.b2ParticleGroupFlag.b2_solidParticleGroup) {
this.m_depthBuffer = this.RequestBuffer(this.m_depthBuffer);
}
this.m_allGroupFlags |= newFlags;
}
group.m_groupFlags = newFlags;
};
b2ParticleSystem.BodyContactCompare = function (lhs, rhs) {
if (lhs.index === rhs.index) {
// Subsort by weight, decreasing.
return lhs.weight > rhs.weight;
}
return lhs.index < rhs.index;
};
b2ParticleSystem.prototype.RemoveSpuriousBodyContacts = function () {
// At this point we have a list of contact candidates based on AABB
// overlap.The AABB query that generated this returns all collidable
// fixtures overlapping particle bounding boxes. This breaks down around
// vertices where two shapes intersect, such as a "ground" surface made
// of multiple b2PolygonShapes; it potentially applies a lot of spurious
// impulses from normals that should not actually contribute. See the
// Ramp example in Testbed.
//
// To correct for this, we apply this algorithm:
// * sort contacts by particle and subsort by weight (nearest to farthest)
// * for each contact per particle:
// - project a point at the contact distance along the inverse of the
// contact normal
// - if this intersects the fixture that generated the contact, apply
// it, otherwise discard as impossible
// - repeat for up to n nearest contacts, currently we get good results
// from n=3.
///std::sort(m_bodyContactBuffer.Begin(), m_bodyContactBuffer.End(), b2ParticleSystem::BodyContactCompare);
std_sort(this.m_bodyContactBuffer.data, 0, this.m_bodyContactBuffer.count, b2ParticleSystem.BodyContactCompare);
///int32 discarded = 0;
///std::remove_if(m_bodyContactBuffer.Begin(), m_bodyContactBuffer.End(), b2ParticleBodyContactRemovePredicate(this, &discarded));
///
///m_bodyContactBuffer.SetCount(m_bodyContactBuffer.GetCount() - discarded);
var s_n = b2ParticleSystem.RemoveSpuriousBodyContacts_s_n;
var s_pos = b2ParticleSystem.RemoveSpuriousBodyContacts_s_pos;
var s_normal = b2ParticleSystem.RemoveSpuriousBodyContacts_s_normal;
// Max number of contacts processed per particle, from nearest to farthest.
// This must be at least 2 for correctness with concave shapes; 3 was
// experimentally arrived at as looking reasonable.
var k_maxContactsPerPoint = 3;
var system = this;
// Index of last particle processed.
var lastIndex = -1;
// Number of contacts processed for the current particle.
var currentContacts = 0;
// Output the number of discarded contacts.
// let discarded = 0;
var b2ParticleBodyContactRemovePredicate = function (contact) {
// This implements the selection criteria described in
// RemoveSpuriousBodyContacts().
// This functor is iterating through a list of Body contacts per
// Particle, ordered from near to far. For up to the maximum number of
// contacts we allow per point per step, we verify that the contact
// normal of the Body that genenerated the contact makes physical sense
// by projecting a point back along that normal and seeing if it
// intersects the fixture generating the contact.
if (contact.index !== lastIndex) {
currentContacts = 0;
lastIndex = contact.index;
}
if (currentContacts++ > k_maxContactsPerPoint) {
// ++discarded;
return true;
}
// Project along inverse normal (as returned in the contact) to get the
// point to check.
///b2Vec2 n = contact.normal;
var n = s_n.Copy(contact.normal);
// weight is 1-(inv(diameter) * distance)
///n *= system.m_particleDiameter * (1 - contact.weight);
n.SelfMul(system.m_particleDiameter * (1 - contact.weight));
///b2Vec2 pos = system.m_positionBuffer.data[contact.index] + n;
if (!system.m_positionBuffer.data) {
throw new Error();
}
var pos = b2Vec2.AddVV(system.m_positionBuffer.data[contact.index], n, s_pos);
// pos is now a point projected back along the contact normal to the
// contact distance. If the surface makes sense for a contact, pos will
// now lie on or in the fixture generating
if (!contact.fixture.TestPoint(pos)) {
var childCount = contact.fixture.GetShape().GetChildCount();
for (var childIndex = 0; childIndex < childCount; childIndex++) {
var normal = s_normal;
var distance = contact.fixture.ComputeDistance(pos, normal, childIndex);
if (distance < b2_linearSlop) {
return false;
}
}
// ++discarded;
return true;
}
return false;
};
this.m_bodyContactBuffer.count = std_remove_if(this.m_bodyContactBuffer.data, b2ParticleBodyContactRemovePredicate, this.m_bodyContactBuffer.count);
};
b2ParticleSystem.prototype.DetectStuckParticle = function (particle) {
// Detect stuck particles
//
// The basic algorithm is to allow the user to specify an optional
// threshold where we detect whenever a particle is contacting
// more than one fixture for more than threshold consecutive
// steps. This is considered to be "stuck", and these are put
// in a list the user can query per step, if enabled, to deal with
// such particles.
if (this.m_stuckThreshold <= 0) {
return;
}
if (!this.m_bodyContactCountBuffer.data) {
throw new Error();
}
if (!this.m_consecutiveContactStepsBuffer.data) {
throw new Error();
}
if (!this.m_lastBodyContactStepBuffer.data) {
throw new Error();
}
// Get the state variables for this particle.
///int32 * const consecutiveCount = &m_consecutiveContactStepsBuffer.data[particle];
///int32 * const lastStep = &m_lastBodyContactStepBuffer.data[particle];
///int32 * const bodyCount = &m_bodyContactCountBuffer.data[particle];
// This is only called when there is a body contact for this particle.
///++(*bodyCount);
++this.m_bodyContactCountBuffer.data[particle];
// We want to only trigger detection once per step, the first time we
// contact more than one fixture in a step for a given particle.
///if (*bodyCount === 2)
if (this.m_bodyContactCountBuffer.data[particle] === 2) {
///++(*consecutiveCount);
++this.m_consecutiveContactStepsBuffer.data[particle];
///if (*consecutiveCount > m_stuckThreshold)
if (this.m_consecutiveContactStepsBuffer.data[particle] > this.m_stuckThreshold) {
///int32& newStuckParticle = m_stuckParticleBuffer.Append();
///newStuckParticle = particle;
this.m_stuckParticleBuffer.data[this.m_stuckParticleBuffer.Append()] = particle;
}
}
///*lastStep = m_timestamp;
this.m_lastBodyContactStepBuffer.data[particle] = this.m_timestamp;
};
/**
* Determine whether a particle index is valid.
*/
b2ParticleSystem.prototype.ValidateParticleIndex = function (index) {
return index >= 0 && index < this.GetParticleCount() &&
index !== b2_invalidParticleIndex;
};
/**
* Get the time elapsed in
* b2ParticleSystemDef::lifetimeGranularity.
*/
b2ParticleSystem.prototype.GetQuantizedTimeElapsed = function () {
///return (int32)(m_timeElapsed >> 32);
return Math.floor(this.m_timeElapsed / 0x100000000);
};
/**
* Convert a lifetime in seconds to an expiration time.
*/
b2ParticleSystem.prototype.LifetimeToExpirationTime = function (lifetime) {
///return m_timeElapsed + (int64)((lifetime / m_def.lifetimeGranularity) * (float32)(1LL << 32));
return this.m_timeElapsed + Math.floor(((lifetime / this.m_def.lifetimeGranularity) * 0x100000000));
};
b2ParticleSystem.prototype.ForceCanBeApplied = function (flags) {
return !(flags & exports.b2ParticleFlag.b2_wallParticle);
};
b2ParticleSystem.prototype.PrepareForceBuffer = function () {
if (!this.m_hasForce) {
///memset(m_forceBuffer, 0, sizeof(*m_forceBuffer) * m_count);
for (var i = 0; i < this.m_count; i++) {
this.m_forceBuffer[i].SetZero();
}
this.m_hasForce = true;
}
};
b2ParticleSystem.prototype.IsRigidGroup = function (group) {
return (group !== null) && ((group.m_groupFlags & exports.b2ParticleGroupFlag.b2_rigidParticleGroup) !== 0);
};
b2ParticleSystem.prototype.GetLinearVelocity = function (group, particleIndex, point, out) {
if (group && this.IsRigidGroup(group)) {
return group.GetLinearVelocityFromWorldPoint(point, out);
}
else {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
///return m_velocityBuffer.data[particleIndex];
return out.Copy(this.m_velocityBuffer.data[particleIndex]);
}
};
b2ParticleSystem.prototype.InitDampingParameter = function (invMass, invInertia, tangentDistance, mass, inertia, center, point, normal) {
///*invMass = mass > 0 ? 1 / mass : 0;
invMass[0] = mass > 0 ? 1 / mass : 0;
///*invInertia = inertia > 0 ? 1 / inertia : 0;
invInertia[0] = inertia > 0 ? 1 / inertia : 0;
///*tangentDistance = b2Cross(point - center, normal);
tangentDistance[0] = b2Vec2.CrossVV(b2Vec2.SubVV(point, center, b2Vec2.s_t0), normal);
};
b2ParticleSystem.prototype.InitDampingParameterWithRigidGroupOrParticle = function (invMass, invInertia, tangentDistance, isRigidGroup, group, particleIndex, point, normal) {
if (group && isRigidGroup) {
this.InitDampingParameter(invMass, invInertia, tangentDistance, group.GetMass(), group.GetInertia(), group.GetCenter(), point, normal);
}
else {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
var flags = this.m_flagsBuffer.data[particleIndex];
this.InitDampingParameter(invMass, invInertia, tangentDistance, flags & exports.b2ParticleFlag.b2_wallParticle ? 0 : this.GetParticleMass(), 0, point, point, normal);
}
};
b2ParticleSystem.prototype.ComputeDampingImpulse = function (invMassA, invInertiaA, tangentDistanceA, invMassB, invInertiaB, tangentDistanceB, normalVelocity) {
var invMass = invMassA + invInertiaA * tangentDistanceA * tangentDistanceA +
invMassB + invInertiaB * tangentDistanceB * tangentDistanceB;
return invMass > 0 ? normalVelocity / invMass : 0;
};
b2ParticleSystem.prototype.ApplyDamping = function (invMass, invInertia, tangentDistance, isRigidGroup, group, particleIndex, impulse, normal) {
if (group && isRigidGroup) {
///group.m_linearVelocity += impulse * invMass * normal;
group.m_linearVelocity.SelfMulAdd(impulse * invMass, normal);
///group.m_angularVelocity += impulse * tangentDistance * invInertia;
group.m_angularVelocity += impulse * tangentDistance * invInertia;
}
else {
if (!this.m_velocityBuffer.data) {
throw new Error();
}
///m_velocityBuffer.data[particleIndex] += impulse * invMass * normal;
this.m_velocityBuffer.data[particleIndex].SelfMulAdd(impulse * invMass, normal);
}
};
b2ParticleSystem.xTruncBits = 12;
b2ParticleSystem.yTruncBits = 12;
b2ParticleSystem.tagBits = 8 * 4; // 8u * sizeof(uint32);
b2ParticleSystem.yOffset = 1 << (b2ParticleSystem.yTruncBits - 1);
b2ParticleSystem.yShift = b2ParticleSystem.tagBits - b2ParticleSystem.yTruncBits;
b2ParticleSystem.xShift = b2ParticleSystem.tagBits - b2ParticleSystem.yTruncBits - b2ParticleSystem.xTruncBits;
b2ParticleSystem.xScale = 1 << b2ParticleSystem.xShift;
b2ParticleSystem.xOffset = b2ParticleSystem.xScale * (1 << (b2ParticleSystem.xTruncBits - 1));
b2ParticleSystem.yMask = ((1 << b2ParticleSystem.yTruncBits) - 1) << b2ParticleSystem.yShift;
b2ParticleSystem.xMask = ~b2ParticleSystem.yMask;
b2ParticleSystem.DestroyParticlesInShape_s_aabb = new b2AABB();
b2ParticleSystem.CreateParticleGroup_s_transform = new b2Transform();
b2ParticleSystem.ComputeCollisionEnergy_s_v = new b2Vec2();
b2ParticleSystem.QueryShapeAABB_s_aabb = new b2AABB();
b2ParticleSystem.QueryPointAABB_s_aabb = new b2AABB();
b2ParticleSystem.RayCast_s_aabb = new b2AABB();
b2ParticleSystem.RayCast_s_p = new b2Vec2();
b2ParticleSystem.RayCast_s_v = new b2Vec2();
b2ParticleSystem.RayCast_s_n = new b2Vec2();
b2ParticleSystem.RayCast_s_point = new b2Vec2();
/**
* All particle types that require creating pairs
*/
b2ParticleSystem.k_pairFlags = exports.b2ParticleFlag.b2_springParticle;
/**
* All particle types that require creating triads
*/
b2ParticleSystem.k_triadFlags = exports.b2ParticleFlag.b2_elasticParticle;
/**
* All particle types that do not produce dynamic pressure
*/
b2ParticleSystem.k_noPressureFlags = exports.b2ParticleFlag.b2_powderParticle | exports.b2ParticleFlag.b2_tensileParticle;
/**
* All particle types that apply extra damping force with bodies
*/
b2ParticleSystem.k_extraDampingFlags = exports.b2ParticleFlag.b2_staticPressureParticle;
b2ParticleSystem.k_barrierWallFlags = exports.b2ParticleFlag.b2_barrierParticle | exports.b2ParticleFlag.b2_wallParticle;
b2ParticleSystem.CreateParticlesStrokeShapeForGroup_s_edge = new b2EdgeShape();
b2ParticleSystem.CreateParticlesStrokeShapeForGroup_s_d = new b2Vec2();
b2ParticleSystem.CreateParticlesStrokeShapeForGroup_s_p = new b2Vec2();
b2ParticleSystem.CreateParticlesFillShapeForGroup_s_aabb = new b2AABB();
b2ParticleSystem.CreateParticlesFillShapeForGroup_s_p = new b2Vec2();
b2ParticleSystem.UpdatePairsAndTriads_s_dab = new b2Vec2();
b2ParticleSystem.UpdatePairsAndTriads_s_dbc = new b2Vec2();
b2ParticleSystem.UpdatePairsAndTriads_s_dca = new b2Vec2();
b2ParticleSystem.AddContact_s_d = new b2Vec2();
b2ParticleSystem.UpdateBodyContacts_s_aabb = new b2AABB();
b2ParticleSystem.Solve_s_subStep = new b2TimeStep();
b2ParticleSystem.SolveCollision_s_aabb = new b2AABB();
b2ParticleSystem.SolveGravity_s_gravity = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_aabb = new b2AABB();
b2ParticleSystem.SolveBarrier_s_va = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_vb = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_pba = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_vba = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_vc = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_pca = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_vca = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_qba = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_qca = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_dv = new b2Vec2();
b2ParticleSystem.SolveBarrier_s_f = new b2Vec2();
b2ParticleSystem.SolvePressure_s_f = new b2Vec2();
b2ParticleSystem.SolveDamping_s_v = new b2Vec2();
b2ParticleSystem.SolveDamping_s_f = new b2Vec2();
b2ParticleSystem.SolveRigidDamping_s_t0 = new b2Vec2();
b2ParticleSystem.SolveRigidDamping_s_t1 = new b2Vec2();
b2ParticleSystem.SolveRigidDamping_s_p = new b2Vec2();
b2ParticleSystem.SolveRigidDamping_s_v = new b2Vec2();
b2ParticleSystem.SolveExtraDamping_s_v = new b2Vec2();
b2ParticleSystem.SolveExtraDamping_s_f = new b2Vec2();
b2ParticleSystem.SolveRigid_s_position = new b2Vec2();
b2ParticleSystem.SolveRigid_s_rotation = new b2Rot();
b2ParticleSystem.SolveRigid_s_transform = new b2Transform();
b2ParticleSystem.SolveRigid_s_velocityTransform = new b2Transform();
b2ParticleSystem.SolveElastic_s_pa = new b2Vec2();
b2ParticleSystem.SolveElastic_s_pb = new b2Vec2();
b2ParticleSystem.SolveElastic_s_pc = new b2Vec2();
b2ParticleSystem.SolveElastic_s_r = new b2Rot();
b2ParticleSystem.SolveElastic_s_t0 = new b2Vec2();
b2ParticleSystem.SolveSpring_s_pa = new b2Vec2();
b2ParticleSystem.SolveSpring_s_pb = new b2Vec2();
b2ParticleSystem.SolveSpring_s_d = new b2Vec2();
b2ParticleSystem.SolveSpring_s_f = new b2Vec2();
b2ParticleSystem.SolveTensile_s_weightedNormal = new b2Vec2();
b2ParticleSystem.SolveTensile_s_s = new b2Vec2();
b2ParticleSystem.SolveTensile_s_f = new b2Vec2();
b2ParticleSystem.SolveViscous_s_v = new b2Vec2();
b2ParticleSystem.SolveViscous_s_f = new b2Vec2();
b2ParticleSystem.SolveRepulsive_s_f = new b2Vec2();
b2ParticleSystem.SolvePowder_s_f = new b2Vec2();
b2ParticleSystem.SolveSolid_s_f = new b2Vec2();
b2ParticleSystem.RemoveSpuriousBodyContacts_s_n = new b2Vec2();
b2ParticleSystem.RemoveSpuriousBodyContacts_s_pos = new b2Vec2();
b2ParticleSystem.RemoveSpuriousBodyContacts_s_normal = new b2Vec2();
return b2ParticleSystem;
}());
(function (b2ParticleSystem) {
var UserOverridableBuffer = /** @class */ (function () {
function UserOverridableBuffer() {
this.data = null;
this.userSuppliedCapacity = 0;
}
return UserOverridableBuffer;
}());
b2ParticleSystem.UserOverridableBuffer = UserOverridableBuffer;
var Proxy = /** @class */ (function () {
function Proxy() {
this.index = b2_invalidParticleIndex;
this.tag = 0;
}
Proxy.CompareProxyProxy = function (a, b) {
return a.tag < b.tag;
};
Proxy.CompareTagProxy = function (a, b) {
return a < b.tag;
};
Proxy.CompareProxyTag = function (a, b) {
return a.tag < b;
};
return Proxy;
}());
b2ParticleSystem.Proxy = Proxy;
var InsideBoundsEnumerator = /** @class */ (function () {
/**
* InsideBoundsEnumerator enumerates all particles inside the
* given bounds.
*
* Construct an enumerator with bounds of tags and a range of
* proxies.
*/
function InsideBoundsEnumerator(system, lower, upper, first, last) {
this.m_system = system;
this.m_xLower = (lower & b2ParticleSystem.xMask) >>> 0;
this.m_xUpper = (upper & b2ParticleSystem.xMask) >>> 0;
this.m_yLower = (lower & b2ParticleSystem.yMask) >>> 0;
this.m_yUpper = (upper & b2ParticleSystem.yMask) >>> 0;
this.m_first = first;
this.m_last = last;
// DEBUG: b2Assert(this.m_first <= this.m_last);
}
/**
* Get index of the next particle. Returns
* b2_invalidParticleIndex if there are no more particles.
*/
InsideBoundsEnumerator.prototype.GetNext = function () {
while (this.m_first < this.m_last) {
var xTag = (this.m_system.m_proxyBuffer.data[this.m_first].tag & b2ParticleSystem.xMask) >>> 0;
// #if B2_ASSERT_ENABLED
// DEBUG: const yTag = (this.m_system.m_proxyBuffer.data[this.m_first].tag & b2ParticleSystem.yMask) >>> 0;
// DEBUG: b2Assert(yTag >= this.m_yLower);
// DEBUG: b2Assert(yTag <= this.m_yUpper);
// #endif
if (xTag >= this.m_xLower && xTag <= this.m_xUpper) {
return (this.m_system.m_proxyBuffer.data[this.m_first++]).index;
}
this.m_first++;
}
return b2_invalidParticleIndex;
};
return InsideBoundsEnumerator;
}());
b2ParticleSystem.InsideBoundsEnumerator = InsideBoundsEnumerator;
var ParticleListNode = /** @class */ (function () {
function ParticleListNode() {
/**
* The next node in the list.
*/
this.next = null;
/**
* Number of entries in the list. Valid only for the node at the
* head of the list.
*/
this.count = 0;
/**
* Particle index.
*/
this.index = 0;
}
return ParticleListNode;
}());
b2ParticleSystem.ParticleListNode = ParticleListNode;
/**
* @constructor
*/
var FixedSetAllocator = /** @class */ (function () {
function FixedSetAllocator() {
}
FixedSetAllocator.prototype.Allocate = function (itemSize, count) {
// TODO
return count;
};
FixedSetAllocator.prototype.Clear = function () {
// TODO
};
FixedSetAllocator.prototype.GetCount = function () {
// TODO
return 0;
};
FixedSetAllocator.prototype.Invalidate = function (itemIndex) {
// TODO
};
FixedSetAllocator.prototype.GetValidBuffer = function () {
// TODO
return [];
};
FixedSetAllocator.prototype.GetBuffer = function () {
// TODO
return [];
};
FixedSetAllocator.prototype.SetCount = function (count) {
// TODO
};
return FixedSetAllocator;
}());
b2ParticleSystem.FixedSetAllocator = FixedSetAllocator;
var FixtureParticle = /** @class */ (function () {
function FixtureParticle(fixture, particle) {
this.second = b2_invalidParticleIndex;
this.first = fixture;
this.second = particle;
}
return FixtureParticle;
}());
b2ParticleSystem.FixtureParticle = FixtureParticle;
var FixtureParticleSet = /** @class */ (function (_super) {
__extends(FixtureParticleSet, _super);
function FixtureParticleSet() {
return _super !== null && _super.apply(this, arguments) || this;
}
FixtureParticleSet.prototype.Initialize = function (bodyContactBuffer, flagsBuffer) {
// TODO
};
FixtureParticleSet.prototype.Find = function (pair) {
// TODO
return b2_invalidParticleIndex;
};
return FixtureParticleSet;
}(b2ParticleSystem.FixedSetAllocator));
b2ParticleSystem.FixtureParticleSet = FixtureParticleSet;
var ParticlePair = /** @class */ (function () {
function ParticlePair(particleA, particleB) {
this.first = b2_invalidParticleIndex;
this.second = b2_invalidParticleIndex;
this.first = particleA;
this.second = particleB;
}
return ParticlePair;
}());
b2ParticleSystem.ParticlePair = ParticlePair;
var b2ParticlePairSet = /** @class */ (function (_super) {
__extends(b2ParticlePairSet, _super);
function b2ParticlePairSet() {
return _super !== null && _super.apply(this, arguments) || this;
}
b2ParticlePairSet.prototype.Initialize = function (contactBuffer, flagsBuffer) {
// TODO
};
b2ParticlePairSet.prototype.Find = function (pair) {
// TODO
return b2_invalidParticleIndex;
};
return b2ParticlePairSet;
}(b2ParticleSystem.FixedSetAllocator));
b2ParticleSystem.b2ParticlePairSet = b2ParticlePairSet;
var ConnectionFilter = /** @class */ (function () {
function ConnectionFilter() {
}
/**
* Is the particle necessary for connection?
* A pair or a triad should contain at least one 'necessary'
* particle.
*/
ConnectionFilter.prototype.IsNecessary = function (index) {
return true;
};
/**
* An additional condition for creating a pair.
*/
ConnectionFilter.prototype.ShouldCreatePair = function (a, b) {
return true;
};
/**
* An additional condition for creating a triad.
*/
ConnectionFilter.prototype.ShouldCreateTriad = function (a, b, c) {
return true;
};
return ConnectionFilter;
}());
b2ParticleSystem.ConnectionFilter = ConnectionFilter;
var DestroyParticlesInShapeCallback = /** @class */ (function (_super) {
__extends(DestroyParticlesInShapeCallback, _super);
function DestroyParticlesInShapeCallback(system, shape, xf, callDestructionListener) {
var _this = _super.call(this) || this;
_this.m_callDestructionListener = false;
_this.m_destroyed = 0;
_this.m_system = system;
_this.m_shape = shape;
_this.m_xf = xf;
_this.m_callDestructionListener = callDestructionListener;
_this.m_destroyed = 0;
return _this;
}
DestroyParticlesInShapeCallback.prototype.ReportFixture = function (fixture) {
return false;
};
DestroyParticlesInShapeCallback.prototype.ReportParticle = function (particleSystem, index) {
if (particleSystem !== this.m_system) {
return false;
}
// DEBUG: b2Assert(index >= 0 && index < this.m_system.m_count);
if (!this.m_system.m_positionBuffer.data) {
throw new Error();
}
if (this.m_shape.TestPoint(this.m_xf, this.m_system.m_positionBuffer.data[index])) {
this.m_system.DestroyParticle(index, this.m_callDestructionListener);
this.m_destroyed++;
}
return true;
};
DestroyParticlesInShapeCallback.prototype.Destroyed = function () {
return this.m_destroyed;
};
return DestroyParticlesInShapeCallback;
}(b2QueryCallback));
b2ParticleSystem.DestroyParticlesInShapeCallback = DestroyParticlesInShapeCallback;
var JoinParticleGroupsFilter = /** @class */ (function (_super) {
__extends(JoinParticleGroupsFilter, _super);
function JoinParticleGroupsFilter(threshold) {
var _this = _super.call(this) || this;
_this.m_threshold = 0;
_this.m_threshold = threshold;
return _this;
}
/**
* An additional condition for creating a pair.
*/
JoinParticleGroupsFilter.prototype.ShouldCreatePair = function (a, b) {
return (a < this.m_threshold && this.m_threshold <= b) ||
(b < this.m_threshold && this.m_threshold <= a);
};
/**
* An additional condition for creating a triad.
*/
JoinParticleGroupsFilter.prototype.ShouldCreateTriad = function (a, b, c) {
return (a < this.m_threshold || b < this.m_threshold || c < this.m_threshold) &&
(this.m_threshold <= a || this.m_threshold <= b || this.m_threshold <= c);
};
return JoinParticleGroupsFilter;
}(b2ParticleSystem.ConnectionFilter));
b2ParticleSystem.JoinParticleGroupsFilter = JoinParticleGroupsFilter;
var CompositeShape = /** @class */ (function (_super) {
__extends(CompositeShape, _super);
function CompositeShape(shapes, shapeCount) {
if (shapeCount === void 0) { shapeCount = shapes.length; }
var _this = _super.call(this, exports.b2ShapeType.e_unknown, 0) || this;
_this.m_shapeCount = 0;
_this.m_shapes = shapes;
_this.m_shapeCount = shapeCount;
return _this;
}
CompositeShape.prototype.Clone = function () {
// DEBUG: b2Assert(false);
throw new Error();
};
CompositeShape.prototype.GetChildCount = function () {
return 1;
};
/**
* @see b2Shape::TestPoint
*/
CompositeShape.prototype.TestPoint = function (xf, p) {
for (var i = 0; i < this.m_shapeCount; i++) {
if (this.m_shapes[i].TestPoint(xf, p)) {
return true;
}
}
return false;
};
/**
* @see b2Shape::ComputeDistance
*/
CompositeShape.prototype.ComputeDistance = function (xf, p, normal, childIndex) {
// DEBUG: b2Assert(false);
return 0;
};
/**
* Implement b2Shape.
*/
CompositeShape.prototype.RayCast = function (output, input, xf, childIndex) {
// DEBUG: b2Assert(false);
return false;
};
/**
* @see b2Shape::ComputeAABB
*/
CompositeShape.prototype.ComputeAABB = function (aabb, xf, childIndex) {
var s_subaabb = new b2AABB();
aabb.lowerBound.x = +b2_maxFloat;
aabb.lowerBound.y = +b2_maxFloat;
aabb.upperBound.x = -b2_maxFloat;
aabb.upperBound.y = -b2_maxFloat;
// DEBUG: b2Assert(childIndex === 0);
for (var i = 0; i < this.m_shapeCount; i++) {
var childCount = this.m_shapes[i].GetChildCount();
for (var j = 0; j < childCount; j++) {
var subaabb = s_subaabb;
this.m_shapes[i].ComputeAABB(subaabb, xf, j);
aabb.Combine1(subaabb);
}
}
};
/**
* @see b2Shape::ComputeMass
*/
CompositeShape.prototype.ComputeMass = function (massData, density) {
// DEBUG: b2Assert(false);
};
CompositeShape.prototype.SetupDistanceProxy = function (proxy, index) {
// DEBUG: b2Assert(false);
};
CompositeShape.prototype.ComputeSubmergedArea = function (normal, offset, xf, c) {
// DEBUG: b2Assert(false);
return 0;
};
CompositeShape.prototype.Dump = function (log) {
// DEBUG: b2Assert(false);
};
return CompositeShape;
}(b2Shape));
b2ParticleSystem.CompositeShape = CompositeShape;
var ReactiveFilter = /** @class */ (function (_super) {
__extends(ReactiveFilter, _super);
function ReactiveFilter(flagsBuffer) {
var _this = _super.call(this) || this;
_this.m_flagsBuffer = flagsBuffer;
return _this;
}
ReactiveFilter.prototype.IsNecessary = function (index) {
if (!this.m_flagsBuffer.data) {
throw new Error();
}
return (this.m_flagsBuffer.data[index] & exports.b2ParticleFlag.b2_reactiveParticle) !== 0;
};
return ReactiveFilter;
}(b2ParticleSystem.ConnectionFilter));
b2ParticleSystem.ReactiveFilter = ReactiveFilter;
var UpdateBodyContactsCallback = /** @class */ (function (_super) {
__extends(UpdateBodyContactsCallback, _super);
function UpdateBodyContactsCallback(system, contactFilter) {
var _this = _super.call(this, system) || this;
_this.m_contactFilter = contactFilter;
return _this;
}
UpdateBodyContactsCallback.prototype.ShouldCollideFixtureParticle = function (fixture, particleSystem, particleIndex) {
// Call the contact filter if it's set, to determine whether to
// filter this contact. Returns true if contact calculations should
// be performed, false otherwise.
if (this.m_contactFilter) {
var flags = this.m_system.GetFlagsBuffer();
if (flags[particleIndex] & exports.b2ParticleFlag.b2_fixtureContactFilterParticle) {
return this.m_contactFilter.ShouldCollideFixtureParticle(fixture, this.m_system, particleIndex);
}
}
return true;
};
UpdateBodyContactsCallback.prototype.ReportFixtureAndParticle = function (fixture, childIndex, a) {
var s_n = b2ParticleSystem.UpdateBodyContactsCallback.ReportFixtureAndParticle_s_n;
var s_rp = b2ParticleSystem.UpdateBodyContactsCallback.ReportFixtureAndParticle_s_rp;
if (!this.m_system.m_flagsBuffer.data) {
throw new Error();
}
if (!this.m_system.m_positionBuffer.data) {
throw new Error();
}
var ap = this.m_system.m_positionBuffer.data[a];
var n = s_n;
var d = fixture.ComputeDistance(ap, n, childIndex);
if (d < this.m_system.m_particleDiameter && this.ShouldCollideFixtureParticle(fixture, this.m_system, a)) {
var b = fixture.GetBody();
var bp = b.GetWorldCenter();
var bm = b.GetMass();
var bI = b.GetInertia() - bm * b.GetLocalCenter().LengthSquared();
var invBm = bm > 0 ? 1 / bm : 0;
var invBI = bI > 0 ? 1 / bI : 0;
var invAm = this.m_system.m_flagsBuffer.data[a] &
exports.b2ParticleFlag.b2_wallParticle ? 0 : this.m_system.GetParticleInvMass();
///b2Vec2 rp = ap - bp;
var rp = b2Vec2.SubVV(ap, bp, s_rp);
var rpn = b2Vec2.CrossVV(rp, n);
var invM = invAm + invBm + invBI * rpn * rpn;
///b2ParticleBodyContact& contact = m_system.m_bodyContactBuffer.Append();
var contact = this.m_system.m_bodyContactBuffer.data[this.m_system.m_bodyContactBuffer.Append()];
contact.index = a;
contact.body = b;
contact.fixture = fixture;
contact.weight = 1 - d * this.m_system.m_inverseDiameter;
///contact.normal = -n;
contact.normal.Copy(n.SelfNeg());
contact.mass = invM > 0 ? 1 / invM : 0;
this.m_system.DetectStuckParticle(a);
}
};
UpdateBodyContactsCallback.ReportFixtureAndParticle_s_n = new b2Vec2();
UpdateBodyContactsCallback.ReportFixtureAndParticle_s_rp = new b2Vec2();
return UpdateBodyContactsCallback;
}(b2FixtureParticleQueryCallback));
b2ParticleSystem.UpdateBodyContactsCallback = UpdateBodyContactsCallback;
var SolveCollisionCallback = /** @class */ (function (_super) {
__extends(SolveCollisionCallback, _super);
function SolveCollisionCallback(system, step) {
var _this = _super.call(this, system) || this;
_this.m_step = step;
return _this;
}
SolveCollisionCallback.prototype.ReportFixtureAndParticle = function (fixture, childIndex, a) {
var s_p1 = b2ParticleSystem.SolveCollisionCallback.ReportFixtureAndParticle_s_p1;
var s_output = b2ParticleSystem.SolveCollisionCallback.ReportFixtureAndParticle_s_output;
var s_input = b2ParticleSystem.SolveCollisionCallback.ReportFixtureAndParticle_s_input;
var s_p = b2ParticleSystem.SolveCollisionCallback.ReportFixtureAndParticle_s_p;
var s_v = b2ParticleSystem.SolveCollisionCallback.ReportFixtureAndParticle_s_v;
var s_f = b2ParticleSystem.SolveCollisionCallback.ReportFixtureAndParticle_s_f;
var body = fixture.GetBody();
if (!this.m_system.m_positionBuffer.data) {
throw new Error();
}
if (!this.m_system.m_velocityBuffer.data) {
throw new Error();
}
var ap = this.m_system.m_positionBuffer.data[a];
var av = this.m_system.m_velocityBuffer.data[a];
var output = s_output;
var input = s_input;
if (this.m_system.m_iterationIndex === 0) {
// Put 'ap' in the local space of the previous frame
///b2Vec2 p1 = b2MulT(body.m_xf0, ap);
var p1 = b2Transform.MulTXV(body.m_xf0, ap, s_p1);
if (fixture.GetShape().GetType() === exports.b2ShapeType.e_circleShape) {
// Make relative to the center of the circle
///p1 -= body.GetLocalCenter();
p1.SelfSub(body.GetLocalCenter());
// Re-apply rotation about the center of the circle
///p1 = b2Mul(body.m_xf0.q, p1);
b2Rot.MulRV(body.m_xf0.q, p1, p1);
// Subtract rotation of the current frame
///p1 = b2MulT(body.m_xf.q, p1);
b2Rot.MulTRV(body.m_xf.q, p1, p1);
// Return to local space
///p1 += body.GetLocalCenter();
p1.SelfAdd(body.GetLocalCenter());
}
// Return to global space and apply rotation of current frame
///input.p1 = b2Mul(body.m_xf, p1);
b2Transform.MulXV(body.m_xf, p1, input.p1);
}
else {
///input.p1 = ap;
input.p1.Copy(ap);
}
///input.p2 = ap + m_step.dt * av;
b2Vec2.AddVMulSV(ap, this.m_step.dt, av, input.p2);
input.maxFraction = 1;
if (fixture.RayCast(output, input, childIndex)) {
var n = output.normal;
///b2Vec2 p = (1 - output.fraction) * input.p1 + output.fraction * input.p2 + b2_linearSlop * n;
var p = s_p;
p.x = (1 - output.fraction) * input.p1.x + output.fraction * input.p2.x + b2_linearSlop * n.x;
p.y = (1 - output.fraction) * input.p1.y + output.fraction * input.p2.y + b2_linearSlop * n.y;
///b2Vec2 v = m_step.inv_dt * (p - ap);
var v = s_v;
v.x = this.m_step.inv_dt * (p.x - ap.x);
v.y = this.m_step.inv_dt * (p.y - ap.y);
///m_system.m_velocityBuffer.data[a] = v;
this.m_system.m_velocityBuffer.data[a].Copy(v);
///b2Vec2 f = m_step.inv_dt * m_system.GetParticleMass() * (av - v);
var f = s_f;
f.x = this.m_step.inv_dt * this.m_system.GetParticleMass() * (av.x - v.x);
f.y = this.m_step.inv_dt * this.m_system.GetParticleMass() * (av.y - v.y);
this.m_system.ParticleApplyForce(a, f);
}
};
SolveCollisionCallback.prototype.ReportParticle = function (system, index) {
return false;
};
SolveCollisionCallback.ReportFixtureAndParticle_s_p1 = new b2Vec2();
SolveCollisionCallback.ReportFixtureAndParticle_s_output = new b2RayCastOutput();
SolveCollisionCallback.ReportFixtureAndParticle_s_input = new b2RayCastInput();
SolveCollisionCallback.ReportFixtureAndParticle_s_p = new b2Vec2();
SolveCollisionCallback.ReportFixtureAndParticle_s_v = new b2Vec2();
SolveCollisionCallback.ReportFixtureAndParticle_s_f = new b2Vec2();
return SolveCollisionCallback;
}(b2FixtureParticleQueryCallback));
b2ParticleSystem.SolveCollisionCallback = SolveCollisionCallback;
})(exports.b2ParticleSystem || (exports.b2ParticleSystem = {}));
// #endif
/*
* Copyright (c) 2006-2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
// #endif
/// The world class manages all physics entities, dynamic simulation,
/// and asynchronous queries. The world also contains efficient memory
/// management facilities.
var b2World = /** @class */ (function () {
// #endif
/// Construct a world object.
/// @param gravity the world gravity vector.
function b2World(gravity) {
// b2BlockAllocator m_blockAllocator;
// b2StackAllocator m_stackAllocator;
this.m_newFixture = false;
this.m_locked = false;
this.m_clearForces = true;
this.m_contactManager = new b2ContactManager();
this.m_bodyList = null;
this.m_jointList = null;
// #if B2_ENABLE_PARTICLE
this.m_particleSystemList = null;
// #endif
this.m_bodyCount = 0;
this.m_jointCount = 0;
this.m_gravity = new b2Vec2();
this.m_allowSleep = true;
this.m_destructionListener = null;
this.m_debugDraw = null;
// This is used to compute the time step ratio to
// support a variable time step.
this.m_inv_dt0 = 0;
// These are for debugging the solver.
this.m_warmStarting = true;
this.m_continuousPhysics = true;
this.m_subStepping = false;
this.m_stepComplete = true;
this.m_profile = new b2Profile();
this.m_island = new b2Island();
this.s_stack = [];
// #if B2_ENABLE_CONTROLLER
this.m_controllerList = null;
this.m_controllerCount = 0;
this.m_gravity.Copy(gravity);
}
/// Register a destruction listener. The listener is owned by you and must
/// remain in scope.
b2World.prototype.SetDestructionListener = function (listener) {
this.m_destructionListener = listener;
};
/// Register a contact filter to provide specific control over collision.
/// Otherwise the default filter is used (b2_defaultFilter). The listener is
/// owned by you and must remain in scope.
b2World.prototype.SetContactFilter = function (filter) {
this.m_contactManager.m_contactFilter = filter;
};
/// Register a contact event listener. The listener is owned by you and must
/// remain in scope.
b2World.prototype.SetContactListener = function (listener) {
this.m_contactManager.m_contactListener = listener;
};
/// Register a routine for debug drawing. The debug draw functions are called
/// inside with b2World::DrawDebugData method. The debug draw object is owned
/// by you and must remain in scope.
b2World.prototype.SetDebugDraw = function (debugDraw) {
this.m_debugDraw = debugDraw;
};
/// Create a rigid body given a definition. No reference to the definition
/// is retained.
/// @warning This function is locked during callbacks.
b2World.prototype.CreateBody = function (def) {
if (def === void 0) { def = {}; }
if (this.IsLocked()) {
throw new Error();
}
var b = new b2Body(def, this);
// Add to world doubly linked list.
b.m_prev = null;
b.m_next = this.m_bodyList;
if (this.m_bodyList) {
this.m_bodyList.m_prev = b;
}
this.m_bodyList = b;
++this.m_bodyCount;
return b;
};
/// Destroy a rigid body given a definition. No reference to the definition
/// is retained. This function is locked during callbacks.
/// @warning This automatically deletes all associated shapes and joints.
/// @warning This function is locked during callbacks.
b2World.prototype.DestroyBody = function (b) {
// DEBUG: b2Assert(this.m_bodyCount > 0);
if (this.IsLocked()) {
throw new Error();
}
// Delete the attached joints.
var je = b.m_jointList;
while (je) {
var je0 = je;
je = je.next;
if (this.m_destructionListener) {
this.m_destructionListener.SayGoodbyeJoint(je0.joint);
}
this.DestroyJoint(je0.joint);
b.m_jointList = je;
}
b.m_jointList = null;
// #if B2_ENABLE_CONTROLLER
// @see b2Controller list
var coe = b.m_controllerList;
while (coe) {
var coe0 = coe;
coe = coe.nextController;
coe0.controller.RemoveBody(b);
}
// #endif
// Delete the attached contacts.
var ce = b.m_contactList;
while (ce) {
var ce0 = ce;
ce = ce.next;
this.m_contactManager.Destroy(ce0.contact);
}
b.m_contactList = null;
// Delete the attached fixtures. This destroys broad-phase proxies.
var f = b.m_fixtureList;
while (f) {
var f0 = f;
f = f.m_next;
if (this.m_destructionListener) {
this.m_destructionListener.SayGoodbyeFixture(f0);
}
f0.DestroyProxies();
f0.Destroy();
b.m_fixtureList = f;
b.m_fixtureCount -= 1;
}
b.m_fixtureList = null;
b.m_fixtureCount = 0;
// Remove world body list.
if (b.m_prev) {
b.m_prev.m_next = b.m_next;
}
if (b.m_next) {
b.m_next.m_prev = b.m_prev;
}
if (b === this.m_bodyList) {
this.m_bodyList = b.m_next;
}
--this.m_bodyCount;
};
b2World._Joint_Create = function (def, allocator) {
switch (def.type) {
case exports.b2JointType.e_distanceJoint: return new b2DistanceJoint(def);
case exports.b2JointType.e_mouseJoint: return new b2MouseJoint(def);
case exports.b2JointType.e_prismaticJoint: return new b2PrismaticJoint(def);
case exports.b2JointType.e_revoluteJoint: return new b2RevoluteJoint(def);
case exports.b2JointType.e_pulleyJoint: return new b2PulleyJoint(def);
case exports.b2JointType.e_gearJoint: return new b2GearJoint(def);
case exports.b2JointType.e_wheelJoint: return new b2WheelJoint(def);
case exports.b2JointType.e_weldJoint: return new b2WeldJoint(def);
case exports.b2JointType.e_frictionJoint: return new b2FrictionJoint(def);
case exports.b2JointType.e_ropeJoint: return new b2RopeJoint(def);
case exports.b2JointType.e_motorJoint: return new b2MotorJoint(def);
case exports.b2JointType.e_areaJoint: return new b2AreaJoint(def);
}
throw new Error();
};
b2World._Joint_Destroy = function (joint, allocator) {
};
b2World.prototype.CreateJoint = function (def) {
if (this.IsLocked()) {
throw new Error();
}
var j = b2World._Joint_Create(def, null);
// Connect to the world list.
j.m_prev = null;
j.m_next = this.m_jointList;
if (this.m_jointList) {
this.m_jointList.m_prev = j;
}
this.m_jointList = j;
++this.m_jointCount;
// Connect to the bodies' doubly linked lists.
// j.m_edgeA.joint = j;
// j.m_edgeA.other = j.m_bodyB;
j.m_edgeA.prev = null;
j.m_edgeA.next = j.m_bodyA.m_jointList;
if (j.m_bodyA.m_jointList) {
j.m_bodyA.m_jointList.prev = j.m_edgeA;
}
j.m_bodyA.m_jointList = j.m_edgeA;
// j.m_edgeB.joint = j;
// j.m_edgeB.other = j.m_bodyA;
j.m_edgeB.prev = null;
j.m_edgeB.next = j.m_bodyB.m_jointList;
if (j.m_bodyB.m_jointList) {
j.m_bodyB.m_jointList.prev = j.m_edgeB;
}
j.m_bodyB.m_jointList = j.m_edgeB;
var bodyA = def.bodyA;
var bodyB = def.bodyB;
// If the joint prevents collisions, then flag any contacts for filtering.
if (!def.collideConnected) {
var edge = bodyB.GetContactList();
while (edge) {
if (edge.other === bodyA) {
// Flag the contact for filtering at the next time step (where either
// body is awake).
edge.contact.FlagForFiltering();
}
edge = edge.next;
}
}
// Note: creating a joint doesn't wake the bodies.
return j;
};
/// Destroy a joint. This may cause the connected bodies to begin colliding.
/// @warning This function is locked during callbacks.
b2World.prototype.DestroyJoint = function (j) {
if (this.IsLocked()) {
throw new Error();
}
var collideConnected = j.m_collideConnected;
// Remove from the doubly linked list.
if (j.m_prev) {
j.m_prev.m_next = j.m_next;
}
if (j.m_next) {
j.m_next.m_prev = j.m_prev;
}
if (j === this.m_jointList) {
this.m_jointList = j.m_next;
}
// Disconnect from island graph.
var bodyA = j.m_bodyA;
var bodyB = j.m_bodyB;
// Wake up connected bodies.
bodyA.SetAwake(true);
bodyB.SetAwake(true);
// Remove from body 1.
if (j.m_edgeA.prev) {
j.m_edgeA.prev.next = j.m_edgeA.next;
}
if (j.m_edgeA.next) {
j.m_edgeA.next.prev = j.m_edgeA.prev;
}
if (j.m_edgeA === bodyA.m_jointList) {
bodyA.m_jointList = j.m_edgeA.next;
}
j.m_edgeA.prev = null;
j.m_edgeA.next = null;
// Remove from body 2
if (j.m_edgeB.prev) {
j.m_edgeB.prev.next = j.m_edgeB.next;
}
if (j.m_edgeB.next) {
j.m_edgeB.next.prev = j.m_edgeB.prev;
}
if (j.m_edgeB === bodyB.m_jointList) {
bodyB.m_jointList = j.m_edgeB.next;
}
j.m_edgeB.prev = null;
j.m_edgeB.next = null;
b2World._Joint_Destroy(j, null);
// DEBUG: b2Assert(this.m_jointCount > 0);
--this.m_jointCount;
// If the joint prevents collisions, then flag any contacts for filtering.
if (!collideConnected) {
var edge = bodyB.GetContactList();
while (edge) {
if (edge.other === bodyA) {
// Flag the contact for filtering at the next time step (where either
// body is awake).
edge.contact.FlagForFiltering();
}
edge = edge.next;
}
}
};
// #if B2_ENABLE_PARTICLE
b2World.prototype.CreateParticleSystem = function (def) {
if (this.IsLocked()) {
throw new Error();
}
var p = new exports.b2ParticleSystem(def, this);
// Add to world doubly linked list.
p.m_prev = null;
p.m_next = this.m_particleSystemList;
if (this.m_particleSystemList) {
this.m_particleSystemList.m_prev = p;
}
this.m_particleSystemList = p;
return p;
};
b2World.prototype.DestroyParticleSystem = function (p) {
if (this.IsLocked()) {
throw new Error();
}
// Remove world particleSystem list.
if (p.m_prev) {
p.m_prev.m_next = p.m_next;
}
if (p.m_next) {
p.m_next.m_prev = p.m_prev;
}
if (p === this.m_particleSystemList) {
this.m_particleSystemList = p.m_next;
}
};
b2World.prototype.CalculateReasonableParticleIterations = function (timeStep) {
if (this.m_particleSystemList === null) {
return 1;
}
function GetSmallestRadius(world) {
var smallestRadius = b2_maxFloat;
for (var system = world.GetParticleSystemList(); system !== null; system = system.m_next) {
smallestRadius = b2Min(smallestRadius, system.GetRadius());
}
return smallestRadius;
}
// Use the smallest radius, since that represents the worst-case.
return b2CalculateParticleIterations(this.m_gravity.Length(), GetSmallestRadius(this), timeStep);
};
// #if B2_ENABLE_PARTICLE
b2World.prototype.Step = function (dt, velocityIterations, positionIterations, particleIterations) {
if (particleIterations === void 0) { particleIterations = this.CalculateReasonableParticleIterations(dt); }
// #else
// public Step(dt: number, velocityIterations: number, positionIterations: number): void {
// #endif
var stepTimer = b2World.Step_s_stepTimer.Reset();
// If new fixtures were added, we need to find the new contacts.
if (this.m_newFixture) {
this.m_contactManager.FindNewContacts();
this.m_newFixture = false;
}
this.m_locked = true;
var step = b2World.Step_s_step;
step.dt = dt;
step.velocityIterations = velocityIterations;
step.positionIterations = positionIterations;
// #if B2_ENABLE_PARTICLE
step.particleIterations = particleIterations;
// #endif
if (dt > 0) {
step.inv_dt = 1 / dt;
}
else {
step.inv_dt = 0;
}
step.dtRatio = this.m_inv_dt0 * dt;
step.warmStarting = this.m_warmStarting;
// Update contacts. This is where some contacts are destroyed.
var timer = b2World.Step_s_timer.Reset();
this.m_contactManager.Collide();
this.m_profile.collide = timer.GetMilliseconds();
// Integrate velocities, solve velocity constraints, and integrate positions.
if (this.m_stepComplete && step.dt > 0) {
var timer_1 = b2World.Step_s_timer.Reset();
// #if B2_ENABLE_PARTICLE
for (var p = this.m_particleSystemList; p; p = p.m_next) {
p.Solve(step); // Particle Simulation
}
// #endif
this.Solve(step);
this.m_profile.solve = timer_1.GetMilliseconds();
}
// Handle TOI events.
if (this.m_continuousPhysics && step.dt > 0) {
var timer_2 = b2World.Step_s_timer.Reset();
this.SolveTOI(step);
this.m_profile.solveTOI = timer_2.GetMilliseconds();
}
if (step.dt > 0) {
this.m_inv_dt0 = step.inv_dt;
}
if (this.m_clearForces) {
this.ClearForces();
}
this.m_locked = false;
this.m_profile.step = stepTimer.GetMilliseconds();
};
/// Manually clear the force buffer on all bodies. By default, forces are cleared automatically
/// after each call to Step. The default behavior is modified by calling SetAutoClearForces.
/// The purpose of this function is to support sub-stepping. Sub-stepping is often used to maintain
/// a fixed sized time step under a variable frame-rate.
/// When you perform sub-stepping you will disable auto clearing of forces and instead call
/// ClearForces after all sub-steps are complete in one pass of your game loop.
/// @see SetAutoClearForces
b2World.prototype.ClearForces = function () {
for (var body = this.m_bodyList; body; body = body.m_next) {
body.m_force.SetZero();
body.m_torque = 0;
}
};
// #if B2_ENABLE_PARTICLE
b2World.prototype.DrawParticleSystem = function (system) {
if (this.m_debugDraw === null) {
return;
}
var particleCount = system.GetParticleCount();
if (particleCount) {
var radius = system.GetRadius();
var positionBuffer = system.GetPositionBuffer();
if (system.m_colorBuffer.data) {
var colorBuffer = system.GetColorBuffer();
this.m_debugDraw.DrawParticles(positionBuffer, radius, colorBuffer, particleCount);
}
else {
this.m_debugDraw.DrawParticles(positionBuffer, radius, null, particleCount);
}
}
};
b2World.prototype.DrawDebugData = function () {
if (this.m_debugDraw === null) {
return;
}
var flags = this.m_debugDraw.GetFlags();
var color = b2World.DrawDebugData_s_color.SetRGB(0, 0, 0);
if (flags & exports.b2DrawFlags.e_shapeBit) {
for (var b = this.m_bodyList; b; b = b.m_next) {
var xf = b.m_xf;
this.m_debugDraw.PushTransform(xf);
for (var f = b.GetFixtureList(); f; f = f.m_next) {
if (!b.IsActive()) {
color.SetRGB(0.5, 0.5, 0.3);
this.DrawShape(f, color);
}
else if (b.GetType() === exports.b2BodyType.b2_staticBody) {
color.SetRGB(0.5, 0.9, 0.5);
this.DrawShape(f, color);
}
else if (b.GetType() === exports.b2BodyType.b2_kinematicBody) {
color.SetRGB(0.5, 0.5, 0.9);
this.DrawShape(f, color);
}
else if (!b.IsAwake()) {
color.SetRGB(0.6, 0.6, 0.6);
this.DrawShape(f, color);
}
else {
color.SetRGB(0.9, 0.7, 0.7);
this.DrawShape(f, color);
}
}
this.m_debugDraw.PopTransform(xf);
}
}
// #if B2_ENABLE_PARTICLE
if (flags & exports.b2DrawFlags.e_particleBit) {
for (var p = this.m_particleSystemList; p; p = p.m_next) {
this.DrawParticleSystem(p);
}
}
// #endif
if (flags & exports.b2DrawFlags.e_jointBit) {
for (var j = this.m_jointList; j; j = j.m_next) {
this.DrawJoint(j);
}
}
/*
if (flags & b2DrawFlags.e_pairBit) {
color.SetRGB(0.3, 0.9, 0.9);
for (let contact = this.m_contactManager.m_contactList; contact; contact = contact.m_next) {
const fixtureA = contact.GetFixtureA();
const fixtureB = contact.GetFixtureB();
const cA = fixtureA.GetAABB().GetCenter();
const cB = fixtureB.GetAABB().GetCenter();
this.m_debugDraw.DrawSegment(cA, cB, color);
}
}
*/
if (flags & exports.b2DrawFlags.e_aabbBit) {
color.SetRGB(0.9, 0.3, 0.9);
var vs = b2World.DrawDebugData_s_vs;
for (var b = this.m_bodyList; b; b = b.m_next) {
if (!b.IsActive()) {
continue;
}
for (var f = b.GetFixtureList(); f; f = f.m_next) {
for (var i = 0; i < f.m_proxyCount; ++i) {
var proxy = f.m_proxies[i];
var aabb = proxy.treeNode.aabb;
vs[0].Set(aabb.lowerBound.x, aabb.lowerBound.y);
vs[1].Set(aabb.upperBound.x, aabb.lowerBound.y);
vs[2].Set(aabb.upperBound.x, aabb.upperBound.y);
vs[3].Set(aabb.lowerBound.x, aabb.upperBound.y);
this.m_debugDraw.DrawPolygon(vs, 4, color);
}
}
}
}
if (flags & exports.b2DrawFlags.e_centerOfMassBit) {
for (var b = this.m_bodyList; b; b = b.m_next) {
var xf = b2World.DrawDebugData_s_xf;
xf.q.Copy(b.m_xf.q);
xf.p.Copy(b.GetWorldCenter());
this.m_debugDraw.DrawTransform(xf);
}
}
// #if B2_ENABLE_CONTROLLER
// @see b2Controller list
if (flags & exports.b2DrawFlags.e_controllerBit) {
for (var c = this.m_controllerList; c; c = c.m_next) {
c.Draw(this.m_debugDraw);
}
}
// #endif
};
/// Query the world for all fixtures that potentially overlap the
/// provided AABB.
/// @param callback a user implemented callback class.
/// @param aabb the query box.
b2World.prototype.QueryAABB = function (callback, aabb, fn) {
this.m_contactManager.m_broadPhase.Query(aabb, function (proxy) {
var fixture_proxy = proxy.userData;
// DEBUG: b2Assert(fixture_proxy instanceof b2FixtureProxy);
var fixture = fixture_proxy.fixture;
if (callback) {
return callback.ReportFixture(fixture);
}
else if (fn) {
return fn(fixture);
}
return true;
});
// #if B2_ENABLE_PARTICLE
if (callback instanceof b2QueryCallback) {
for (var p = this.m_particleSystemList; p; p = p.m_next) {
if (callback.ShouldQueryParticleSystem(p)) {
p.QueryAABB(callback, aabb);
}
}
}
// #endif
};
b2World.prototype.QueryAllAABB = function (aabb, out) {
if (out === void 0) { out = []; }
this.QueryAABB(null, aabb, function (fixture) { out.push(fixture); return true; });
return out;
};
/// Query the world for all fixtures that potentially overlap the
/// provided point.
/// @param callback a user implemented callback class.
/// @param point the query point.
b2World.prototype.QueryPointAABB = function (callback, point, fn) {
this.m_contactManager.m_broadPhase.QueryPoint(point, function (proxy) {
var fixture_proxy = proxy.userData;
// DEBUG: b2Assert(fixture_proxy instanceof b2FixtureProxy);
var fixture = fixture_proxy.fixture;
if (callback) {
return callback.ReportFixture(fixture);
}
else if (fn) {
return fn(fixture);
}
return true;
});
// #if B2_ENABLE_PARTICLE
if (callback instanceof b2QueryCallback) {
for (var p = this.m_particleSystemList; p; p = p.m_next) {
if (callback.ShouldQueryParticleSystem(p)) {
p.QueryPointAABB(callback, point);
}
}
}
// #endif
};
b2World.prototype.QueryAllPointAABB = function (point, out) {
if (out === void 0) { out = []; }
this.QueryPointAABB(null, point, function (fixture) { out.push(fixture); return true; });
return out;
};
b2World.prototype.QueryFixtureShape = function (callback, shape, index, transform, fn) {
var aabb = b2World.QueryFixtureShape_s_aabb;
shape.ComputeAABB(aabb, transform, index);
this.m_contactManager.m_broadPhase.Query(aabb, function (proxy) {
var fixture_proxy = proxy.userData;
// DEBUG: b2Assert(fixture_proxy instanceof b2FixtureProxy);
var fixture = fixture_proxy.fixture;
if (b2TestOverlapShape(shape, index, fixture.GetShape(), fixture_proxy.childIndex, transform, fixture.GetBody().GetTransform())) {
if (callback) {
return callback.ReportFixture(fixture);
}
else if (fn) {
return fn(fixture);
}
}
return true;
});
// #if B2_ENABLE_PARTICLE
if (callback instanceof b2QueryCallback) {
for (var p = this.m_particleSystemList; p; p = p.m_next) {
if (callback.ShouldQueryParticleSystem(p)) {
p.QueryAABB(callback, aabb);
}
}
}
// #endif
};
b2World.prototype.QueryAllFixtureShape = function (shape, index, transform, out) {
if (out === void 0) { out = []; }
this.QueryFixtureShape(null, shape, index, transform, function (fixture) { out.push(fixture); return true; });
return out;
};
b2World.prototype.QueryFixturePoint = function (callback, point, fn) {
this.m_contactManager.m_broadPhase.QueryPoint(point, function (proxy) {
var fixture_proxy = proxy.userData;
// DEBUG: b2Assert(fixture_proxy instanceof b2FixtureProxy);
var fixture = fixture_proxy.fixture;
if (fixture.TestPoint(point)) {
if (callback) {
return callback.ReportFixture(fixture);
}
else if (fn) {
return fn(fixture);
}
}
return true;
});
// #if B2_ENABLE_PARTICLE
if (callback) {
for (var p = this.m_particleSystemList; p; p = p.m_next) {
if (callback.ShouldQueryParticleSystem(p)) {
p.QueryPointAABB(callback, point);
}
}
}
// #endif
};
b2World.prototype.QueryAllFixturePoint = function (point, out) {
if (out === void 0) { out = []; }
this.QueryFixturePoint(null, point, function (fixture) { out.push(fixture); return true; });
return out;
};
b2World.prototype.RayCast = function (callback, point1, point2, fn) {
var input = b2World.RayCast_s_input;
input.maxFraction = 1;
input.p1.Copy(point1);
input.p2.Copy(point2);
this.m_contactManager.m_broadPhase.RayCast(input, function (input, proxy) {
var fixture_proxy = proxy.userData;
// DEBUG: b2Assert(fixture_proxy instanceof b2FixtureProxy);
var fixture = fixture_proxy.fixture;
var index = fixture_proxy.childIndex;
var output = b2World.RayCast_s_output;
var hit = fixture.RayCast(output, input, index);
if (hit) {
var fraction = output.fraction;
var point = b2World.RayCast_s_point;
point.Set((1 - fraction) * point1.x + fraction * point2.x, (1 - fraction) * point1.y + fraction * point2.y);
if (callback) {
return callback.ReportFixture(fixture, point, output.normal, fraction);
}
else if (fn) {
return fn(fixture, point, output.normal, fraction);
}
}
return input.maxFraction;
});
// #if B2_ENABLE_PARTICLE
if (callback) {
for (var p = this.m_particleSystemList; p; p = p.m_next) {
if (callback.ShouldQueryParticleSystem(p)) {
p.RayCast(callback, point1, point2);
}
}
}
// #endif
};
b2World.prototype.RayCastOne = function (point1, point2) {
var result = null;
var min_fraction = 1;
this.RayCast(null, point1, point2, function (fixture, point, normal, fraction) {
if (fraction < min_fraction) {
min_fraction = fraction;
result = fixture;
}
return min_fraction;
});
return result;
};
b2World.prototype.RayCastAll = function (point1, point2, out) {
if (out === void 0) { out = []; }
this.RayCast(null, point1, point2, function (fixture, point, normal, fraction) {
out.push(fixture);
return 1;
});
return out;
};
/// Get the world body list. With the returned body, use b2Body::GetNext to get
/// the next body in the world list. A NULL body indicates the end of the list.
/// @return the head of the world body list.
b2World.prototype.GetBodyList = function () {
return this.m_bodyList;
};
/// Get the world joint list. With the returned joint, use b2Joint::GetNext to get
/// the next joint in the world list. A NULL joint indicates the end of the list.
/// @return the head of the world joint list.
b2World.prototype.GetJointList = function () {
return this.m_jointList;
};
// #if B2_ENABLE_PARTICLE
b2World.prototype.GetParticleSystemList = function () {
return this.m_particleSystemList;
};
// #endif
/// Get the world contact list. With the returned contact, use b2Contact::GetNext to get
/// the next contact in the world list. A NULL contact indicates the end of the list.
/// @return the head of the world contact list.
/// @warning contacts are created and destroyed in the middle of a time step.
/// Use b2ContactListener to avoid missing contacts.
b2World.prototype.GetContactList = function () {
return this.m_contactManager.m_contactList;
};
/// Enable/disable sleep.
b2World.prototype.SetAllowSleeping = function (flag) {
if (flag === this.m_allowSleep) {
return;
}
this.m_allowSleep = flag;
if (!this.m_allowSleep) {
for (var b = this.m_bodyList; b; b = b.m_next) {
b.SetAwake(true);
}
}
};
b2World.prototype.GetAllowSleeping = function () {
return this.m_allowSleep;
};
/// Enable/disable warm starting. For testing.
b2World.prototype.SetWarmStarting = function (flag) {
this.m_warmStarting = flag;
};
b2World.prototype.GetWarmStarting = function () {
return this.m_warmStarting;
};
/// Enable/disable continuous physics. For testing.
b2World.prototype.SetContinuousPhysics = function (flag) {
this.m_continuousPhysics = flag;
};
b2World.prototype.GetContinuousPhysics = function () {
return this.m_continuousPhysics;
};
/// Enable/disable single stepped continuous physics. For testing.
b2World.prototype.SetSubStepping = function (flag) {
this.m_subStepping = flag;
};
b2World.prototype.GetSubStepping = function () {
return this.m_subStepping;
};
/// Get the number of broad-phase proxies.
b2World.prototype.GetProxyCount = function () {
return this.m_contactManager.m_broadPhase.GetProxyCount();
};
/// Get the number of bodies.
b2World.prototype.GetBodyCount = function () {
return this.m_bodyCount;
};
/// Get the number of joints.
b2World.prototype.GetJointCount = function () {
return this.m_jointCount;
};
/// Get the number of contacts (each may have 0 or more contact points).
b2World.prototype.GetContactCount = function () {
return this.m_contactManager.m_contactCount;
};
/// Get the height of the dynamic tree.
b2World.prototype.GetTreeHeight = function () {
return this.m_contactManager.m_broadPhase.GetTreeHeight();
};
/// Get the balance of the dynamic tree.
b2World.prototype.GetTreeBalance = function () {
return this.m_contactManager.m_broadPhase.GetTreeBalance();
};
/// Get the quality metric of the dynamic tree. The smaller the better.
/// The minimum is 1.
b2World.prototype.GetTreeQuality = function () {
return this.m_contactManager.m_broadPhase.GetTreeQuality();
};
/// Change the global gravity vector.
b2World.prototype.SetGravity = function (gravity, wake) {
if (wake === void 0) { wake = true; }
if (!b2Vec2.IsEqualToV(this.m_gravity, gravity)) {
this.m_gravity.Copy(gravity);
if (wake) {
for (var b = this.m_bodyList; b; b = b.m_next) {
b.SetAwake(true);
}
}
}
};
/// Get the global gravity vector.
b2World.prototype.GetGravity = function () {
return this.m_gravity;
};
/// Is the world locked (in the middle of a time step).
b2World.prototype.IsLocked = function () {
return this.m_locked;
};
/// Set flag to control automatic clearing of forces after each time step.
b2World.prototype.SetAutoClearForces = function (flag) {
this.m_clearForces = flag;
};
/// Get the flag that controls automatic clearing of forces after each time step.
b2World.prototype.GetAutoClearForces = function () {
return this.m_clearForces;
};
/// Shift the world origin. Useful for large worlds.
/// The body shift formula is: position -= newOrigin
/// @param newOrigin the new origin with respect to the old origin
b2World.prototype.ShiftOrigin = function (newOrigin) {
if (this.IsLocked()) {
throw new Error();
}
for (var b = this.m_bodyList; b; b = b.m_next) {
b.m_xf.p.SelfSub(newOrigin);
b.m_sweep.c0.SelfSub(newOrigin);
b.m_sweep.c.SelfSub(newOrigin);
}
for (var j = this.m_jointList; j; j = j.m_next) {
j.ShiftOrigin(newOrigin);
}
this.m_contactManager.m_broadPhase.ShiftOrigin(newOrigin);
};
/// Get the contact manager for testing.
b2World.prototype.GetContactManager = function () {
return this.m_contactManager;
};
/// Get the current profile.
b2World.prototype.GetProfile = function () {
return this.m_profile;
};
/// Dump the world into the log file.
/// @warning this should be called outside of a time step.
b2World.prototype.Dump = function (log) {
if (this.m_locked) {
return;
}
log("const g: b2Vec2 = new b2Vec2(%.15f, %.15f);\n", this.m_gravity.x, this.m_gravity.y);
log("this.m_world.SetGravity(g);\n");
log("const bodies: b2Body[] = [];\n");
log("const joints: b2Joint[] = [];\n");
var i = 0;
for (var b = this.m_bodyList; b; b = b.m_next) {
b.m_islandIndex = i;
b.Dump(log);
++i;
}
i = 0;
for (var j = this.m_jointList; j; j = j.m_next) {
j.m_index = i;
++i;
}
// First pass on joints, skip gear joints.
for (var j = this.m_jointList; j; j = j.m_next) {
if (j.m_type === exports.b2JointType.e_gearJoint) {
continue;
}
log("{\n");
j.Dump(log);
log("}\n");
}
// Second pass on joints, only gear joints.
for (var j = this.m_jointList; j; j = j.m_next) {
if (j.m_type !== exports.b2JointType.e_gearJoint) {
continue;
}
log("{\n");
j.Dump(log);
log("}\n");
}
};
b2World.prototype.DrawJoint = function (joint) {
if (this.m_debugDraw === null) {
return;
}
var bodyA = joint.GetBodyA();
var bodyB = joint.GetBodyB();
var xf1 = bodyA.m_xf;
var xf2 = bodyB.m_xf;
var x1 = xf1.p;
var x2 = xf2.p;
var p1 = joint.GetAnchorA(b2World.DrawJoint_s_p1);
var p2 = joint.GetAnchorB(b2World.DrawJoint_s_p2);
var color = b2World.DrawJoint_s_color.SetRGB(0.5, 0.8, 0.8);
switch (joint.m_type) {
case exports.b2JointType.e_distanceJoint:
this.m_debugDraw.DrawSegment(p1, p2, color);
break;
case exports.b2JointType.e_pulleyJoint: {
var pulley = joint;
var s1 = pulley.GetGroundAnchorA();
var s2 = pulley.GetGroundAnchorB();
this.m_debugDraw.DrawSegment(s1, p1, color);
this.m_debugDraw.DrawSegment(s2, p2, color);
this.m_debugDraw.DrawSegment(s1, s2, color);
break;
}
case exports.b2JointType.e_mouseJoint: {
var c = b2World.DrawJoint_s_c;
c.Set(0.0, 1.0, 0.0);
this.m_debugDraw.DrawPoint(p1, 4.0, c);
this.m_debugDraw.DrawPoint(p2, 4.0, c);
c.Set(0.8, 0.8, 0.8);
this.m_debugDraw.DrawSegment(p1, p2, c);
break;
}
default:
this.m_debugDraw.DrawSegment(x1, p1, color);
this.m_debugDraw.DrawSegment(p1, p2, color);
this.m_debugDraw.DrawSegment(x2, p2, color);
}
};
b2World.prototype.DrawShape = function (fixture, color) {
if (this.m_debugDraw === null) {
return;
}
var shape = fixture.GetShape();
switch (shape.m_type) {
case exports.b2ShapeType.e_circleShape: {
var circle = shape;
var center = circle.m_p;
var radius = circle.m_radius;
var axis = b2Vec2.UNITX;
this.m_debugDraw.DrawSolidCircle(center, radius, axis, color);
break;
}
case exports.b2ShapeType.e_edgeShape: {
var edge = shape;
var v1 = edge.m_vertex1;
var v2 = edge.m_vertex2;
this.m_debugDraw.DrawSegment(v1, v2, color);
break;
}
case exports.b2ShapeType.e_chainShape: {
var chain = shape;
var count = chain.m_count;
var vertices = chain.m_vertices;
var ghostColor = b2World.DrawShape_s_ghostColor.SetRGBA(0.75 * color.r, 0.75 * color.g, 0.75 * color.b, color.a);
var v1 = vertices[0];
this.m_debugDraw.DrawPoint(v1, 4.0, color);
if (chain.m_hasPrevVertex) {
var vp = chain.m_prevVertex;
this.m_debugDraw.DrawSegment(vp, v1, ghostColor);
this.m_debugDraw.DrawCircle(vp, 0.1, ghostColor);
}
for (var i = 1; i < count; ++i) {
var v2 = vertices[i];
this.m_debugDraw.DrawSegment(v1, v2, color);
this.m_debugDraw.DrawPoint(v2, 4.0, color);
v1 = v2;
}
if (chain.m_hasNextVertex) {
var vn = chain.m_nextVertex;
this.m_debugDraw.DrawSegment(vn, v1, ghostColor);
this.m_debugDraw.DrawCircle(vn, 0.1, ghostColor);
}
break;
}
case exports.b2ShapeType.e_polygonShape: {
var poly = shape;
var vertexCount = poly.m_count;
var vertices = poly.m_vertices;
this.m_debugDraw.DrawSolidPolygon(vertices, vertexCount, color);
break;
}
}
};
b2World.prototype.Solve = function (step) {
// #if B2_ENABLE_PARTICLE
// update previous transforms
for (var b = this.m_bodyList; b; b = b.m_next) {
b.m_xf0.Copy(b.m_xf);
}
// #endif
// #if B2_ENABLE_CONTROLLER
// @see b2Controller list
for (var controller = this.m_controllerList; controller; controller = controller.m_next) {
controller.Step(step);
}
// #endif
this.m_profile.solveInit = 0;
this.m_profile.solveVelocity = 0;
this.m_profile.solvePosition = 0;
// Size the island for the worst case.
var island = this.m_island;
island.Initialize(this.m_bodyCount, this.m_contactManager.m_contactCount, this.m_jointCount, null, // this.m_stackAllocator,
this.m_contactManager.m_contactListener);
// Clear all the island flags.
for (var b = this.m_bodyList; b; b = b.m_next) {
b.m_islandFlag = false;
}
for (var c = this.m_contactManager.m_contactList; c; c = c.m_next) {
c.m_islandFlag = false;
}
for (var j = this.m_jointList; j; j = j.m_next) {
j.m_islandFlag = false;
}
// Build and simulate all awake islands.
// DEBUG: const stackSize: number = this.m_bodyCount;
var stack = this.s_stack;
for (var seed = this.m_bodyList; seed; seed = seed.m_next) {
if (seed.m_islandFlag) {
continue;
}
if (!seed.IsAwake() || !seed.IsActive()) {
continue;
}
// The seed can be dynamic or kinematic.
if (seed.GetType() === exports.b2BodyType.b2_staticBody) {
continue;
}
// Reset island and stack.
island.Clear();
var stackCount = 0;
stack[stackCount++] = seed;
seed.m_islandFlag = true;
// Perform a depth first search (DFS) on the constraint graph.
while (stackCount > 0) {
// Grab the next body off the stack and add it to the island.
var b = stack[--stackCount];
if (!b) {
throw new Error();
}
// DEBUG: b2Assert(b.IsActive());
island.AddBody(b);
// Make sure the body is awake. (without resetting sleep timer).
b.m_awakeFlag = true;
// To keep islands as small as possible, we don't
// propagate islands across static bodies.
if (b.GetType() === exports.b2BodyType.b2_staticBody) {
continue;
}
// Search all contacts connected to this body.
for (var ce = b.m_contactList; ce; ce = ce.next) {
var contact = ce.contact;
// Has this contact already been added to an island?
if (contact.m_islandFlag) {
continue;
}
// Is this contact solid and touching?
if (!contact.IsEnabled() || !contact.IsTouching()) {
continue;
}
// Skip sensors.
var sensorA = contact.m_fixtureA.m_isSensor;
var sensorB = contact.m_fixtureB.m_isSensor;
if (sensorA || sensorB) {
continue;
}
island.AddContact(contact);
contact.m_islandFlag = true;
var other = ce.other;
if (!other) {
throw new Error();
}
// Was the other body already added to this island?
if (other.m_islandFlag) {
continue;
}
// DEBUG: b2Assert(stackCount < stackSize);
stack[stackCount++] = other;
other.m_islandFlag = true;
}
// Search all joints connect to this body.
for (var je = b.m_jointList; je; je = je.next) {
if (je.joint.m_islandFlag) {
continue;
}
var other = je.other;
// Don't simulate joints connected to inactive bodies.
if (!other.IsActive()) {
continue;
}
island.AddJoint(je.joint);
je.joint.m_islandFlag = true;
if (other.m_islandFlag) {
continue;
}
// DEBUG: b2Assert(stackCount < stackSize);
stack[stackCount++] = other;
other.m_islandFlag = true;
}
}
var profile = new b2Profile();
island.Solve(profile, step, this.m_gravity, this.m_allowSleep);
this.m_profile.solveInit += profile.solveInit;
this.m_profile.solveVelocity += profile.solveVelocity;
this.m_profile.solvePosition += profile.solvePosition;
// Post solve cleanup.
for (var i = 0; i < island.m_bodyCount; ++i) {
// Allow static bodies to participate in other islands.
var b = island.m_bodies[i];
if (b.GetType() === exports.b2BodyType.b2_staticBody) {
b.m_islandFlag = false;
}
}
}
for (var i = 0; i < stack.length; ++i) {
if (!stack[i]) {
break;
}
stack[i] = null;
}
var timer = new b2Timer();
// Synchronize fixtures, check for out of range bodies.
for (var b = this.m_bodyList; b; b = b.m_next) {
// If a body was not in an island then it did not move.
if (!b.m_islandFlag) {
continue;
}
if (b.GetType() === exports.b2BodyType.b2_staticBody) {
continue;
}
// Update fixtures (for broad-phase).
b.SynchronizeFixtures();
}
// Look for new contacts.
this.m_contactManager.FindNewContacts();
this.m_profile.broadphase = timer.GetMilliseconds();
};
b2World.prototype.SolveTOI = function (step) {
// b2Island island(2 * b2_maxTOIContacts, b2_maxTOIContacts, 0, &m_stackAllocator, m_contactManager.m_contactListener);
var island = this.m_island;
island.Initialize(2 * b2_maxTOIContacts, b2_maxTOIContacts, 0, null, this.m_contactManager.m_contactListener);
if (this.m_stepComplete) {
for (var b = this.m_bodyList; b; b = b.m_next) {
b.m_islandFlag = false;
b.m_sweep.alpha0 = 0;
}
for (var c = this.m_contactManager.m_contactList; c; c = c.m_next) {
// Invalidate TOI
c.m_toiFlag = false;
c.m_islandFlag = false;
c.m_toiCount = 0;
c.m_toi = 1;
}
}
// Find TOI events and solve them.
for (;;) {
// Find the first TOI.
var minContact = null;
var minAlpha = 1;
for (var c = this.m_contactManager.m_contactList; c; c = c.m_next) {
// Is this contact disabled?
if (!c.IsEnabled()) {
continue;
}
// Prevent excessive sub-stepping.
if (c.m_toiCount > b2_maxSubSteps) {
continue;
}
var alpha = 1;
if (c.m_toiFlag) {
// This contact has a valid cached TOI.
alpha = c.m_toi;
}
else {
var fA_1 = c.GetFixtureA();
var fB_1 = c.GetFixtureB();
// Is there a sensor?
if (fA_1.IsSensor() || fB_1.IsSensor()) {
continue;
}
var bA_1 = fA_1.GetBody();
var bB_1 = fB_1.GetBody();
var typeA = bA_1.m_type;
var typeB = bB_1.m_type;
// DEBUG: b2Assert(typeA !== b2BodyType.b2_staticBody || typeB !== b2BodyType.b2_staticBody);
var activeA = bA_1.IsAwake() && typeA !== exports.b2BodyType.b2_staticBody;
var activeB = bB_1.IsAwake() && typeB !== exports.b2BodyType.b2_staticBody;
// Is at least one body active (awake and dynamic or kinematic)?
if (!activeA && !activeB) {
continue;
}
var collideA = bA_1.IsBullet() || typeA !== exports.b2BodyType.b2_dynamicBody;
var collideB = bB_1.IsBullet() || typeB !== exports.b2BodyType.b2_dynamicBody;
// Are these two non-bullet dynamic bodies?
if (!collideA && !collideB) {
continue;
}
// Compute the TOI for this contact.
// Put the sweeps onto the same time interval.
var alpha0 = bA_1.m_sweep.alpha0;
if (bA_1.m_sweep.alpha0 < bB_1.m_sweep.alpha0) {
alpha0 = bB_1.m_sweep.alpha0;
bA_1.m_sweep.Advance(alpha0);
}
else if (bB_1.m_sweep.alpha0 < bA_1.m_sweep.alpha0) {
alpha0 = bA_1.m_sweep.alpha0;
bB_1.m_sweep.Advance(alpha0);
}
// DEBUG: b2Assert(alpha0 < 1);
var indexA = c.GetChildIndexA();
var indexB = c.GetChildIndexB();
// Compute the time of impact in interval [0, minTOI]
var input = b2World.SolveTOI_s_toi_input;
input.proxyA.SetShape(fA_1.GetShape(), indexA);
input.proxyB.SetShape(fB_1.GetShape(), indexB);
input.sweepA.Copy(bA_1.m_sweep);
input.sweepB.Copy(bB_1.m_sweep);
input.tMax = 1;
var output = b2World.SolveTOI_s_toi_output;
b2TimeOfImpact(output, input);
// Beta is the fraction of the remaining portion of the .
var beta = output.t;
if (output.state === exports.b2TOIOutputState.e_touching) {
alpha = b2Min(alpha0 + (1 - alpha0) * beta, 1);
}
else {
alpha = 1;
}
c.m_toi = alpha;
c.m_toiFlag = true;
}
if (alpha < minAlpha) {
// This is the minimum TOI found so far.
minContact = c;
minAlpha = alpha;
}
}
if (minContact === null || 1 - 10 * b2_epsilon < minAlpha) {
// No more TOI events. Done!
this.m_stepComplete = true;
break;
}
// Advance the bodies to the TOI.
var fA = minContact.GetFixtureA();
var fB = minContact.GetFixtureB();
var bA = fA.GetBody();
var bB = fB.GetBody();
var backup1 = b2World.SolveTOI_s_backup1.Copy(bA.m_sweep);
var backup2 = b2World.SolveTOI_s_backup2.Copy(bB.m_sweep);
bA.Advance(minAlpha);
bB.Advance(minAlpha);
// The TOI contact likely has some new contact points.
minContact.Update(this.m_contactManager.m_contactListener);
minContact.m_toiFlag = false;
++minContact.m_toiCount;
// Is the contact solid?
if (!minContact.IsEnabled() || !minContact.IsTouching()) {
// Restore the sweeps.
minContact.SetEnabled(false);
bA.m_sweep.Copy(backup1);
bB.m_sweep.Copy(backup2);
bA.SynchronizeTransform();
bB.SynchronizeTransform();
continue;
}
bA.SetAwake(true);
bB.SetAwake(true);
// Build the island
island.Clear();
island.AddBody(bA);
island.AddBody(bB);
island.AddContact(minContact);
bA.m_islandFlag = true;
bB.m_islandFlag = true;
minContact.m_islandFlag = true;
// Get contacts on bodyA and bodyB.
// const bodies: b2Body[] = [bA, bB];
for (var i = 0; i < 2; ++i) {
var body = (i === 0) ? (bA) : (bB); // bodies[i];
if (body.m_type === exports.b2BodyType.b2_dynamicBody) {
for (var ce = body.m_contactList; ce; ce = ce.next) {
if (island.m_bodyCount === island.m_bodyCapacity) {
break;
}
if (island.m_contactCount === island.m_contactCapacity) {
break;
}
var contact = ce.contact;
// Has this contact already been added to the island?
if (contact.m_islandFlag) {
continue;
}
// Only add static, kinematic, or bullet bodies.
var other = ce.other;
if (other.m_type === exports.b2BodyType.b2_dynamicBody &&
!body.IsBullet() && !other.IsBullet()) {
continue;
}
// Skip sensors.
var sensorA = contact.m_fixtureA.m_isSensor;
var sensorB = contact.m_fixtureB.m_isSensor;
if (sensorA || sensorB) {
continue;
}
// Tentatively advance the body to the TOI.
var backup = b2World.SolveTOI_s_backup.Copy(other.m_sweep);
if (!other.m_islandFlag) {
other.Advance(minAlpha);
}
// Update the contact points
contact.Update(this.m_contactManager.m_contactListener);
// Was the contact disabled by the user?
if (!contact.IsEnabled()) {
other.m_sweep.Copy(backup);
other.SynchronizeTransform();
continue;
}
// Are there contact points?
if (!contact.IsTouching()) {
other.m_sweep.Copy(backup);
other.SynchronizeTransform();
continue;
}
// Add the contact to the island
contact.m_islandFlag = true;
island.AddContact(contact);
// Has the other body already been added to the island?
if (other.m_islandFlag) {
continue;
}
// Add the other body to the island.
other.m_islandFlag = true;
if (other.m_type !== exports.b2BodyType.b2_staticBody) {
other.SetAwake(true);
}
island.AddBody(other);
}
}
}
var subStep = b2World.SolveTOI_s_subStep;
subStep.dt = (1 - minAlpha) * step.dt;
subStep.inv_dt = 1 / subStep.dt;
subStep.dtRatio = 1;
subStep.positionIterations = 20;
subStep.velocityIterations = step.velocityIterations;
// #if B2_ENABLE_PARTICLE
subStep.particleIterations = step.particleIterations;
// #endif
subStep.warmStarting = false;
island.SolveTOI(subStep, bA.m_islandIndex, bB.m_islandIndex);
// Reset island flags and synchronize broad-phase proxies.
for (var i = 0; i < island.m_bodyCount; ++i) {
var body = island.m_bodies[i];
body.m_islandFlag = false;
if (body.m_type !== exports.b2BodyType.b2_dynamicBody) {
continue;
}
body.SynchronizeFixtures();
// Invalidate all contact TOIs on this displaced body.
for (var ce = body.m_contactList; ce; ce = ce.next) {
ce.contact.m_toiFlag = false;
ce.contact.m_islandFlag = false;
}
}
// Commit fixture proxy movements to the broad-phase so that new contacts are created.
// Also, some contacts can be destroyed.
this.m_contactManager.FindNewContacts();
if (this.m_subStepping) {
this.m_stepComplete = false;
break;
}
}
};
// #if B2_ENABLE_CONTROLLER
b2World.prototype.AddController = function (controller) {
// b2Assert(controller.m_world === null, "Controller can only be a member of one world");
// controller.m_world = this;
controller.m_next = this.m_controllerList;
controller.m_prev = null;
if (this.m_controllerList) {
this.m_controllerList.m_prev = controller;
}
this.m_controllerList = controller;
++this.m_controllerCount;
return controller;
};
b2World.prototype.RemoveController = function (controller) {
// b2Assert(controller.m_world === this, "Controller is not a member of this world");
if (controller.m_prev) {
controller.m_prev.m_next = controller.m_next;
}
if (controller.m_next) {
controller.m_next.m_prev = controller.m_prev;
}
if (this.m_controllerList === controller) {
this.m_controllerList = controller.m_next;
}
--this.m_controllerCount;
controller.m_prev = null;
controller.m_next = null;
// delete controller.m_world; // = null;
return controller;
};
// #endif
/// Take a time step. This performs collision detection, integration,
/// and constraint solution.
/// @param timeStep the amount of time to simulate, this should not vary.
/// @param velocityIterations for the velocity constraint solver.
/// @param positionIterations for the position constraint solver.
b2World.Step_s_step = new b2TimeStep();
b2World.Step_s_stepTimer = new b2Timer();
b2World.Step_s_timer = new b2Timer();
// #endif
/// Call this to draw shapes and other debug draw data.
b2World.DrawDebugData_s_color = new b2Color(0, 0, 0);
b2World.DrawDebugData_s_vs = b2Vec2.MakeArray(4);
b2World.DrawDebugData_s_xf = new b2Transform();
b2World.QueryFixtureShape_s_aabb = new b2AABB();
/// Ray-cast the world for all fixtures in the path of the ray. Your callback
/// controls whether you get the closest point, any point, or n-points.
/// The ray-cast ignores shapes that contain the starting point.
/// @param callback a user implemented callback class.
/// @param point1 the ray starting point
/// @param point2 the ray ending point
b2World.RayCast_s_input = new b2RayCastInput();
b2World.RayCast_s_output = new b2RayCastOutput();
b2World.RayCast_s_point = new b2Vec2();
b2World.DrawJoint_s_p1 = new b2Vec2();
b2World.DrawJoint_s_p2 = new b2Vec2();
b2World.DrawJoint_s_color = new b2Color(0.5, 0.8, 0.8);
b2World.DrawJoint_s_c = new b2Color();
b2World.DrawShape_s_ghostColor = new b2Color();
b2World.SolveTOI_s_subStep = new b2TimeStep();
b2World.SolveTOI_s_backup = new b2Sweep();
b2World.SolveTOI_s_backup1 = new b2Sweep();
b2World.SolveTOI_s_backup2 = new b2Sweep();
b2World.SolveTOI_s_toi_input = new b2TOIInput();
b2World.SolveTOI_s_toi_output = new b2TOIOutput();
return b2World;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/**
* A controller edge is used to connect bodies and controllers
* together in a bipartite graph.
*/
var b2ControllerEdge = /** @class */ (function () {
function b2ControllerEdge(controller, body) {
this.prevBody = null; ///< the previous controller edge in the controllers's joint list
this.nextBody = null; ///< the next controller edge in the controllers's joint list
this.prevController = null; ///< the previous controller edge in the body's joint list
this.nextController = null; ///< the next controller edge in the body's joint list
this.controller = controller;
this.body = body;
}
return b2ControllerEdge;
}());
/**
* Base class for controllers. Controllers are a convience for
* encapsulating common per-step functionality.
*/
var b2Controller = /** @class */ (function () {
function b2Controller() {
// m_world: b2World;
this.m_bodyList = null;
this.m_bodyCount = 0;
this.m_prev = null;
this.m_next = null;
}
/**
* Get the next controller in the world's body list.
*/
b2Controller.prototype.GetNext = function () {
return this.m_next;
};
/**
* Get the previous controller in the world's body list.
*/
b2Controller.prototype.GetPrev = function () {
return this.m_prev;
};
/**
* Get the parent world of this body.
*/
// GetWorld() {
// return this.m_world;
// }
/**
* Get the attached body list
*/
b2Controller.prototype.GetBodyList = function () {
return this.m_bodyList;
};
/**
* Adds a body to the controller list.
*/
b2Controller.prototype.AddBody = function (body) {
var edge = new b2ControllerEdge(this, body);
//Add edge to controller list
edge.nextBody = this.m_bodyList;
edge.prevBody = null;
if (this.m_bodyList) {
this.m_bodyList.prevBody = edge;
}
this.m_bodyList = edge;
++this.m_bodyCount;
//Add edge to body list
edge.nextController = body.m_controllerList;
edge.prevController = null;
if (body.m_controllerList) {
body.m_controllerList.prevController = edge;
}
body.m_controllerList = edge;
++body.m_controllerCount;
};
/**
* Removes a body from the controller list.
*/
b2Controller.prototype.RemoveBody = function (body) {
//Assert that the controller is not empty
if (this.m_bodyCount <= 0) {
throw new Error();
}
//Find the corresponding edge
/*b2ControllerEdge*/
var edge = this.m_bodyList;
while (edge && edge.body !== body) {
edge = edge.nextBody;
}
//Assert that we are removing a body that is currently attached to the controller
if (edge === null) {
throw new Error();
}
//Remove edge from controller list
if (edge.prevBody) {
edge.prevBody.nextBody = edge.nextBody;
}
if (edge.nextBody) {
edge.nextBody.prevBody = edge.prevBody;
}
if (this.m_bodyList === edge) {
this.m_bodyList = edge.nextBody;
}
--this.m_bodyCount;
//Remove edge from body list
if (edge.nextController) {
edge.nextController.prevController = edge.prevController;
}
if (edge.prevController) {
edge.prevController.nextController = edge.nextController;
}
if (body.m_controllerList === edge) {
body.m_controllerList = edge.nextController;
}
--body.m_controllerCount;
};
/**
* Removes all bodies from the controller list.
*/
b2Controller.prototype.Clear = function () {
while (this.m_bodyList) {
this.RemoveBody(this.m_bodyList.body);
}
this.m_bodyCount = 0;
};
return b2Controller;
}());
// #endif
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/**
* Calculates buoyancy forces for fluids in the form of a half
* plane.
*/
var b2BuoyancyController = /** @class */ (function (_super) {
__extends(b2BuoyancyController, _super);
function b2BuoyancyController() {
var _this = _super !== null && _super.apply(this, arguments) || this;
/**
* The outer surface normal
*/
_this.normal = new b2Vec2(0, 1);
/**
* The height of the fluid surface along the normal
*/
_this.offset = 0;
/**
* The fluid density
*/
_this.density = 0;
/**
* Fluid velocity, for drag calculations
*/
_this.velocity = new b2Vec2(0, 0);
/**
* Linear drag co-efficient
*/
_this.linearDrag = 0;
/**
* Angular drag co-efficient
*/
_this.angularDrag = 0;
/**
* If false, bodies are assumed to be uniformly dense, otherwise
* use the shapes densities
*/
_this.useDensity = false; //False by default to prevent a gotcha
/**
* If true, gravity is taken from the world instead of the
*/
_this.useWorldGravity = true;
/**
* Gravity vector, if the world's gravity is not used
*/
_this.gravity = new b2Vec2(0, 0);
return _this;
}
b2BuoyancyController.prototype.Step = function (step) {
if (!this.m_bodyList) {
return;
}
if (this.useWorldGravity) {
this.gravity.Copy(this.m_bodyList.body.GetWorld().GetGravity());
}
for (var i = this.m_bodyList; i; i = i.nextBody) {
var body = i.body;
if (!body.IsAwake()) {
//Buoyancy force is just a function of position,
//so unlike most forces, it is safe to ignore sleeping bodes
continue;
}
var areac = new b2Vec2();
var massc = new b2Vec2();
var area = 0;
var mass = 0;
for (var fixture = body.GetFixtureList(); fixture; fixture = fixture.m_next) {
var sc = new b2Vec2();
var sarea = fixture.GetShape().ComputeSubmergedArea(this.normal, this.offset, body.GetTransform(), sc);
area += sarea;
areac.x += sarea * sc.x;
areac.y += sarea * sc.y;
var shapeDensity = 0;
if (this.useDensity) {
//TODO: Expose density publicly
shapeDensity = fixture.GetDensity();
}
else {
shapeDensity = 1;
}
mass += sarea * shapeDensity;
massc.x += sarea * sc.x * shapeDensity;
massc.y += sarea * sc.y * shapeDensity;
}
areac.x /= area;
areac.y /= area;
// b2Vec2 localCentroid = b2MulT(body->GetXForm(),areac);
massc.x /= mass;
massc.y /= mass;
if (area < b2_epsilon) {
continue;
}
//Buoyancy
var buoyancyForce = this.gravity.Clone().SelfNeg();
buoyancyForce.SelfMul(this.density * area);
body.ApplyForce(buoyancyForce, massc);
//Linear drag
var dragForce = body.GetLinearVelocityFromWorldPoint(areac, new b2Vec2());
dragForce.SelfSub(this.velocity);
dragForce.SelfMul((-this.linearDrag * area));
body.ApplyForce(dragForce, areac);
//Angular drag
//TODO: Something that makes more physical sense?
body.ApplyTorque((-body.GetInertia() / body.GetMass() * area * body.GetAngularVelocity() * this.angularDrag));
}
};
b2BuoyancyController.prototype.Draw = function (debugDraw) {
var r = 100;
var p1 = new b2Vec2();
var p2 = new b2Vec2();
p1.x = this.normal.x * this.offset + this.normal.y * r;
p1.y = this.normal.y * this.offset - this.normal.x * r;
p2.x = this.normal.x * this.offset - this.normal.y * r;
p2.y = this.normal.y * this.offset + this.normal.x * r;
var color = new b2Color(0, 0, 0.8);
debugDraw.DrawSegment(p1, p2, color);
};
return b2BuoyancyController;
}(b2Controller));
// #endif
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/**
* Applies a force every frame
*/
var b2ConstantAccelController = /** @class */ (function (_super) {
__extends(b2ConstantAccelController, _super);
function b2ConstantAccelController() {
var _this = _super !== null && _super.apply(this, arguments) || this;
/**
* The acceleration to apply
*/
_this.A = new b2Vec2(0, 0);
return _this;
}
b2ConstantAccelController.prototype.Step = function (step) {
var dtA = b2Vec2.MulSV(step.dt, this.A, b2ConstantAccelController.Step_s_dtA);
for (var i = this.m_bodyList; i; i = i.nextBody) {
var body = i.body;
if (!body.IsAwake()) {
continue;
}
body.SetLinearVelocity(b2Vec2.AddVV(body.GetLinearVelocity(), dtA, b2Vec2.s_t0));
}
};
b2ConstantAccelController.prototype.Draw = function (draw) { };
b2ConstantAccelController.Step_s_dtA = new b2Vec2();
return b2ConstantAccelController;
}(b2Controller));
// #endif
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/**
* Applies a force every frame
*/
var b2ConstantForceController = /** @class */ (function (_super) {
__extends(b2ConstantForceController, _super);
function b2ConstantForceController() {
var _this = _super !== null && _super.apply(this, arguments) || this;
/**
* The force to apply
*/
_this.F = new b2Vec2(0, 0);
return _this;
}
b2ConstantForceController.prototype.Step = function (step) {
for (var i = this.m_bodyList; i; i = i.nextBody) {
var body = i.body;
if (!body.IsAwake()) {
continue;
}
body.ApplyForce(this.F, body.GetWorldCenter());
}
};
b2ConstantForceController.prototype.Draw = function (draw) { };
return b2ConstantForceController;
}(b2Controller));
// #endif
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/**
* Applies simplified gravity between every pair of bodies
*/
var b2GravityController = /** @class */ (function (_super) {
__extends(b2GravityController, _super);
function b2GravityController() {
var _this = _super !== null && _super.apply(this, arguments) || this;
/**
* Specifies the strength of the gravitiation force
*/
_this.G = 1;
/**
* If true, gravity is proportional to r^-2, otherwise r^-1
*/
_this.invSqr = true;
return _this;
}
/**
* @see b2Controller::Step
*/
b2GravityController.prototype.Step = function (step) {
if (this.invSqr) {
for (var i = this.m_bodyList; i; i = i.nextBody) {
var body1 = i.body;
var p1 = body1.GetWorldCenter();
var mass1 = body1.GetMass();
for (var j = this.m_bodyList; j && j !== i; j = j.nextBody) {
var body2 = j.body;
var p2 = body2.GetWorldCenter();
var mass2 = body2.GetMass();
var dx = p2.x - p1.x;
var dy = p2.y - p1.y;
var r2 = dx * dx + dy * dy;
if (r2 < b2_epsilon) {
continue;
}
var f = b2GravityController.Step_s_f.Set(dx, dy);
f.SelfMul(this.G / r2 / b2Sqrt(r2) * mass1 * mass2);
if (body1.IsAwake()) {
body1.ApplyForce(f, p1);
}
if (body2.IsAwake()) {
body2.ApplyForce(f.SelfMul(-1), p2);
}
}
}
}
else {
for (var i = this.m_bodyList; i; i = i.nextBody) {
var body1 = i.body;
var p1 = body1.GetWorldCenter();
var mass1 = body1.GetMass();
for (var j = this.m_bodyList; j && j !== i; j = j.nextBody) {
var body2 = j.body;
var p2 = body2.GetWorldCenter();
var mass2 = body2.GetMass();
var dx = p2.x - p1.x;
var dy = p2.y - p1.y;
var r2 = dx * dx + dy * dy;
if (r2 < b2_epsilon) {
continue;
}
var f = b2GravityController.Step_s_f.Set(dx, dy);
f.SelfMul(this.G / r2 * mass1 * mass2);
if (body1.IsAwake()) {
body1.ApplyForce(f, p1);
}
if (body2.IsAwake()) {
body2.ApplyForce(f.SelfMul(-1), p2);
}
}
}
}
};
b2GravityController.prototype.Draw = function (draw) { };
b2GravityController.Step_s_f = new b2Vec2();
return b2GravityController;
}(b2Controller));
// #endif
/*
* Copyright (c) 2006-2007 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
/**
* Applies top down linear damping to the controlled bodies
* The damping is calculated by multiplying velocity by a matrix
* in local co-ordinates.
*/
var b2TensorDampingController = /** @class */ (function (_super) {
__extends(b2TensorDampingController, _super);
function b2TensorDampingController() {
var _this = _super !== null && _super.apply(this, arguments) || this;
/// Tensor to use in damping model
_this.T = new b2Mat22();
/*Some examples (matrixes in format (row1; row2))
(-a 0; 0 -a) Standard isotropic damping with strength a
( 0 a; -a 0) Electron in fixed field - a force at right angles to velocity with proportional magnitude
(-a 0; 0 -b) Differing x and y damping. Useful e.g. for top-down wheels.
*/
//By the way, tensor in this case just means matrix, don't let the terminology get you down.
/// Set this to a positive number to clamp the maximum amount of damping done.
_this.maxTimestep = 0;
return _this;
}
// Typically one wants maxTimestep to be 1/(max eigenvalue of T), so that damping will never cause something to reverse direction
/**
* @see b2Controller::Step
*/
b2TensorDampingController.prototype.Step = function (step) {
var timestep = step.dt;
if (timestep <= b2_epsilon) {
return;
}
if (timestep > this.maxTimestep && this.maxTimestep > 0) {
timestep = this.maxTimestep;
}
for (var i = this.m_bodyList; i; i = i.nextBody) {
var body = i.body;
if (!body.IsAwake()) {
continue;
}
var damping = body.GetWorldVector(b2Mat22.MulMV(this.T, body.GetLocalVector(body.GetLinearVelocity(), b2Vec2.s_t0), b2Vec2.s_t1), b2TensorDampingController.Step_s_damping);
// body->SetLinearVelocity(body->GetLinearVelocity() + timestep * damping);
body.SetLinearVelocity(b2Vec2.AddVV(body.GetLinearVelocity(), b2Vec2.MulSV(timestep, damping, b2Vec2.s_t0), b2Vec2.s_t1));
}
};
b2TensorDampingController.prototype.Draw = function (draw) { };
/**
* Sets damping independantly along the x and y axes
*/
b2TensorDampingController.prototype.SetAxisAligned = function (xDamping, yDamping) {
this.T.ex.x = (-xDamping);
this.T.ex.y = 0;
this.T.ey.x = 0;
this.T.ey.y = (-yDamping);
if (xDamping > 0 || yDamping > 0) {
this.maxTimestep = 1 / b2Max(xDamping, yDamping);
}
else {
this.maxTimestep = 0;
}
};
b2TensorDampingController.Step_s_damping = new b2Vec2();
return b2TensorDampingController;
}(b2Controller));
// #endif
/*
* Copyright (c) 2011 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
///
var b2RopeDef = /** @class */ (function () {
function b2RopeDef() {
///
this.vertices = [];
///
this.count = 0;
///
this.masses = [];
///
this.gravity = new b2Vec2(0, 0);
///
this.damping = 0.1;
/// Stretching stiffness
this.k2 = 0.9;
/// Bending stiffness. Values above 0.5 can make the simulation blow up.
this.k3 = 0.1;
}
return b2RopeDef;
}());
///
var b2Rope = /** @class */ (function () {
function b2Rope() {
this.m_count = 0;
this.m_ps = [];
this.m_p0s = [];
this.m_vs = [];
this.m_ims = [];
this.m_Ls = [];
this.m_as = [];
this.m_gravity = new b2Vec2();
this.m_damping = 0;
this.m_k2 = 1;
this.m_k3 = 0.1;
}
b2Rope.prototype.GetVertexCount = function () {
return this.m_count;
};
b2Rope.prototype.GetVertices = function () {
return this.m_ps;
};
///
b2Rope.prototype.Initialize = function (def) {
// DEBUG: b2Assert(def.count >= 3);
this.m_count = def.count;
// this.m_ps = (b2Vec2*)b2Alloc(this.m_count * sizeof(b2Vec2));
this.m_ps = b2Vec2.MakeArray(this.m_count);
// this.m_p0s = (b2Vec2*)b2Alloc(this.m_count * sizeof(b2Vec2));
this.m_p0s = b2Vec2.MakeArray(this.m_count);
// this.m_vs = (b2Vec2*)b2Alloc(this.m_count * sizeof(b2Vec2));
this.m_vs = b2Vec2.MakeArray(this.m_count);
// this.m_ims = (float32*)b2Alloc(this.m_count * sizeof(float32));
this.m_ims = b2MakeNumberArray(this.m_count);
for (var i = 0; i < this.m_count; ++i) {
this.m_ps[i].Copy(def.vertices[i]);
this.m_p0s[i].Copy(def.vertices[i]);
this.m_vs[i].SetZero();
var m = def.masses[i];
if (m > 0) {
this.m_ims[i] = 1 / m;
}
else {
this.m_ims[i] = 0;
}
}
var count2 = this.m_count - 1;
var count3 = this.m_count - 2;
// this.m_Ls = (float32*)be2Alloc(count2 * sizeof(float32));
this.m_Ls = b2MakeNumberArray(count2);
// this.m_as = (float32*)b2Alloc(count3 * sizeof(float32));
this.m_as = b2MakeNumberArray(count3);
for (var i = 0; i < count2; ++i) {
var p1 = this.m_ps[i];
var p2 = this.m_ps[i + 1];
this.m_Ls[i] = b2Vec2.DistanceVV(p1, p2);
}
for (var i = 0; i < count3; ++i) {
var p1 = this.m_ps[i];
var p2 = this.m_ps[i + 1];
var p3 = this.m_ps[i + 2];
var d1 = b2Vec2.SubVV(p2, p1, b2Vec2.s_t0);
var d2 = b2Vec2.SubVV(p3, p2, b2Vec2.s_t1);
var a = b2Vec2.CrossVV(d1, d2);
var b = b2Vec2.DotVV(d1, d2);
this.m_as[i] = b2Atan2(a, b);
}
this.m_gravity.Copy(def.gravity);
this.m_damping = def.damping;
this.m_k2 = def.k2;
this.m_k3 = def.k3;
};
///
b2Rope.prototype.Step = function (h, iterations) {
if (h === 0) {
return;
}
var d = Math.exp(-h * this.m_damping);
for (var i = 0; i < this.m_count; ++i) {
this.m_p0s[i].Copy(this.m_ps[i]);
if (this.m_ims[i] > 0) {
this.m_vs[i].SelfMulAdd(h, this.m_gravity);
}
this.m_vs[i].SelfMul(d);
this.m_ps[i].SelfMulAdd(h, this.m_vs[i]);
}
for (var i = 0; i < iterations; ++i) {
this.SolveC2();
this.SolveC3();
this.SolveC2();
}
var inv_h = 1 / h;
for (var i = 0; i < this.m_count; ++i) {
b2Vec2.MulSV(inv_h, b2Vec2.SubVV(this.m_ps[i], this.m_p0s[i], b2Vec2.s_t0), this.m_vs[i]);
}
};
b2Rope.prototype.SolveC2 = function () {
var count2 = this.m_count - 1;
for (var i = 0; i < count2; ++i) {
var p1 = this.m_ps[i];
var p2 = this.m_ps[i + 1];
var d = b2Vec2.SubVV(p2, p1, b2Rope.s_d);
var L = d.Normalize();
var im1 = this.m_ims[i];
var im2 = this.m_ims[i + 1];
if (im1 + im2 === 0) {
continue;
}
var s1 = im1 / (im1 + im2);
var s2 = im2 / (im1 + im2);
p1.SelfMulSub(this.m_k2 * s1 * (this.m_Ls[i] - L), d);
p2.SelfMulAdd(this.m_k2 * s2 * (this.m_Ls[i] - L), d);
// this.m_ps[i] = p1;
// this.m_ps[i + 1] = p2;
}
};
b2Rope.prototype.SetAngle = function (angle) {
var count3 = this.m_count - 2;
for (var i = 0; i < count3; ++i) {
this.m_as[i] = angle;
}
};
b2Rope.prototype.SolveC3 = function () {
var count3 = this.m_count - 2;
for (var i = 0; i < count3; ++i) {
var p1 = this.m_ps[i];
var p2 = this.m_ps[i + 1];
var p3 = this.m_ps[i + 2];
var m1 = this.m_ims[i];
var m2 = this.m_ims[i + 1];
var m3 = this.m_ims[i + 2];
var d1 = b2Vec2.SubVV(p2, p1, b2Rope.s_d1);
var d2 = b2Vec2.SubVV(p3, p2, b2Rope.s_d2);
var L1sqr = d1.LengthSquared();
var L2sqr = d2.LengthSquared();
if (L1sqr * L2sqr === 0) {
continue;
}
var a = b2Vec2.CrossVV(d1, d2);
var b = b2Vec2.DotVV(d1, d2);
var angle = b2Atan2(a, b);
var Jd1 = b2Vec2.MulSV((-1 / L1sqr), d1.SelfSkew(), b2Rope.s_Jd1);
var Jd2 = b2Vec2.MulSV((1 / L2sqr), d2.SelfSkew(), b2Rope.s_Jd2);
var J1 = b2Vec2.NegV(Jd1, b2Rope.s_J1);
var J2 = b2Vec2.SubVV(Jd1, Jd2, b2Rope.s_J2);
var J3 = Jd2;
var mass = m1 * b2Vec2.DotVV(J1, J1) + m2 * b2Vec2.DotVV(J2, J2) + m3 * b2Vec2.DotVV(J3, J3);
if (mass === 0) {
continue;
}
mass = 1 / mass;
var C = angle - this.m_as[i];
while (C > b2_pi) {
angle -= 2 * b2_pi;
C = angle - this.m_as[i];
}
while (C < -b2_pi) {
angle += 2 * b2_pi;
C = angle - this.m_as[i];
}
var impulse = -this.m_k3 * mass * C;
p1.SelfMulAdd((m1 * impulse), J1);
p2.SelfMulAdd((m2 * impulse), J2);
p3.SelfMulAdd((m3 * impulse), J3);
// this.m_ps[i] = p1;
// this.m_ps[i + 1] = p2;
// this.m_ps[i + 2] = p3;
}
};
b2Rope.prototype.Draw = function (draw) {
var c = new b2Color(0.4, 0.5, 0.7);
for (var i = 0; i < this.m_count - 1; ++i) {
draw.DrawSegment(this.m_ps[i], this.m_ps[i + 1], c);
}
};
///
b2Rope.s_d = new b2Vec2();
b2Rope.s_d1 = new b2Vec2();
b2Rope.s_d2 = new b2Vec2();
b2Rope.s_Jd1 = new b2Vec2();
b2Rope.s_Jd2 = new b2Vec2();
b2Rope.s_J1 = new b2Vec2();
b2Rope.s_J2 = new b2Vec2();
return b2Rope;
}());
/*
* Copyright (c) 2006-2009 Erin Catto http://www.box2d.org
*
* This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages
* arising from the use of this software.
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
* 1. The origin of this software must not be misrepresented; you must not
* claim that you wrote the original software. If you use this software
* in a product, an acknowledgment in the product documentation would be
* appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
* misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
*/
exports.b2Assert = b2Assert;
exports.b2Maybe = b2Maybe;
exports.b2_maxFloat = b2_maxFloat;
exports.b2_epsilon = b2_epsilon;
exports.b2_epsilon_sq = b2_epsilon_sq;
exports.b2_pi = b2_pi;
exports.b2_maxManifoldPoints = b2_maxManifoldPoints;
exports.b2_maxPolygonVertices = b2_maxPolygonVertices;
exports.b2_aabbExtension = b2_aabbExtension;
exports.b2_aabbMultiplier = b2_aabbMultiplier;
exports.b2_linearSlop = b2_linearSlop;
exports.b2_angularSlop = b2_angularSlop;
exports.b2_polygonRadius = b2_polygonRadius;
exports.b2_maxSubSteps = b2_maxSubSteps;
exports.b2_maxTOIContacts = b2_maxTOIContacts;
exports.b2_velocityThreshold = b2_velocityThreshold;
exports.b2_maxLinearCorrection = b2_maxLinearCorrection;
exports.b2_maxAngularCorrection = b2_maxAngularCorrection;
exports.b2_maxTranslation = b2_maxTranslation;
exports.b2_maxTranslationSquared = b2_maxTranslationSquared;
exports.b2_maxRotation = b2_maxRotation;
exports.b2_maxRotationSquared = b2_maxRotationSquared;
exports.b2_baumgarte = b2_baumgarte;
exports.b2_toiBaumgarte = b2_toiBaumgarte;
exports.b2_invalidParticleIndex = b2_invalidParticleIndex;
exports.b2_maxParticleIndex = b2_maxParticleIndex;
exports.b2_particleStride = b2_particleStride;
exports.b2_minParticleWeight = b2_minParticleWeight;
exports.b2_maxParticlePressure = b2_maxParticlePressure;
exports.b2_maxParticleForce = b2_maxParticleForce;
exports.b2_maxTriadDistance = b2_maxTriadDistance;
exports.b2_maxTriadDistanceSquared = b2_maxTriadDistanceSquared;
exports.b2_minParticleSystemBufferCapacity = b2_minParticleSystemBufferCapacity;
exports.b2_barrierCollisionTime = b2_barrierCollisionTime;
exports.b2_timeToSleep = b2_timeToSleep;
exports.b2_linearSleepTolerance = b2_linearSleepTolerance;
exports.b2_angularSleepTolerance = b2_angularSleepTolerance;
exports.b2Alloc = b2Alloc;
exports.b2Free = b2Free;
exports.b2Log = b2Log;
exports.b2Version = b2Version;
exports.b2_version = b2_version;
exports.b2_branch = b2_branch;
exports.b2_commit = b2_commit;
exports.b2ParseInt = b2ParseInt;
exports.b2ParseUInt = b2ParseUInt;
exports.b2MakeArray = b2MakeArray;
exports.b2MakeNullArray = b2MakeNullArray;
exports.b2MakeNumberArray = b2MakeNumberArray;
exports.b2_pi_over_180 = b2_pi_over_180;
exports.b2_180_over_pi = b2_180_over_pi;
exports.b2_two_pi = b2_two_pi;
exports.b2Abs = b2Abs;
exports.b2Min = b2Min;
exports.b2Max = b2Max;
exports.b2Clamp = b2Clamp;
exports.b2Swap = b2Swap;
exports.b2IsValid = b2IsValid;
exports.b2Sq = b2Sq;
exports.b2InvSqrt = b2InvSqrt;
exports.b2Sqrt = b2Sqrt;
exports.b2Pow = b2Pow;
exports.b2DegToRad = b2DegToRad;
exports.b2RadToDeg = b2RadToDeg;
exports.b2Cos = b2Cos;
exports.b2Sin = b2Sin;
exports.b2Acos = b2Acos;
exports.b2Asin = b2Asin;
exports.b2Atan2 = b2Atan2;
exports.b2NextPowerOfTwo = b2NextPowerOfTwo;
exports.b2IsPowerOfTwo = b2IsPowerOfTwo;
exports.b2Random = b2Random;
exports.b2RandomRange = b2RandomRange;
exports.b2Vec2 = b2Vec2;
exports.b2Vec2_zero = b2Vec2_zero;
exports.b2Vec3 = b2Vec3;
exports.b2Mat22 = b2Mat22;
exports.b2Mat33 = b2Mat33;
exports.b2Rot = b2Rot;
exports.b2Transform = b2Transform;
exports.b2Sweep = b2Sweep;
exports.b2Color = b2Color;
exports.b2Draw = b2Draw;
exports.b2Timer = b2Timer;
exports.b2Counter = b2Counter;
exports.b2GrowableStack = b2GrowableStack;
exports.b2BlockAllocator = b2BlockAllocator;
exports.b2StackAllocator = b2StackAllocator;
exports.b2ContactFeature = b2ContactFeature;
exports.b2ContactID = b2ContactID;
exports.b2ManifoldPoint = b2ManifoldPoint;
exports.b2Manifold = b2Manifold;
exports.b2WorldManifold = b2WorldManifold;
exports.b2GetPointStates = b2GetPointStates;
exports.b2ClipVertex = b2ClipVertex;
exports.b2RayCastInput = b2RayCastInput;
exports.b2RayCastOutput = b2RayCastOutput;
exports.b2AABB = b2AABB;
exports.b2TestOverlapAABB = b2TestOverlapAABB;
exports.b2ClipSegmentToLine = b2ClipSegmentToLine;
exports.b2TestOverlapShape = b2TestOverlapShape;
exports.b2DistanceProxy = b2DistanceProxy;
exports.b2SimplexCache = b2SimplexCache;
exports.b2DistanceInput = b2DistanceInput;
exports.b2DistanceOutput = b2DistanceOutput;
exports.b2ShapeCastInput = b2ShapeCastInput;
exports.b2ShapeCastOutput = b2ShapeCastOutput;
exports.b2_gjk_reset = b2_gjk_reset;
exports.b2SimplexVertex = b2SimplexVertex;
exports.b2Simplex = b2Simplex;
exports.b2Distance = b2Distance;
exports.b2ShapeCast = b2ShapeCast;
exports.b2Pair = b2Pair;
exports.b2BroadPhase = b2BroadPhase;
exports.b2PairLessThan = b2PairLessThan;
exports.b2TreeNode = b2TreeNode;
exports.b2DynamicTree = b2DynamicTree;
exports.b2_toi_reset = b2_toi_reset;
exports.b2TOIInput = b2TOIInput;
exports.b2TOIOutput = b2TOIOutput;
exports.b2SeparationFunction = b2SeparationFunction;
exports.b2TimeOfImpact = b2TimeOfImpact;
exports.b2CollideCircles = b2CollideCircles;
exports.b2CollidePolygonAndCircle = b2CollidePolygonAndCircle;
exports.b2CollidePolygons = b2CollidePolygons;
exports.b2CollideEdgeAndCircle = b2CollideEdgeAndCircle;
exports.b2CollideEdgeAndPolygon = b2CollideEdgeAndPolygon;
exports.b2MassData = b2MassData;
exports.b2Shape = b2Shape;
exports.b2CircleShape = b2CircleShape;
exports.b2PolygonShape = b2PolygonShape;
exports.b2EdgeShape = b2EdgeShape;
exports.b2ChainShape = b2ChainShape;
exports.b2Filter = b2Filter;
exports.b2FixtureDef = b2FixtureDef;
exports.b2FixtureProxy = b2FixtureProxy;
exports.b2Fixture = b2Fixture;
exports.b2BodyDef = b2BodyDef;
exports.b2Body = b2Body;
exports.b2World = b2World;
exports.b2DestructionListener = b2DestructionListener;
exports.b2ContactFilter = b2ContactFilter;
exports.b2ContactImpulse = b2ContactImpulse;
exports.b2ContactListener = b2ContactListener;
exports.b2QueryCallback = b2QueryCallback;
exports.b2RayCastCallback = b2RayCastCallback;
exports.b2Island = b2Island;
exports.b2Profile = b2Profile;
exports.b2TimeStep = b2TimeStep;
exports.b2Position = b2Position;
exports.b2Velocity = b2Velocity;
exports.b2SolverData = b2SolverData;
exports.b2ContactManager = b2ContactManager;
exports.b2MixFriction = b2MixFriction;
exports.b2MixRestitution = b2MixRestitution;
exports.b2ContactEdge = b2ContactEdge;
exports.b2Contact = b2Contact;
exports.b2ContactRegister = b2ContactRegister;
exports.b2ContactFactory = b2ContactFactory;
exports.g_blockSolve = g_blockSolve;
exports.b2VelocityConstraintPoint = b2VelocityConstraintPoint;
exports.b2ContactVelocityConstraint = b2ContactVelocityConstraint;
exports.b2ContactPositionConstraint = b2ContactPositionConstraint;
exports.b2ContactSolverDef = b2ContactSolverDef;
exports.b2PositionSolverManifold = b2PositionSolverManifold;
exports.b2ContactSolver = b2ContactSolver;
exports.b2CircleContact = b2CircleContact;
exports.b2PolygonContact = b2PolygonContact;
exports.b2PolygonAndCircleContact = b2PolygonAndCircleContact;
exports.b2EdgeAndCircleContact = b2EdgeAndCircleContact;
exports.b2EdgeAndPolygonContact = b2EdgeAndPolygonContact;
exports.b2ChainAndCircleContact = b2ChainAndCircleContact;
exports.b2ChainAndPolygonContact = b2ChainAndPolygonContact;
exports.b2Jacobian = b2Jacobian;
exports.b2JointEdge = b2JointEdge;
exports.b2JointDef = b2JointDef;
exports.b2Joint = b2Joint;
exports.b2AreaJointDef = b2AreaJointDef;
exports.b2AreaJoint = b2AreaJoint;
exports.b2DistanceJointDef = b2DistanceJointDef;
exports.b2DistanceJoint = b2DistanceJoint;
exports.b2FrictionJointDef = b2FrictionJointDef;
exports.b2FrictionJoint = b2FrictionJoint;
exports.b2GearJointDef = b2GearJointDef;
exports.b2GearJoint = b2GearJoint;
exports.b2MotorJointDef = b2MotorJointDef;
exports.b2MotorJoint = b2MotorJoint;
exports.b2MouseJointDef = b2MouseJointDef;
exports.b2MouseJoint = b2MouseJoint;
exports.b2PrismaticJointDef = b2PrismaticJointDef;
exports.b2PrismaticJoint = b2PrismaticJoint;
exports.b2_minPulleyLength = b2_minPulleyLength;
exports.b2PulleyJointDef = b2PulleyJointDef;
exports.b2PulleyJoint = b2PulleyJoint;
exports.b2RevoluteJointDef = b2RevoluteJointDef;
exports.b2RevoluteJoint = b2RevoluteJoint;
exports.b2RopeJointDef = b2RopeJointDef;
exports.b2RopeJoint = b2RopeJoint;
exports.b2WeldJointDef = b2WeldJointDef;
exports.b2WeldJoint = b2WeldJoint;
exports.b2WheelJointDef = b2WheelJointDef;
exports.b2WheelJoint = b2WheelJoint;
exports.b2ControllerEdge = b2ControllerEdge;
exports.b2Controller = b2Controller;
exports.b2BuoyancyController = b2BuoyancyController;
exports.b2ConstantAccelController = b2ConstantAccelController;
exports.b2ConstantForceController = b2ConstantForceController;
exports.b2GravityController = b2GravityController;
exports.b2TensorDampingController = b2TensorDampingController;
exports.b2ParticleDef = b2ParticleDef;
exports.b2CalculateParticleIterations = b2CalculateParticleIterations;
exports.b2ParticleHandle = b2ParticleHandle;
exports.b2ParticleGroupDef = b2ParticleGroupDef;
exports.b2ParticleGroup = b2ParticleGroup;
exports.b2GrowableBuffer = b2GrowableBuffer;
exports.b2FixtureParticleQueryCallback = b2FixtureParticleQueryCallback;
exports.b2ParticleContact = b2ParticleContact;
exports.b2ParticleBodyContact = b2ParticleBodyContact;
exports.b2ParticlePair = b2ParticlePair;
exports.b2ParticleTriad = b2ParticleTriad;
exports.b2ParticleSystemDef = b2ParticleSystemDef;
exports.b2RopeDef = b2RopeDef;
exports.b2Rope = b2Rope;
Object.defineProperty(exports, '__esModule', { value: true });
})));