Constraints

What is it

A constraint represents a connection between two bodies. This connection can apply restrictions on the relative movement of the bodies, and allows us to model all sorts of physical connections between bodies such as ropes, doors, character joints, and more.

Because the two bodies are still simulated independently, A constraint (PhysicsConstraint) definition includes transforms from the local space of each body to the pivot position/orientation of the constraint on that body. Together these are known as the "constraint space". During simulation, constraints will act on the two bodies to maintain the pivot position/orientation across the two constrained bodies. For example, a ball-and-socket constraint will apply forces to the bodies so that the pivots point positions (defined respectively to each body's local space) will coincide. Likewise, a hinge constraint will also maintain the common rotation axis.

Constraint Space

In rigid body dynamics, each body has 6 degrees of freedom: 3 translational degrees of freedom, and 3 rotational degrees of freedom. A constraint definition includes limitations on one or more of these degrees of freedom for its constrained bodies. Different types of constraints are generally defined by the number and type of these limitations. For example, in a ball-and-socket constraint, the constrained objects have no linear freedom relative to each other in any direction as they are attached together at the point, but are completely free to rotate around the constraint pivot point. In a hinge constraint, the objects no linear freedom and also have restricted relative orientation. In addition, limits can be provided for each degree of freedom. For example, limiting the distance rotation of a hinge constraint allows you for example to create a door that will not rotate beyond a given angle.

Constraint Linear Limits

Constraint Angular Limits

Constraint Types

Babylon supports several constraint types; the most generic of which is the "6 DoF," which allows control over each degree of freedom individually. In addition to the 6 DoF, Babylon also provides several helper types for common constraints. Many of these constraints are specializations of the 6 DoF, but they don't require you to specify the full constraint space, which can make initialization easier in those cases.

EnumNameNotes
Enum
LOCK
Name
Lock
Notes
Locked A locked joint attempts to keep the two constraint spaces completely lined up, allowing no relative movement.
Enum
BALL_AND_SOCKET
Name
Ball and socket
Notes
Ball and Socket A ball and socket joint attempts to line up the pivot positions but puts no restrictions on the relative rotation of the two bodies.
Enum
DISTANCE
Name
Distance
Notes
Distance A distance joint attempts to keep the positions of the constraint spaces within a specified distance and provides no restriction on relative rotation.
Enum
HINGE
Name
Hinge
Notes
Hinge A hinge will keep the positions of the constraint spaces aligned as well as two of the angular axes, only allowing relative rotation around one axis.
Enum
PRISMATIC
Name
Prismatic
Notes
Prismatic A prismatic joint allows the constraint spaces to translate along one axis and allows no relative rotation of the two spaces.
Enum
SLIDER
Name
Slider
Notes
Slider Similar to the prismatic joint, but also allows the bodies to rotate around the translation axis.
Enum
SIX_DOF
Name
6 Degrees Of Freedom
Notes
Image representing 6 Degrees of Freedom
6 degrees of freedom: three translational and three rotational axes. By GregorDS - Own work, CC BY-SA 4.0, https://commons.wikimedia.org/w/index.php?curid=38429678
The most generic type. Does not provide any restrictions by default, so the limits can be applied as you choose.

How to use it

const bodyA = new BABYLON.PhysicsBody(objectA, BABYLON.PhysicsMotionType.DYNAMIC, scene);
const bodyB = new BABYLON.PhysicsBody(objectB, BABYLON.PhysicsMotionType.DYNAMIC, scene);
const constraint = new BABYLON.DistanceConstraint(
10, // max distance between the two bodies
scene
);
bodyA.addConstraint(bodyB, constraint);

If one or both bodies are instanced, you need to specify the instance to which the constraint applies:

// This will add a constraint between the instance in index 3 of body A, and the instance in index 2 of body B
bodyA.addConstraint(bodyB, constraint, 3, 2);
// The constraint can also be between two instances in the same body
bodyA.addConstraint(bodyA, constraint, 4, 7);

For the 6DOF (6 Degrees of Freedom) constraint, you should pass an array with entries specifying the translational and rotational axis that have min and max limits:

// This will constraint the bodies to mantain a distance of at least 1 and at most 2, and to rotate at most 1.58 rad along the perpendicular axis
let constraint = new BABYLON.Physics6DoFConstraint({
pivotA: new BABYLON.Vector3(0, -0.5, 0),
pivotB: new BABYLON.Vector3(0, 0.5, 0),
perpAxisA: new BABYLON.Vector3(1, 0, 0),
perpAxisB: new BABYLON.Vector3(1, 0, 0),
}, [
{axis: BABYLON.PhysicsConstraintAxis.LINEAR_DISTANCE, minLimit: 1, maxLimit: 2},
{axis: BABYLON.PhysicsConstraintAxis.ANGULAR_Y, minLimit: 0, maxLimit: 1.58}
], scene);

Best practices

A problem that's very noticeable to users of your application is when a third body comes between the constraint spaces of a pair of constrained bodies, resulting in visual penetration, as seen in the picture below.

Ragdoll bone gap

This can look very unnatural and, in addition, cause jitter between the constrained bodies, as the collision detection will "fight" against the constraint. Overlapping the constrained bodies greatly helps to avoid situations like this.

Ragdoll with overlapping bones

Even when the constraint is not attempting to force the bodies into an overlapping position, most use-cases for constraints still attempt to position the bodies very close to each other. To avoid the collision detection from "fighting" the constraint resolution, by default, we disable collisions between each pair of constrained bodies. If you do want the bodies to collide, however, this can be controlled by PhysicsConstraint.isCollisionsEnabled.

When simulating a long chain of constrained bodies, not all of the constraints might be solved perfectly by the physics engine, which can result in stretching or misalignment of the constraint - this is more likely to happen if the mass ratio between two constrained bodies is very high. A common approach to mitigate this is to tweak the mass ratios of the constrained bodies - increase the mass and inertia of bodies which are closer to the "root" of the chain and decrease as the bodies are further from the root. e.g. if you have a ragdoll, ensure that the torso has the most mass and decrease the mass as you move down the limbs of the ragdoll.

Examples

Constraints Swinging pendulums

Further reading

Using A Physics Engine
Learn how to use one of the available physics engines in Babylon.js.
Using A Physics Engine
Forces
Learn all about applying physical forces in Babylon.js.
Forces