1692 lines
56 KiB
TypeScript
1692 lines
56 KiB
TypeScript
|
|
import {RawRigidBodySet, RawRigidBodyType} from "../raw";
|
|||
|
|
import {Rotation, RotationOps, Vector, VectorOps} from "../math";
|
|||
|
|
// #if DIM3
|
|||
|
|
import {SdpMatrix3, SdpMatrix3Ops} from "../math";
|
|||
|
|
// #endif
|
|||
|
|
import {Collider, ColliderSet} from "../geometry";
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The integer identifier of a collider added to a `ColliderSet`.
|
|||
|
|
*/
|
|||
|
|
export type RigidBodyHandle = number;
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The simulation status of a rigid-body.
|
|||
|
|
*/
|
|||
|
|
// TODO: rename this to RigidBodyType
|
|||
|
|
export enum RigidBodyType {
|
|||
|
|
/**
|
|||
|
|
* A `RigidBodyType::Dynamic` body can be affected by all external forces.
|
|||
|
|
*/
|
|||
|
|
Dynamic = 0,
|
|||
|
|
/**
|
|||
|
|
* A `RigidBodyType::Fixed` body cannot be affected by external forces.
|
|||
|
|
*/
|
|||
|
|
Fixed,
|
|||
|
|
/**
|
|||
|
|
* A `RigidBodyType::KinematicPositionBased` body cannot be affected by any external forces but can be controlled
|
|||
|
|
* by the user at the position level while keeping realistic one-way interaction with dynamic bodies.
|
|||
|
|
*
|
|||
|
|
* One-way interaction means that a kinematic body can push a dynamic body, but a kinematic body
|
|||
|
|
* cannot be pushed by anything. In other words, the trajectory of a kinematic body can only be
|
|||
|
|
* modified by the user and is independent from any contact or joint it is involved in.
|
|||
|
|
*/
|
|||
|
|
KinematicPositionBased,
|
|||
|
|
/**
|
|||
|
|
* A `RigidBodyType::KinematicVelocityBased` body cannot be affected by any external forces but can be controlled
|
|||
|
|
* by the user at the velocity level while keeping realistic one-way interaction with dynamic bodies.
|
|||
|
|
*
|
|||
|
|
* One-way interaction means that a kinematic body can push a dynamic body, but a kinematic body
|
|||
|
|
* cannot be pushed by anything. In other words, the trajectory of a kinematic body can only be
|
|||
|
|
* modified by the user and is independent from any contact or joint it is involved in.
|
|||
|
|
*/
|
|||
|
|
KinematicVelocityBased,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body.
|
|||
|
|
*/
|
|||
|
|
export class RigidBody {
|
|||
|
|
private rawSet: RawRigidBodySet; // The RigidBody won't need to free this.
|
|||
|
|
private colliderSet: ColliderSet;
|
|||
|
|
readonly handle: RigidBodyHandle;
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* An arbitrary user-defined object associated with this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public userData?: unknown;
|
|||
|
|
|
|||
|
|
constructor(
|
|||
|
|
rawSet: RawRigidBodySet,
|
|||
|
|
colliderSet: ColliderSet,
|
|||
|
|
handle: RigidBodyHandle,
|
|||
|
|
) {
|
|||
|
|
this.rawSet = rawSet;
|
|||
|
|
this.colliderSet = colliderSet;
|
|||
|
|
this.handle = handle;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/** @internal */
|
|||
|
|
public finalizeDeserialization(colliderSet: ColliderSet) {
|
|||
|
|
this.colliderSet = colliderSet;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Checks if this rigid-body is still valid (i.e. that it has
|
|||
|
|
* not been deleted from the rigid-body set yet.
|
|||
|
|
*/
|
|||
|
|
public isValid(): boolean {
|
|||
|
|
return this.rawSet.contains(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to translate.
|
|||
|
|
*
|
|||
|
|
* @param locked - If `true`, this rigid-body will no longer translate due to forces and impulses.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
*/
|
|||
|
|
public lockTranslations(locked: boolean, wakeUp: boolean) {
|
|||
|
|
return this.rawSet.rbLockTranslations(this.handle, locked, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to rotate.
|
|||
|
|
*
|
|||
|
|
* @param locked - If `true`, this rigid-body will no longer rotate due to torques and impulses.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
*/
|
|||
|
|
public lockRotations(locked: boolean, wakeUp: boolean) {
|
|||
|
|
return this.rawSet.rbLockRotations(this.handle, locked, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to translation along individual coordinate axes.
|
|||
|
|
*
|
|||
|
|
* @param enableX - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the X coordinate axis.
|
|||
|
|
* @param enableY - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the Y coordinate axis.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
*/
|
|||
|
|
public setEnabledTranslations(
|
|||
|
|
enableX: boolean,
|
|||
|
|
enableY: boolean,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
return this.rawSet.rbSetEnabledTranslations(
|
|||
|
|
this.handle,
|
|||
|
|
enableX,
|
|||
|
|
enableY,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to translation along individual coordinate axes.
|
|||
|
|
*
|
|||
|
|
* @param enableX - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the X coordinate axis.
|
|||
|
|
* @param enableY - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the Y coordinate axis.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
* @deprecated use `this.setEnabledTranslations` with the same arguments instead.
|
|||
|
|
*/
|
|||
|
|
public restrictTranslations(
|
|||
|
|
enableX: boolean,
|
|||
|
|
enableY: boolean,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
this.setEnabledTranslations(enableX, enableX, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to translate along individual coordinate axes.
|
|||
|
|
*
|
|||
|
|
* @param enableX - If `false`, this rigid-body will no longer translate due to torques and impulses, along the X coordinate axis.
|
|||
|
|
* @param enableY - If `false`, this rigid-body will no longer translate due to torques and impulses, along the Y coordinate axis.
|
|||
|
|
* @param enableZ - If `false`, this rigid-body will no longer translate due to torques and impulses, along the Z coordinate axis.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
*/
|
|||
|
|
public setEnabledTranslations(
|
|||
|
|
enableX: boolean,
|
|||
|
|
enableY: boolean,
|
|||
|
|
enableZ: boolean,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
return this.rawSet.rbSetEnabledTranslations(
|
|||
|
|
this.handle,
|
|||
|
|
enableX,
|
|||
|
|
enableY,
|
|||
|
|
enableZ,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to translate along individual coordinate axes.
|
|||
|
|
*
|
|||
|
|
* @param enableX - If `false`, this rigid-body will no longer translate due to torques and impulses, along the X coordinate axis.
|
|||
|
|
* @param enableY - If `false`, this rigid-body will no longer translate due to torques and impulses, along the Y coordinate axis.
|
|||
|
|
* @param enableZ - If `false`, this rigid-body will no longer translate due to torques and impulses, along the Z coordinate axis.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
* @deprecated use `this.setEnabledTranslations` with the same arguments instead.
|
|||
|
|
*/
|
|||
|
|
public restrictTranslations(
|
|||
|
|
enableX: boolean,
|
|||
|
|
enableY: boolean,
|
|||
|
|
enableZ: boolean,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
this.setEnabledTranslations(enableX, enableY, enableZ, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to rotate along individual coordinate axes.
|
|||
|
|
*
|
|||
|
|
* @param enableX - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the X coordinate axis.
|
|||
|
|
* @param enableY - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the Y coordinate axis.
|
|||
|
|
* @param enableZ - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the Z coordinate axis.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
*/
|
|||
|
|
public setEnabledRotations(
|
|||
|
|
enableX: boolean,
|
|||
|
|
enableY: boolean,
|
|||
|
|
enableZ: boolean,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
return this.rawSet.rbSetEnabledRotations(
|
|||
|
|
this.handle,
|
|||
|
|
enableX,
|
|||
|
|
enableY,
|
|||
|
|
enableZ,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks or unlocks the ability of this rigid-body to rotate along individual coordinate axes.
|
|||
|
|
*
|
|||
|
|
* @param enableX - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the X coordinate axis.
|
|||
|
|
* @param enableY - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the Y coordinate axis.
|
|||
|
|
* @param enableZ - If `false`, this rigid-body will no longer rotate due to torques and impulses, along the Z coordinate axis.
|
|||
|
|
* @param wakeUp - If `true`, this rigid-body will be automatically awaken if it is currently asleep.
|
|||
|
|
* @deprecated use `this.setEnabledRotations` with the same arguments instead.
|
|||
|
|
*/
|
|||
|
|
public restrictRotations(
|
|||
|
|
enableX: boolean,
|
|||
|
|
enableY: boolean,
|
|||
|
|
enableZ: boolean,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
this.setEnabledRotations(enableX, enableY, enableZ, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The dominance group, in [-127, +127] this rigid-body is part of.
|
|||
|
|
*/
|
|||
|
|
public dominanceGroup(): number {
|
|||
|
|
return this.rawSet.rbDominanceGroup(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the dominance group of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param group - The dominance group of this rigid-body. Must be a signed integer in the range [-127, +127].
|
|||
|
|
*/
|
|||
|
|
public setDominanceGroup(group: number) {
|
|||
|
|
this.rawSet.rbSetDominanceGroup(this.handle, group);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The number of additional solver iterations that will be run for this
|
|||
|
|
* rigid-body and everything that interacts with it directly or indirectly
|
|||
|
|
* through contacts or joints.
|
|||
|
|
*/
|
|||
|
|
public additionalSolverIterations(): number {
|
|||
|
|
return this.rawSet.rbAdditionalSolverIterations(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the number of additional solver iterations that will be run for this
|
|||
|
|
* rigid-body and everything that interacts with it directly or indirectly
|
|||
|
|
* through contacts or joints.
|
|||
|
|
*
|
|||
|
|
* Compared to increasing the global `World.numSolverIteration`, setting this
|
|||
|
|
* value lets you increase accuracy on only a subset of the scene, resulting in reduced
|
|||
|
|
* performance loss.
|
|||
|
|
*
|
|||
|
|
* @param iters - The new number of additional solver iterations (default: 0).
|
|||
|
|
*/
|
|||
|
|
public setAdditionalSolverIterations(iters: number) {
|
|||
|
|
this.rawSet.rbSetAdditionalSolverIterations(this.handle, iters);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Enable or disable CCD (Continuous Collision Detection) for this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param enabled - If `true`, CCD will be enabled for this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public enableCcd(enabled: boolean) {
|
|||
|
|
this.rawSet.rbEnableCcd(this.handle, enabled);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the soft-CCD prediction distance for this rigid-body.
|
|||
|
|
*
|
|||
|
|
* See the documentation of `RigidBodyDesc.setSoftCcdPrediction` for
|
|||
|
|
* additional details.
|
|||
|
|
*/
|
|||
|
|
public setSoftCcdPrediction(distance: number) {
|
|||
|
|
this.rawSet.rbSetSoftCcdPrediction(this.handle, distance);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Gets the soft-CCD prediction distance for this rigid-body.
|
|||
|
|
*
|
|||
|
|
* See the documentation of `RigidBodyDesc.setSoftCcdPrediction` for
|
|||
|
|
* additional details.
|
|||
|
|
*/
|
|||
|
|
public softCcdPrediction(): number {
|
|||
|
|
return this.rawSet.rbSoftCcdPrediction(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The world-space translation of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public translation(): Vector {
|
|||
|
|
let res = this.rawSet.rbTranslation(this.handle);
|
|||
|
|
return VectorOps.fromRaw(res);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The world-space orientation of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public rotation(): Rotation {
|
|||
|
|
let res = this.rawSet.rbRotation(this.handle);
|
|||
|
|
return RotationOps.fromRaw(res);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The world-space next translation of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* If this rigid-body is kinematic this value is set by the `setNextKinematicTranslation`
|
|||
|
|
* method and is used for estimating the kinematic body velocity at the next timestep.
|
|||
|
|
* For non-kinematic bodies, this value is currently unspecified.
|
|||
|
|
*/
|
|||
|
|
public nextTranslation(): Vector {
|
|||
|
|
let res = this.rawSet.rbNextTranslation(this.handle);
|
|||
|
|
return VectorOps.fromRaw(res);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The world-space next orientation of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* If this rigid-body is kinematic this value is set by the `setNextKinematicRotation`
|
|||
|
|
* method and is used for estimating the kinematic body velocity at the next timestep.
|
|||
|
|
* For non-kinematic bodies, this value is currently unspecified.
|
|||
|
|
*/
|
|||
|
|
public nextRotation(): Rotation {
|
|||
|
|
let res = this.rawSet.rbNextRotation(this.handle);
|
|||
|
|
return RotationOps.fromRaw(res);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the translation of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param tra - The world-space position of the rigid-body.
|
|||
|
|
* @param wakeUp - Forces the rigid-body to wake-up so it is properly affected by forces if it
|
|||
|
|
* wasn't moving before modifying its position.
|
|||
|
|
*/
|
|||
|
|
public setTranslation(tra: Vector, wakeUp: boolean) {
|
|||
|
|
// #if DIM2
|
|||
|
|
this.rawSet.rbSetTranslation(this.handle, tra.x, tra.y, wakeUp);
|
|||
|
|
// #endif
|
|||
|
|
// #if DIM3
|
|||
|
|
this.rawSet.rbSetTranslation(this.handle, tra.x, tra.y, tra.z, wakeUp);
|
|||
|
|
// #endif
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the linear velocity of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param vel - The linear velocity to set.
|
|||
|
|
* @param wakeUp - Forces the rigid-body to wake-up if it was asleep.
|
|||
|
|
*/
|
|||
|
|
public setLinvel(vel: Vector, wakeUp: boolean) {
|
|||
|
|
let rawVel = VectorOps.intoRaw(vel);
|
|||
|
|
this.rawSet.rbSetLinvel(this.handle, rawVel, wakeUp);
|
|||
|
|
rawVel.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The scale factor applied to the gravity affecting
|
|||
|
|
* this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public gravityScale(): number {
|
|||
|
|
return this.rawSet.rbGravityScale(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the scale factor applied to the gravity affecting
|
|||
|
|
* this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param factor - The scale factor to set. A value of 0.0 means
|
|||
|
|
* that this rigid-body will on longer be affected by gravity.
|
|||
|
|
* @param wakeUp - Forces the rigid-body to wake-up if it was asleep.
|
|||
|
|
*/
|
|||
|
|
public setGravityScale(factor: number, wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbSetGravityScale(this.handle, factor, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Sets the rotation quaternion of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* This does nothing if a zero quaternion is provided.
|
|||
|
|
*
|
|||
|
|
* @param rotation - The rotation to set.
|
|||
|
|
* @param wakeUp - Forces the rigid-body to wake-up so it is properly affected by forces if it
|
|||
|
|
* wasn't moving before modifying its position.
|
|||
|
|
*/
|
|||
|
|
public setRotation(rot: Rotation, wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbSetRotation(
|
|||
|
|
this.handle,
|
|||
|
|
rot.x,
|
|||
|
|
rot.y,
|
|||
|
|
rot.z,
|
|||
|
|
rot.w,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the angular velocity fo this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param vel - The angular velocity to set.
|
|||
|
|
* @param wakeUp - Forces the rigid-body to wake-up if it was asleep.
|
|||
|
|
*/
|
|||
|
|
public setAngvel(vel: Vector, wakeUp: boolean) {
|
|||
|
|
let rawVel = VectorOps.intoRaw(vel);
|
|||
|
|
this.rawSet.rbSetAngvel(this.handle, rawVel, wakeUp);
|
|||
|
|
rawVel.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Sets the rotation angle of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param angle - The rotation angle, in radians.
|
|||
|
|
* @param wakeUp - Forces the rigid-body to wake-up so it is properly affected by forces if it
|
|||
|
|
* wasn't moving before modifying its position.
|
|||
|
|
*/
|
|||
|
|
public setRotation(angle: number, wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbSetRotation(this.handle, angle, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the angular velocity fo this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param vel - The angular velocity to set.
|
|||
|
|
* @param wakeUp - Forces the rigid-body to wake-up if it was asleep.
|
|||
|
|
*/
|
|||
|
|
public setAngvel(vel: number, wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbSetAngvel(this.handle, vel, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* If this rigid body is kinematic, sets its future translation after the next timestep integration.
|
|||
|
|
*
|
|||
|
|
* This should be used instead of `rigidBody.setTranslation` to make the dynamic object
|
|||
|
|
* interacting with this kinematic body behave as expected. Internally, Rapier will compute
|
|||
|
|
* an artificial velocity for this rigid-body from its current position and its next kinematic
|
|||
|
|
* position. This velocity will be used to compute forces on dynamic bodies interacting with
|
|||
|
|
* this body.
|
|||
|
|
*
|
|||
|
|
* @param t - The kinematic translation to set.
|
|||
|
|
*/
|
|||
|
|
public setNextKinematicTranslation(t: Vector) {
|
|||
|
|
// #if DIM2
|
|||
|
|
this.rawSet.rbSetNextKinematicTranslation(this.handle, t.x, t.y);
|
|||
|
|
// #endif
|
|||
|
|
// #if DIM3
|
|||
|
|
this.rawSet.rbSetNextKinematicTranslation(this.handle, t.x, t.y, t.z);
|
|||
|
|
// #endif
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* If this rigid body is kinematic, sets its future rotation after the next timestep integration.
|
|||
|
|
*
|
|||
|
|
* This should be used instead of `rigidBody.setRotation` to make the dynamic object
|
|||
|
|
* interacting with this kinematic body behave as expected. Internally, Rapier will compute
|
|||
|
|
* an artificial velocity for this rigid-body from its current position and its next kinematic
|
|||
|
|
* position. This velocity will be used to compute forces on dynamic bodies interacting with
|
|||
|
|
* this body.
|
|||
|
|
*
|
|||
|
|
* @param rot - The kinematic rotation to set.
|
|||
|
|
*/
|
|||
|
|
public setNextKinematicRotation(rot: Rotation) {
|
|||
|
|
this.rawSet.rbSetNextKinematicRotation(
|
|||
|
|
this.handle,
|
|||
|
|
rot.x,
|
|||
|
|
rot.y,
|
|||
|
|
rot.z,
|
|||
|
|
rot.w,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* If this rigid body is kinematic, sets its future rotation after the next timestep integration.
|
|||
|
|
*
|
|||
|
|
* This should be used instead of `rigidBody.setRotation` to make the dynamic object
|
|||
|
|
* interacting with this kinematic body behave as expected. Internally, Rapier will compute
|
|||
|
|
* an artificial velocity for this rigid-body from its current position and its next kinematic
|
|||
|
|
* position. This velocity will be used to compute forces on dynamic bodies interacting with
|
|||
|
|
* this body.
|
|||
|
|
*
|
|||
|
|
* @param angle - The kinematic rotation angle, in radians.
|
|||
|
|
*/
|
|||
|
|
public setNextKinematicRotation(angle: number) {
|
|||
|
|
this.rawSet.rbSetNextKinematicRotation(this.handle, angle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The linear velocity of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public linvel(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbLinvel(this.handle));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The velocity of the given world-space point on this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public velocityAtPoint(point: Vector): Vector {
|
|||
|
|
const rawPoint = VectorOps.intoRaw(point);
|
|||
|
|
let result = VectorOps.fromRaw(
|
|||
|
|
this.rawSet.rbVelocityAtPoint(this.handle, rawPoint),
|
|||
|
|
);
|
|||
|
|
rawPoint.free();
|
|||
|
|
return result;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* The angular velocity of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public angvel(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbAngvel(this.handle));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* The angular velocity of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public angvel(): number {
|
|||
|
|
return this.rawSet.rbAngvel(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The mass of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public mass(): number {
|
|||
|
|
return this.rawSet.rbMass(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The inverse mass taking into account translation locking.
|
|||
|
|
*/
|
|||
|
|
public effectiveInvMass(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbEffectiveInvMass(this.handle));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The inverse of the mass of a rigid-body.
|
|||
|
|
*
|
|||
|
|
* If this is zero, the rigid-body is assumed to have infinite mass.
|
|||
|
|
*/
|
|||
|
|
public invMass(): number {
|
|||
|
|
return this.rawSet.rbInvMass(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The center of mass of a rigid-body expressed in its local-space.
|
|||
|
|
*/
|
|||
|
|
public localCom(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbLocalCom(this.handle));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The world-space center of mass of the rigid-body.
|
|||
|
|
*/
|
|||
|
|
public worldCom(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbWorldCom(this.handle));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* The inverse of the principal angular inertia of the rigid-body.
|
|||
|
|
*
|
|||
|
|
* Components set to zero are assumed to be infinite along the corresponding principal axis.
|
|||
|
|
*/
|
|||
|
|
public invPrincipalInertia(): number {
|
|||
|
|
return this.rawSet.rbInvPrincipalInertia(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* The inverse of the principal angular inertia of the rigid-body.
|
|||
|
|
*
|
|||
|
|
* Components set to zero are assumed to be infinite along the corresponding principal axis.
|
|||
|
|
*/
|
|||
|
|
public invPrincipalInertia(): Vector {
|
|||
|
|
return VectorOps.fromRaw(
|
|||
|
|
this.rawSet.rbInvPrincipalInertia(this.handle),
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* The angular inertia along the principal inertia axes of the rigid-body.
|
|||
|
|
*/
|
|||
|
|
public principalInertia(): number {
|
|||
|
|
return this.rawSet.rbPrincipalInertia(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* The angular inertia along the principal inertia axes of the rigid-body.
|
|||
|
|
*/
|
|||
|
|
public principalInertia(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbPrincipalInertia(this.handle));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* The principal vectors of the local angular inertia tensor of the rigid-body.
|
|||
|
|
*/
|
|||
|
|
public principalInertiaLocalFrame(): Rotation {
|
|||
|
|
return RotationOps.fromRaw(
|
|||
|
|
this.rawSet.rbPrincipalInertiaLocalFrame(this.handle),
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* The world-space inverse angular inertia tensor of the rigid-body,
|
|||
|
|
* taking into account rotation locking.
|
|||
|
|
*/
|
|||
|
|
public effectiveWorldInvInertia(): number {
|
|||
|
|
return this.rawSet.rbEffectiveWorldInvInertia(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* The world-space inverse angular inertia tensor of the rigid-body,
|
|||
|
|
* taking into account rotation locking.
|
|||
|
|
*/
|
|||
|
|
public effectiveWorldInvInertia(): SdpMatrix3 {
|
|||
|
|
return SdpMatrix3Ops.fromRaw(
|
|||
|
|
this.rawSet.rbEffectiveWorldInvInertia(this.handle),
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* The effective world-space angular inertia (that takes the potential rotation locking into account) of
|
|||
|
|
* this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public effectiveAngularInertia(): number {
|
|||
|
|
return this.rawSet.rbEffectiveAngularInertia(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* The effective world-space angular inertia (that takes the potential rotation locking into account) of
|
|||
|
|
* this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public effectiveAngularInertia(): SdpMatrix3 {
|
|||
|
|
return SdpMatrix3Ops.fromRaw(
|
|||
|
|
this.rawSet.rbEffectiveAngularInertia(this.handle),
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Put this rigid body to sleep.
|
|||
|
|
*
|
|||
|
|
* A sleeping body no longer moves and is no longer simulated by the physics engine unless
|
|||
|
|
* it is waken up. It can be woken manually with `this.wakeUp()` or automatically due to
|
|||
|
|
* external forces like contacts.
|
|||
|
|
*/
|
|||
|
|
public sleep() {
|
|||
|
|
this.rawSet.rbSleep(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Wakes this rigid-body up.
|
|||
|
|
*
|
|||
|
|
* A dynamic rigid-body that does not move during several consecutive frames will
|
|||
|
|
* be put to sleep by the physics engine, i.e., it will stop being simulated in order
|
|||
|
|
* to avoid useless computations.
|
|||
|
|
* This methods forces a sleeping rigid-body to wake-up. This is useful, e.g., before modifying
|
|||
|
|
* the position of a dynamic body so that it is properly simulated afterwards.
|
|||
|
|
*/
|
|||
|
|
public wakeUp() {
|
|||
|
|
this.rawSet.rbWakeUp(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Is CCD enabled for this rigid-body?
|
|||
|
|
*/
|
|||
|
|
public isCcdEnabled(): boolean {
|
|||
|
|
return this.rawSet.rbIsCcdEnabled(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The number of colliders attached to this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public numColliders(): number {
|
|||
|
|
return this.rawSet.rbNumColliders(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Retrieves the `i-th` collider attached to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param i - The index of the collider to retrieve. Must be a number in `[0, this.numColliders()[`.
|
|||
|
|
* This index is **not** the same as the unique identifier of the collider.
|
|||
|
|
*/
|
|||
|
|
public collider(i: number): Collider {
|
|||
|
|
return this.colliderSet.get(this.rawSet.rbCollider(this.handle, i));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets whether this rigid-body is enabled or not.
|
|||
|
|
*
|
|||
|
|
* @param enabled - Set to `false` to disable this rigid-body and all its attached colliders.
|
|||
|
|
*/
|
|||
|
|
public setEnabled(enabled: boolean) {
|
|||
|
|
this.rawSet.rbSetEnabled(this.handle, enabled);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Is this rigid-body enabled?
|
|||
|
|
*/
|
|||
|
|
public isEnabled(): boolean {
|
|||
|
|
return this.rawSet.rbIsEnabled(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The status of this rigid-body: static, dynamic, or kinematic.
|
|||
|
|
*/
|
|||
|
|
public bodyType(): RigidBodyType {
|
|||
|
|
return this.rawSet.rbBodyType(this.handle) as number as RigidBodyType;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Set a new status for this rigid-body: static, dynamic, or kinematic.
|
|||
|
|
*/
|
|||
|
|
public setBodyType(type: RigidBodyType, wakeUp: boolean) {
|
|||
|
|
return this.rawSet.rbSetBodyType(
|
|||
|
|
this.handle,
|
|||
|
|
type as number as RawRigidBodyType,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Is this rigid-body sleeping?
|
|||
|
|
*/
|
|||
|
|
public isSleeping(): boolean {
|
|||
|
|
return this.rawSet.rbIsSleeping(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Is the velocity of this rigid-body not zero?
|
|||
|
|
*/
|
|||
|
|
public isMoving(): boolean {
|
|||
|
|
return this.rawSet.rbIsMoving(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Is this rigid-body static?
|
|||
|
|
*/
|
|||
|
|
public isFixed(): boolean {
|
|||
|
|
return this.rawSet.rbIsFixed(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Is this rigid-body kinematic?
|
|||
|
|
*/
|
|||
|
|
public isKinematic(): boolean {
|
|||
|
|
return this.rawSet.rbIsKinematic(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Is this rigid-body dynamic?
|
|||
|
|
*/
|
|||
|
|
public isDynamic(): boolean {
|
|||
|
|
return this.rawSet.rbIsDynamic(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The linear damping coefficient of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public linearDamping(): number {
|
|||
|
|
return this.rawSet.rbLinearDamping(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* The angular damping coefficient of this rigid-body.
|
|||
|
|
*/
|
|||
|
|
public angularDamping(): number {
|
|||
|
|
return this.rawSet.rbAngularDamping(this.handle);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the linear damping factor applied to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param factor - The damping factor to set.
|
|||
|
|
*/
|
|||
|
|
public setLinearDamping(factor: number) {
|
|||
|
|
this.rawSet.rbSetLinearDamping(this.handle, factor);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Recompute the mass-properties of this rigid-bodies based on its currently attached colliders.
|
|||
|
|
*/
|
|||
|
|
public recomputeMassPropertiesFromColliders() {
|
|||
|
|
this.rawSet.rbRecomputeMassPropertiesFromColliders(
|
|||
|
|
this.handle,
|
|||
|
|
this.colliderSet.raw,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the rigid-body's additional mass.
|
|||
|
|
*
|
|||
|
|
* The total angular inertia of the rigid-body will be scaled automatically based on this additional mass. If this
|
|||
|
|
* scaling effect isn’t desired, use Self::additional_mass_properties instead of this method.
|
|||
|
|
*
|
|||
|
|
* This is only the "additional" mass because the total mass of the rigid-body is equal to the sum of this
|
|||
|
|
* additional mass and the mass computed from the colliders (with non-zero densities) attached to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* That total mass (which includes the attached colliders’ contributions) will be updated at the name physics step,
|
|||
|
|
* or can be updated manually with `this.recomputeMassPropertiesFromColliders`.
|
|||
|
|
*
|
|||
|
|
* This will override any previous additional mass-properties set by `this.setAdditionalMass`,
|
|||
|
|
* `this.setAdditionalMassProperties`, `RigidBodyDesc::setAdditionalMass`, or
|
|||
|
|
* `RigidBodyDesc.setAdditionalMassfProperties` for this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param mass - The additional mass to set.
|
|||
|
|
* @param wakeUp - If `true` then the rigid-body will be woken up if it was put to sleep because it did not move for a while.
|
|||
|
|
*/
|
|||
|
|
public setAdditionalMass(mass: number, wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbSetAdditionalMass(this.handle, mass, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Sets the rigid-body's additional mass-properties.
|
|||
|
|
*
|
|||
|
|
* This is only the "additional" mass-properties because the total mass-properties of the rigid-body is equal to the
|
|||
|
|
* sum of this additional mass-properties and the mass computed from the colliders (with non-zero densities) attached
|
|||
|
|
* to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* That total mass-properties (which include the attached colliders’ contributions) will be updated at the name
|
|||
|
|
* physics step, or can be updated manually with `this.recomputeMassPropertiesFromColliders`.
|
|||
|
|
*
|
|||
|
|
* This will override any previous mass-properties set by `this.setAdditionalMass`,
|
|||
|
|
* `this.setAdditionalMassProperties`, `RigidBodyDesc.setAdditionalMass`, or `RigidBodyDesc.setAdditionalMassProperties`
|
|||
|
|
* for this rigid-body.
|
|||
|
|
*
|
|||
|
|
* If `wake_up` is true then the rigid-body will be woken up if it was put to sleep because it did not move for a while.
|
|||
|
|
*/
|
|||
|
|
public setAdditionalMassProperties(
|
|||
|
|
mass: number,
|
|||
|
|
centerOfMass: Vector,
|
|||
|
|
principalAngularInertia: Vector,
|
|||
|
|
angularInertiaLocalFrame: Rotation,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
let rawCom = VectorOps.intoRaw(centerOfMass);
|
|||
|
|
let rawPrincipalInertia = VectorOps.intoRaw(principalAngularInertia);
|
|||
|
|
let rawInertiaFrame = RotationOps.intoRaw(angularInertiaLocalFrame);
|
|||
|
|
|
|||
|
|
this.rawSet.rbSetAdditionalMassProperties(
|
|||
|
|
this.handle,
|
|||
|
|
mass,
|
|||
|
|
rawCom,
|
|||
|
|
rawPrincipalInertia,
|
|||
|
|
rawInertiaFrame,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
|
|||
|
|
rawCom.free();
|
|||
|
|
rawPrincipalInertia.free();
|
|||
|
|
rawInertiaFrame.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Sets the rigid-body's additional mass-properties.
|
|||
|
|
*
|
|||
|
|
* This is only the "additional" mass-properties because the total mass-properties of the rigid-body is equal to the
|
|||
|
|
* sum of this additional mass-properties and the mass computed from the colliders (with non-zero densities) attached
|
|||
|
|
* to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* That total mass-properties (which include the attached colliders’ contributions) will be updated at the name
|
|||
|
|
* physics step, or can be updated manually with `this.recomputeMassPropertiesFromColliders`.
|
|||
|
|
*
|
|||
|
|
* This will override any previous mass-properties set by `this.setAdditionalMass`,
|
|||
|
|
* `this.setAdditionalMassProperties`, `RigidBodyDesc.setAdditionalMass`, or `RigidBodyDesc.setAdditionalMassProperties`
|
|||
|
|
* for this rigid-body.
|
|||
|
|
*
|
|||
|
|
* If `wake_up` is true then the rigid-body will be woken up if it was put to sleep because it did not move for a while.
|
|||
|
|
*/
|
|||
|
|
public setAdditionalMassProperties(
|
|||
|
|
mass: number,
|
|||
|
|
centerOfMass: Vector,
|
|||
|
|
principalAngularInertia: number,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
let rawCom = VectorOps.intoRaw(centerOfMass);
|
|||
|
|
this.rawSet.rbSetAdditionalMassProperties(
|
|||
|
|
this.handle,
|
|||
|
|
mass,
|
|||
|
|
rawCom,
|
|||
|
|
principalAngularInertia,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
rawCom.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the linear damping factor applied to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param factor - The damping factor to set.
|
|||
|
|
*/
|
|||
|
|
public setAngularDamping(factor: number) {
|
|||
|
|
this.rawSet.rbSetAngularDamping(this.handle, factor);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Resets to zero the user forces (but not torques) applied to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public resetForces(wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbResetForces(this.handle, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Resets to zero the user torques applied to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public resetTorques(wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbResetTorques(this.handle, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Adds a force at the center-of-mass of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param force - the world-space force to add to the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public addForce(force: Vector, wakeUp: boolean) {
|
|||
|
|
const rawForce = VectorOps.intoRaw(force);
|
|||
|
|
this.rawSet.rbAddForce(this.handle, rawForce, wakeUp);
|
|||
|
|
rawForce.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Applies an impulse at the center-of-mass of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param impulse - the world-space impulse to apply on the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public applyImpulse(impulse: Vector, wakeUp: boolean) {
|
|||
|
|
const rawImpulse = VectorOps.intoRaw(impulse);
|
|||
|
|
this.rawSet.rbApplyImpulse(this.handle, rawImpulse, wakeUp);
|
|||
|
|
rawImpulse.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Adds a torque at the center-of-mass of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param torque - the torque to add to the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public addTorque(torque: number, wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbAddTorque(this.handle, torque, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Adds a torque at the center-of-mass of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param torque - the world-space torque to add to the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public addTorque(torque: Vector, wakeUp: boolean) {
|
|||
|
|
const rawTorque = VectorOps.intoRaw(torque);
|
|||
|
|
this.rawSet.rbAddTorque(this.handle, rawTorque, wakeUp);
|
|||
|
|
rawTorque.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Applies an impulsive torque at the center-of-mass of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param torqueImpulse - the torque impulse to apply on the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public applyTorqueImpulse(torqueImpulse: number, wakeUp: boolean) {
|
|||
|
|
this.rawSet.rbApplyTorqueImpulse(this.handle, torqueImpulse, wakeUp);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Applies an impulsive torque at the center-of-mass of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param torqueImpulse - the world-space torque impulse to apply on the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public applyTorqueImpulse(torqueImpulse: Vector, wakeUp: boolean) {
|
|||
|
|
const rawTorqueImpulse = VectorOps.intoRaw(torqueImpulse);
|
|||
|
|
this.rawSet.rbApplyTorqueImpulse(this.handle, rawTorqueImpulse, wakeUp);
|
|||
|
|
rawTorqueImpulse.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Adds a force at the given world-space point of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param force - the world-space force to add to the rigid-body.
|
|||
|
|
* @param point - the world-space point where the impulse is to be applied on the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public addForceAtPoint(force: Vector, point: Vector, wakeUp: boolean) {
|
|||
|
|
const rawForce = VectorOps.intoRaw(force);
|
|||
|
|
const rawPoint = VectorOps.intoRaw(point);
|
|||
|
|
this.rawSet.rbAddForceAtPoint(this.handle, rawForce, rawPoint, wakeUp);
|
|||
|
|
rawForce.free();
|
|||
|
|
rawPoint.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Applies an impulse at the given world-space point of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param impulse - the world-space impulse to apply on the rigid-body.
|
|||
|
|
* @param point - the world-space point where the impulse is to be applied on the rigid-body.
|
|||
|
|
* @param wakeUp - should the rigid-body be automatically woken-up?
|
|||
|
|
*/
|
|||
|
|
public applyImpulseAtPoint(
|
|||
|
|
impulse: Vector,
|
|||
|
|
point: Vector,
|
|||
|
|
wakeUp: boolean,
|
|||
|
|
) {
|
|||
|
|
const rawImpulse = VectorOps.intoRaw(impulse);
|
|||
|
|
const rawPoint = VectorOps.intoRaw(point);
|
|||
|
|
this.rawSet.rbApplyImpulseAtPoint(
|
|||
|
|
this.handle,
|
|||
|
|
rawImpulse,
|
|||
|
|
rawPoint,
|
|||
|
|
wakeUp,
|
|||
|
|
);
|
|||
|
|
rawImpulse.free();
|
|||
|
|
rawPoint.free();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Retrieves the constant force(s) the user added to this rigid-body
|
|||
|
|
* Returns zero if the rigid-body is not dynamic.
|
|||
|
|
*/
|
|||
|
|
public userForce(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbUserForce(this.handle));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Retrieves the constant torque(s) the user added to this rigid-body
|
|||
|
|
* Returns zero if the rigid-body is not dynamic.
|
|||
|
|
*/
|
|||
|
|
public userTorque(): number {
|
|||
|
|
return this.rawSet.rbUserTorque(this.handle);
|
|||
|
|
}
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Retrieves the constant torque(s) the user added to this rigid-body
|
|||
|
|
* Returns zero if the rigid-body is not dynamic.
|
|||
|
|
*/
|
|||
|
|
public userTorque(): Vector {
|
|||
|
|
return VectorOps.fromRaw(this.rawSet.rbUserTorque(this.handle));
|
|||
|
|
}
|
|||
|
|
// #endif
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
export class RigidBodyDesc {
|
|||
|
|
enabled: boolean;
|
|||
|
|
translation: Vector;
|
|||
|
|
rotation: Rotation;
|
|||
|
|
gravityScale: number;
|
|||
|
|
mass: number;
|
|||
|
|
massOnly: boolean;
|
|||
|
|
centerOfMass: Vector;
|
|||
|
|
translationsEnabledX: boolean;
|
|||
|
|
translationsEnabledY: boolean;
|
|||
|
|
linvel: Vector;
|
|||
|
|
// #if DIM2
|
|||
|
|
angvel: number;
|
|||
|
|
principalAngularInertia: number;
|
|||
|
|
rotationsEnabled: boolean;
|
|||
|
|
// #endif
|
|||
|
|
// #if DIM3
|
|||
|
|
angvel: Vector;
|
|||
|
|
principalAngularInertia: Vector;
|
|||
|
|
angularInertiaLocalFrame: Rotation;
|
|||
|
|
translationsEnabledZ: boolean;
|
|||
|
|
rotationsEnabledX: boolean;
|
|||
|
|
rotationsEnabledY: boolean;
|
|||
|
|
rotationsEnabledZ: boolean;
|
|||
|
|
// #endif
|
|||
|
|
linearDamping: number;
|
|||
|
|
angularDamping: number;
|
|||
|
|
status: RigidBodyType;
|
|||
|
|
canSleep: boolean;
|
|||
|
|
sleeping: boolean;
|
|||
|
|
ccdEnabled: boolean;
|
|||
|
|
softCcdPrediction: number;
|
|||
|
|
dominanceGroup: number;
|
|||
|
|
additionalSolverIterations: number;
|
|||
|
|
userData?: unknown;
|
|||
|
|
|
|||
|
|
constructor(status: RigidBodyType) {
|
|||
|
|
this.enabled = true;
|
|||
|
|
this.status = status;
|
|||
|
|
this.translation = VectorOps.zeros();
|
|||
|
|
this.rotation = RotationOps.identity();
|
|||
|
|
this.gravityScale = 1.0;
|
|||
|
|
this.linvel = VectorOps.zeros();
|
|||
|
|
this.mass = 0.0;
|
|||
|
|
this.massOnly = false;
|
|||
|
|
this.centerOfMass = VectorOps.zeros();
|
|||
|
|
this.translationsEnabledX = true;
|
|||
|
|
this.translationsEnabledY = true;
|
|||
|
|
// #if DIM2
|
|||
|
|
this.angvel = 0.0;
|
|||
|
|
this.principalAngularInertia = 0.0;
|
|||
|
|
this.rotationsEnabled = true;
|
|||
|
|
// #endif
|
|||
|
|
// #if DIM3
|
|||
|
|
this.angvel = VectorOps.zeros();
|
|||
|
|
this.principalAngularInertia = VectorOps.zeros();
|
|||
|
|
this.angularInertiaLocalFrame = RotationOps.identity();
|
|||
|
|
this.translationsEnabledZ = true;
|
|||
|
|
this.rotationsEnabledX = true;
|
|||
|
|
this.rotationsEnabledY = true;
|
|||
|
|
this.rotationsEnabledZ = true;
|
|||
|
|
// #endif
|
|||
|
|
this.linearDamping = 0.0;
|
|||
|
|
this.angularDamping = 0.0;
|
|||
|
|
this.canSleep = true;
|
|||
|
|
this.sleeping = false;
|
|||
|
|
this.ccdEnabled = false;
|
|||
|
|
this.softCcdPrediction = 0.0;
|
|||
|
|
this.dominanceGroup = 0;
|
|||
|
|
this.additionalSolverIterations = 0;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a dynamic rigid-body.
|
|||
|
|
*/
|
|||
|
|
public static dynamic(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.Dynamic);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a position-based kinematic rigid-body.
|
|||
|
|
*/
|
|||
|
|
public static kinematicPositionBased(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.KinematicPositionBased);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a velocity-based kinematic rigid-body.
|
|||
|
|
*/
|
|||
|
|
public static kinematicVelocityBased(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.KinematicVelocityBased);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a fixed rigid-body.
|
|||
|
|
*/
|
|||
|
|
public static fixed(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.Fixed);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a dynamic rigid-body.
|
|||
|
|
*
|
|||
|
|
* @deprecated The method has been renamed to `.dynamic()`.
|
|||
|
|
*/
|
|||
|
|
public static newDynamic(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.Dynamic);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a position-based kinematic rigid-body.
|
|||
|
|
*
|
|||
|
|
* @deprecated The method has been renamed to `.kinematicPositionBased()`.
|
|||
|
|
*/
|
|||
|
|
public static newKinematicPositionBased(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.KinematicPositionBased);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a velocity-based kinematic rigid-body.
|
|||
|
|
*
|
|||
|
|
* @deprecated The method has been renamed to `.kinematicVelocityBased()`.
|
|||
|
|
*/
|
|||
|
|
public static newKinematicVelocityBased(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.KinematicVelocityBased);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* A rigid-body descriptor used to build a fixed rigid-body.
|
|||
|
|
*
|
|||
|
|
* @deprecated The method has been renamed to `.fixed()`.
|
|||
|
|
*/
|
|||
|
|
public static newStatic(): RigidBodyDesc {
|
|||
|
|
return new RigidBodyDesc(RigidBodyType.Fixed);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
public setDominanceGroup(group: number): RigidBodyDesc {
|
|||
|
|
this.dominanceGroup = group;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the number of additional solver iterations that will be run for this
|
|||
|
|
* rigid-body and everything that interacts with it directly or indirectly
|
|||
|
|
* through contacts or joints.
|
|||
|
|
*
|
|||
|
|
* Compared to increasing the global `World.numSolverIteration`, setting this
|
|||
|
|
* value lets you increase accuracy on only a subset of the scene, resulting in reduced
|
|||
|
|
* performance loss.
|
|||
|
|
*
|
|||
|
|
* @param iters - The new number of additional solver iterations (default: 0).
|
|||
|
|
*/
|
|||
|
|
public setAdditionalSolverIterations(iters: number): RigidBodyDesc {
|
|||
|
|
this.additionalSolverIterations = iters;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets whether the created rigid-body will be enabled or disabled.
|
|||
|
|
* @param enabled − If set to `false` the rigid-body will be disabled at creation.
|
|||
|
|
*/
|
|||
|
|
public setEnabled(enabled: boolean): RigidBodyDesc {
|
|||
|
|
this.enabled = enabled;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Sets the initial translation of the rigid-body to create.
|
|||
|
|
*/
|
|||
|
|
public setTranslation(x: number, y: number): RigidBodyDesc {
|
|||
|
|
if (typeof x != "number" || typeof y != "number")
|
|||
|
|
throw TypeError("The translation components must be numbers.");
|
|||
|
|
|
|||
|
|
this.translation = {x: x, y: y};
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Sets the initial translation of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* @param tra - The translation to set.
|
|||
|
|
*/
|
|||
|
|
public setTranslation(x: number, y: number, z: number): RigidBodyDesc {
|
|||
|
|
if (
|
|||
|
|
typeof x != "number" ||
|
|||
|
|
typeof y != "number" ||
|
|||
|
|
typeof z != "number"
|
|||
|
|
)
|
|||
|
|
throw TypeError("The translation components must be numbers.");
|
|||
|
|
|
|||
|
|
this.translation = {x: x, y: y, z: z};
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the initial rotation of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* @param rot - The rotation to set.
|
|||
|
|
*/
|
|||
|
|
public setRotation(rot: Rotation): RigidBodyDesc {
|
|||
|
|
// #if DIM2
|
|||
|
|
this.rotation = rot;
|
|||
|
|
// #endif
|
|||
|
|
// #if DIM3
|
|||
|
|
RotationOps.copy(this.rotation, rot);
|
|||
|
|
// #endif
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the scale factor applied to the gravity affecting
|
|||
|
|
* the rigid-body being built.
|
|||
|
|
*
|
|||
|
|
* @param scale - The scale factor. Set this to `0.0` if the rigid-body
|
|||
|
|
* needs to ignore gravity.
|
|||
|
|
*/
|
|||
|
|
public setGravityScale(scale: number): RigidBodyDesc {
|
|||
|
|
this.gravityScale = scale;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the initial mass of the rigid-body being built, before adding colliders' contributions.
|
|||
|
|
*
|
|||
|
|
* @param mass − The initial mass of the rigid-body to create.
|
|||
|
|
*/
|
|||
|
|
public setAdditionalMass(mass: number): RigidBodyDesc {
|
|||
|
|
this.mass = mass;
|
|||
|
|
this.massOnly = true;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #if DIM2
|
|||
|
|
/**
|
|||
|
|
* Sets the initial linear velocity of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* @param x - The linear velocity to set along the `x` axis.
|
|||
|
|
* @param y - The linear velocity to set along the `y` axis.
|
|||
|
|
*/
|
|||
|
|
public setLinvel(x: number, y: number): RigidBodyDesc {
|
|||
|
|
if (typeof x != "number" || typeof y != "number")
|
|||
|
|
throw TypeError("The linvel components must be numbers.");
|
|||
|
|
|
|||
|
|
this.linvel = {x: x, y: y};
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the initial angular velocity of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* @param vel - The angular velocity to set.
|
|||
|
|
*/
|
|||
|
|
public setAngvel(vel: number): RigidBodyDesc {
|
|||
|
|
this.angvel = vel;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the mass properties of the rigid-body being built.
|
|||
|
|
*
|
|||
|
|
* Note that the final mass properties of the rigid-bodies depends
|
|||
|
|
* on the initial mass-properties of the rigid-body (set by this method)
|
|||
|
|
* to which is added the contributions of all the colliders with non-zero density
|
|||
|
|
* attached to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* Therefore, if you want your provided mass properties to be the final
|
|||
|
|
* mass properties of your rigid-body, don't attach colliders to it, or
|
|||
|
|
* only attach colliders with densities equal to zero.
|
|||
|
|
*
|
|||
|
|
* @param mass − The initial mass of the rigid-body to create.
|
|||
|
|
* @param centerOfMass − The initial center-of-mass of the rigid-body to create.
|
|||
|
|
* @param principalAngularInertia − The initial principal angular inertia of the rigid-body to create.
|
|||
|
|
*/
|
|||
|
|
public setAdditionalMassProperties(
|
|||
|
|
mass: number,
|
|||
|
|
centerOfMass: Vector,
|
|||
|
|
principalAngularInertia: number,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
this.mass = mass;
|
|||
|
|
VectorOps.copy(this.centerOfMass, centerOfMass);
|
|||
|
|
this.principalAngularInertia = principalAngularInertia;
|
|||
|
|
this.massOnly = false;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Allow translation of this rigid-body only along specific axes.
|
|||
|
|
* @param translationsEnabledX - Are translations along the X axis enabled?
|
|||
|
|
* @param translationsEnabledY - Are translations along the y axis enabled?
|
|||
|
|
*/
|
|||
|
|
public enabledTranslations(
|
|||
|
|
translationsEnabledX: boolean,
|
|||
|
|
translationsEnabledY: boolean,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
this.translationsEnabledX = translationsEnabledX;
|
|||
|
|
this.translationsEnabledY = translationsEnabledY;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Allow translation of this rigid-body only along specific axes.
|
|||
|
|
* @param translationsEnabledX - Are translations along the X axis enabled?
|
|||
|
|
* @param translationsEnabledY - Are translations along the y axis enabled?
|
|||
|
|
* @deprecated use `this.enabledTranslations` with the same arguments instead.
|
|||
|
|
*/
|
|||
|
|
public restrictTranslations(
|
|||
|
|
translationsEnabledX: boolean,
|
|||
|
|
translationsEnabledY: boolean,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
return this.enabledTranslations(
|
|||
|
|
translationsEnabledX,
|
|||
|
|
translationsEnabledY,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks all translations that would have resulted from forces on
|
|||
|
|
* the created rigid-body.
|
|||
|
|
*/
|
|||
|
|
public lockTranslations(): RigidBodyDesc {
|
|||
|
|
return this.restrictTranslations(false, false);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks all rotations that would have resulted from forces on
|
|||
|
|
* the created rigid-body.
|
|||
|
|
*/
|
|||
|
|
public lockRotations(): RigidBodyDesc {
|
|||
|
|
this.rotationsEnabled = false;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
// #if DIM3
|
|||
|
|
/**
|
|||
|
|
* Sets the initial linear velocity of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* @param x - The linear velocity to set along the `x` axis.
|
|||
|
|
* @param y - The linear velocity to set along the `y` axis.
|
|||
|
|
* @param z - The linear velocity to set along the `z` axis.
|
|||
|
|
*/
|
|||
|
|
public setLinvel(x: number, y: number, z: number): RigidBodyDesc {
|
|||
|
|
if (
|
|||
|
|
typeof x != "number" ||
|
|||
|
|
typeof y != "number" ||
|
|||
|
|
typeof z != "number"
|
|||
|
|
)
|
|||
|
|
throw TypeError("The linvel components must be numbers.");
|
|||
|
|
|
|||
|
|
this.linvel = {x: x, y: y, z: z};
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the initial angular velocity of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* @param vel - The angular velocity to set.
|
|||
|
|
*/
|
|||
|
|
public setAngvel(vel: Vector): RigidBodyDesc {
|
|||
|
|
VectorOps.copy(this.angvel, vel);
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the mass properties of the rigid-body being built.
|
|||
|
|
*
|
|||
|
|
* Note that the final mass properties of the rigid-bodies depends
|
|||
|
|
* on the initial mass-properties of the rigid-body (set by this method)
|
|||
|
|
* to which is added the contributions of all the colliders with non-zero density
|
|||
|
|
* attached to this rigid-body.
|
|||
|
|
*
|
|||
|
|
* Therefore, if you want your provided mass properties to be the final
|
|||
|
|
* mass properties of your rigid-body, don't attach colliders to it, or
|
|||
|
|
* only attach colliders with densities equal to zero.
|
|||
|
|
*
|
|||
|
|
* @param mass − The initial mass of the rigid-body to create.
|
|||
|
|
* @param centerOfMass − The initial center-of-mass of the rigid-body to create.
|
|||
|
|
* @param principalAngularInertia − The initial principal angular inertia of the rigid-body to create.
|
|||
|
|
* These are the eigenvalues of the angular inertia matrix.
|
|||
|
|
* @param angularInertiaLocalFrame − The initial local angular inertia frame of the rigid-body to create.
|
|||
|
|
* These are the eigenvectors of the angular inertia matrix.
|
|||
|
|
*/
|
|||
|
|
public setAdditionalMassProperties(
|
|||
|
|
mass: number,
|
|||
|
|
centerOfMass: Vector,
|
|||
|
|
principalAngularInertia: Vector,
|
|||
|
|
angularInertiaLocalFrame: Rotation,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
this.mass = mass;
|
|||
|
|
VectorOps.copy(this.centerOfMass, centerOfMass);
|
|||
|
|
VectorOps.copy(this.principalAngularInertia, principalAngularInertia);
|
|||
|
|
RotationOps.copy(
|
|||
|
|
this.angularInertiaLocalFrame,
|
|||
|
|
angularInertiaLocalFrame,
|
|||
|
|
);
|
|||
|
|
this.massOnly = false;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Allow translation of this rigid-body only along specific axes.
|
|||
|
|
* @param translationsEnabledX - Are translations along the X axis enabled?
|
|||
|
|
* @param translationsEnabledY - Are translations along the y axis enabled?
|
|||
|
|
* @param translationsEnabledZ - Are translations along the Z axis enabled?
|
|||
|
|
*/
|
|||
|
|
public enabledTranslations(
|
|||
|
|
translationsEnabledX: boolean,
|
|||
|
|
translationsEnabledY: boolean,
|
|||
|
|
translationsEnabledZ: boolean,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
this.translationsEnabledX = translationsEnabledX;
|
|||
|
|
this.translationsEnabledY = translationsEnabledY;
|
|||
|
|
this.translationsEnabledZ = translationsEnabledZ;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Allow translation of this rigid-body only along specific axes.
|
|||
|
|
* @param translationsEnabledX - Are translations along the X axis enabled?
|
|||
|
|
* @param translationsEnabledY - Are translations along the y axis enabled?
|
|||
|
|
* @param translationsEnabledZ - Are translations along the Z axis enabled?
|
|||
|
|
* @deprecated use `this.enabledTranslations` with the same arguments instead.
|
|||
|
|
*/
|
|||
|
|
public restrictTranslations(
|
|||
|
|
translationsEnabledX: boolean,
|
|||
|
|
translationsEnabledY: boolean,
|
|||
|
|
translationsEnabledZ: boolean,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
return this.enabledTranslations(
|
|||
|
|
translationsEnabledX,
|
|||
|
|
translationsEnabledY,
|
|||
|
|
translationsEnabledZ,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks all translations that would have resulted from forces on
|
|||
|
|
* the created rigid-body.
|
|||
|
|
*/
|
|||
|
|
public lockTranslations(): RigidBodyDesc {
|
|||
|
|
return this.enabledTranslations(false, false, false);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Allow rotation of this rigid-body only along specific axes.
|
|||
|
|
* @param rotationsEnabledX - Are rotations along the X axis enabled?
|
|||
|
|
* @param rotationsEnabledY - Are rotations along the y axis enabled?
|
|||
|
|
* @param rotationsEnabledZ - Are rotations along the Z axis enabled?
|
|||
|
|
*/
|
|||
|
|
public enabledRotations(
|
|||
|
|
rotationsEnabledX: boolean,
|
|||
|
|
rotationsEnabledY: boolean,
|
|||
|
|
rotationsEnabledZ: boolean,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
this.rotationsEnabledX = rotationsEnabledX;
|
|||
|
|
this.rotationsEnabledY = rotationsEnabledY;
|
|||
|
|
this.rotationsEnabledZ = rotationsEnabledZ;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Allow rotation of this rigid-body only along specific axes.
|
|||
|
|
* @param rotationsEnabledX - Are rotations along the X axis enabled?
|
|||
|
|
* @param rotationsEnabledY - Are rotations along the y axis enabled?
|
|||
|
|
* @param rotationsEnabledZ - Are rotations along the Z axis enabled?
|
|||
|
|
* @deprecated use `this.enabledRotations` with the same arguments instead.
|
|||
|
|
*/
|
|||
|
|
public restrictRotations(
|
|||
|
|
rotationsEnabledX: boolean,
|
|||
|
|
rotationsEnabledY: boolean,
|
|||
|
|
rotationsEnabledZ: boolean,
|
|||
|
|
): RigidBodyDesc {
|
|||
|
|
return this.enabledRotations(
|
|||
|
|
rotationsEnabledX,
|
|||
|
|
rotationsEnabledY,
|
|||
|
|
rotationsEnabledZ,
|
|||
|
|
);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Locks all rotations that would have resulted from forces on
|
|||
|
|
* the created rigid-body.
|
|||
|
|
*/
|
|||
|
|
public lockRotations(): RigidBodyDesc {
|
|||
|
|
return this.restrictRotations(false, false, false);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// #endif
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the linear damping of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* This will progressively slowdown the translational movement of the rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param damping - The angular damping coefficient. Should be >= 0. The higher this
|
|||
|
|
* value is, the stronger the translational slowdown will be.
|
|||
|
|
*/
|
|||
|
|
public setLinearDamping(damping: number): RigidBodyDesc {
|
|||
|
|
this.linearDamping = damping;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the angular damping of the rigid-body to create.
|
|||
|
|
*
|
|||
|
|
* This will progressively slowdown the rotational movement of the rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param damping - The angular damping coefficient. Should be >= 0. The higher this
|
|||
|
|
* value is, the stronger the rotational slowdown will be.
|
|||
|
|
*/
|
|||
|
|
public setAngularDamping(damping: number): RigidBodyDesc {
|
|||
|
|
this.angularDamping = damping;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets whether or not the rigid-body to create can sleep.
|
|||
|
|
*
|
|||
|
|
* @param can - true if the rigid-body can sleep, false if it can't.
|
|||
|
|
*/
|
|||
|
|
public setCanSleep(can: boolean): RigidBodyDesc {
|
|||
|
|
this.canSleep = can;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets whether or not the rigid-body is to be created asleep.
|
|||
|
|
*
|
|||
|
|
* @param can - true if the rigid-body should be in sleep, default false.
|
|||
|
|
*/
|
|||
|
|
setSleeping(sleeping: boolean): RigidBodyDesc {
|
|||
|
|
this.sleeping = sleeping;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets whether Continuous Collision Detection (CCD) is enabled for this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param enabled - true if the rigid-body has CCD enabled.
|
|||
|
|
*/
|
|||
|
|
public setCcdEnabled(enabled: boolean): RigidBodyDesc {
|
|||
|
|
this.ccdEnabled = enabled;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the maximum prediction distance Soft Continuous Collision-Detection.
|
|||
|
|
*
|
|||
|
|
* When set to 0, soft-CCD is disabled. Soft-CCD helps prevent tunneling especially of
|
|||
|
|
* slow-but-thin to moderately fast objects. The soft CCD prediction distance indicates how
|
|||
|
|
* far in the object’s path the CCD algorithm is allowed to inspect. Large values can impact
|
|||
|
|
* performance badly by increasing the work needed from the broad-phase.
|
|||
|
|
*
|
|||
|
|
* It is a generally cheaper variant of regular CCD (that can be enabled with
|
|||
|
|
* `RigidBodyDesc::setCcdEnabled` since it relies on predictive constraints instead of
|
|||
|
|
* shape-cast and substeps.
|
|||
|
|
*/
|
|||
|
|
public setSoftCcdPrediction(distance: number): RigidBodyDesc {
|
|||
|
|
this.softCcdPrediction = distance;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Sets the user-defined object of this rigid-body.
|
|||
|
|
*
|
|||
|
|
* @param userData - The user-defined object to set.
|
|||
|
|
*/
|
|||
|
|
public setUserData(data?: unknown): RigidBodyDesc {
|
|||
|
|
this.userData = data;
|
|||
|
|
return this;
|
|||
|
|
}
|
|||
|
|
}
|