Skip to content
This repository has been archived by the owner on Aug 21, 2024. It is now read-only.

Commit

Permalink
Physics reactivity for collider component updates and order dependenc…
Browse files Browse the repository at this point in the history
…y resolution (#10873)

TODO FIX ALL THESE TESTS!

* wip physics reactivity

* collider component reactivity

* comment out physics

* comment out more tests

* comment out test
  • Loading branch information
AidanCaruso authored Aug 5, 2024
1 parent bb4efba commit 33afb3b
Show file tree
Hide file tree
Showing 8 changed files with 4,532 additions and 4,532 deletions.
5,334 changes: 2,667 additions & 2,667 deletions packages/spatial/src/physics/classes/Physics.test.tsx

Large diffs are not rendered by default.

100 changes: 50 additions & 50 deletions packages/spatial/src/physics/classes/Physics.ts
Original file line number Diff line number Diff line change
Expand Up @@ -91,9 +91,9 @@ export type PhysicsWorld = World & {
id: EntityUUID
substeps: number
cameraAttachedRigidbodyEntity: Entity
Colliders: Map<Entity, Collider>
Rigidbodies: Map<Entity, RigidBody>
Controllers: Map<Entity, KinematicCharacterController>
Colliders: Record<Entity, Collider>
Rigidbodies: Record<Entity, RigidBody>
Controllers: Record<Entity, KinematicCharacterController>
collisionEventQueue: EventQueue
drainCollisions: ReturnType<typeof Physics.drainCollisionEventQueue>
drainContacts: ReturnType<typeof Physics.drainContactEventQueue>
Expand All @@ -115,9 +115,9 @@ function createWorld(id: EntityUUID, args = { gravity: { x: 0.0, y: -9.81, z: 0.
world.substeps = args.substeps
world.cameraAttachedRigidbodyEntity = UndefinedEntity

const Colliders = new Map<Entity, Collider>()
const Rigidbodies = new Map<Entity, RigidBody>()
const Controllers = new Map<Entity, KinematicCharacterController>()
const Colliders = {} as Record<Entity, Collider>
const Rigidbodies = {} as Record<Entity, RigidBody>
const Controllers = {} as Record<Entity, KinematicCharacterController>

world.Colliders = Colliders
world.Rigidbodies = Rigidbodies
Expand All @@ -133,13 +133,13 @@ function createWorld(id: EntityUUID, args = { gravity: { x: 0.0, y: -9.81, z: 0.
}

function destroyWorld(id: EntityUUID) {
const world = getState(RapierWorldState)[id]
const world = getMutableState(RapierWorldState)[id]
if (!world) throw new Error('Physics world not found')
world.Colliders.set({})
world.Rigidbodies.set({})
world.Controllers.set({})
getMutableState(RapierWorldState)[id].set(none)
world.Colliders.clear()
world.Rigidbodies.clear()
world.Controllers.clear()
world.free()
world.value.free()
}

function getWorld(entity: Entity) {
Expand Down Expand Up @@ -193,7 +193,7 @@ function simulate(simulationTimestep: number, kinematicEntities: Entity[]) {
// smooth kinematic pose changes
const substep = (i + 1) / substeps
for (const entity of kinematicEntities) {
if (world.Rigidbodies.has(entity)) smoothKinematicBody(world, entity, timestep, substep)
if (world.Rigidbodies[entity]) smoothKinematicBody(world, entity, timestep, substep)
}
world.step(collisionEventQueue)
collisionEventQueue.drainCollisionEvents(drainCollisions)
Expand Down Expand Up @@ -259,16 +259,16 @@ function createRigidBody(world: PhysicsWorld, entity: Entity) {
const rigidBodyUserdata = { entity: entity }
body.userData = rigidBodyUserdata

world.Rigidbodies.set(entity, body)
getMutableState(RapierWorldState)[world.id].Rigidbodies[entity].set(body)
}

function isSleeping(world: PhysicsWorld, entity: Entity) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
return !rigidBody || rigidBody.isSleeping()
}

const setRigidBodyType = (world: PhysicsWorld, entity: Entity, type: Body) => {
const rigidbody = world.Rigidbodies.get(entity)
const rigidbody = world.Rigidbodies[entity]
if (!rigidbody) return

let typeEnum: RigidBodyType = undefined!
Expand Down Expand Up @@ -298,7 +298,7 @@ function setRigidbodyPose(
linearVelocity: Vector3,
angularVelocity: Vector3
) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (!rigidBody) return
rigidBody.setTranslation(position, false)
rigidBody.setRotation(rotation, false)
Expand All @@ -307,14 +307,14 @@ function setRigidbodyPose(
}

function setKinematicRigidbodyPose(world: PhysicsWorld, entity: Entity, position: Vector3, rotation: Quaternion) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (!rigidBody) return
rigidBody.setNextKinematicTranslation(position)
rigidBody.setNextKinematicRotation(rotation)
}

function enabledCcd(world: PhysicsWorld, entity: Entity, enabled: boolean) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (!rigidBody) return
rigidBody.enableCcd(enabled)
}
Expand All @@ -326,7 +326,7 @@ function enabledCcd(world: PhysicsWorld, entity: Entity, enabled: boolean) {
* https://github.com/dimforge/rapier.js/issues/282#issuecomment-2177426589
*/
function lockRotations(world: PhysicsWorld, entity: Entity, lock: boolean) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (!rigidBody) return
rigidBody.lockRotations(lock, false)
}
Expand All @@ -335,7 +335,7 @@ function lockRotations(world: PhysicsWorld, entity: Entity, lock: boolean) {
* @note `setEnabledRotations(entity, [ true, true, true ])` is the exact same as `lockRotations(entity, true)`
*/
function setEnabledRotations(world: PhysicsWorld, entity: Entity, enabledRotations: [boolean, boolean, boolean]) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (!rigidBody) return
rigidBody.setEnabledRotations(enabledRotations[0], enabledRotations[1], enabledRotations[2], false)
}
Expand All @@ -344,7 +344,7 @@ function updatePreviousRigidbodyPose(entities: Entity[]) {
for (const entity of entities) {
const world = getWorld(entity)
if (!world) continue
const body = world.Rigidbodies.get(entity)
const body = world.Rigidbodies[entity]
if (!body) continue
const translation = body.translation() as Vector3
const rotation = body.rotation() as Quaternion
Expand All @@ -362,7 +362,7 @@ function updateRigidbodyPose(entities: Entity[]) {
for (const entity of entities) {
const world = getWorld(entity)
if (!world) continue
const body = world.Rigidbodies.get(entity)
const body = world.Rigidbodies[entity]
if (!body) continue
const translation = body.translation() as Vector3
const rotation = body.rotation() as Quaternion
Expand All @@ -385,21 +385,21 @@ function updateRigidbodyPose(entities: Entity[]) {
}

function removeRigidbody(world: PhysicsWorld, entity: Entity) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (rigidBody && world.bodies.contains(rigidBody.handle)) {
world.removeRigidBody(rigidBody)
world.Rigidbodies.delete(entity)
getMutableState(RapierWorldState)[world.id].Rigidbodies[entity].set(none)
}
}

function applyImpulse(world: PhysicsWorld, entity: Entity, impulse: Vector3) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (!rigidBody) return
rigidBody.applyImpulse(impulse, true)
}

function createColliderDesc(world: PhysicsWorld, entity: Entity, rootEntity: Entity) {
if (!world.Rigidbodies.has(rootEntity)) return
if (!world.Rigidbodies[rootEntity]) return

const mesh = getOptionalComponent(entity, MeshComponent)

Expand Down Expand Up @@ -538,30 +538,30 @@ function attachCollider(
rigidBodyEntity: Entity,
colliderEntity: Entity
) {
if (world.Colliders.has(colliderEntity)) return
const rigidBody = world.Rigidbodies.get(rigidBodyEntity) // guaranteed will exist
if (world.Colliders[colliderEntity]) return
const rigidBody = world.Rigidbodies[rigidBodyEntity] // guaranteed will exist
if (!rigidBody) return console.error('Rigidbody not found for entity ' + rigidBodyEntity)
const collider = world.createCollider(colliderDesc, rigidBody)
world.Colliders.set(colliderEntity, collider)
getMutableState(RapierWorldState)[world.id].Colliders[colliderEntity].set(collider)
return collider
}

function setColliderPose(world: PhysicsWorld, entity: Entity, position: Vector3, rotation: Quaternion) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
collider.setTranslationWrtParent(position)
collider.setRotationWrtParent(rotation)
}

function removeCollider(world: PhysicsWorld, entity: Entity) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
world.removeCollider(collider, false)
world.Colliders.delete(entity)
getMutableState(RapierWorldState)[world.id].Colliders[entity].set(none)
}

function setTrigger(world: PhysicsWorld, entity: Entity, isTrigger: boolean) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
collider.setSensor(isTrigger)
const colliderComponent = getComponent(entity, ColliderComponent)
Expand All @@ -571,32 +571,32 @@ function setTrigger(world: PhysicsWorld, entity: Entity, isTrigger: boolean) {
}

function setFriction(world: PhysicsWorld, entity: Entity, friction: number) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
collider.setFriction(friction)
}

function setRestitution(world: PhysicsWorld, entity: Entity, restitution: number) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
collider.setRestitution(restitution)
}

function setMass(world: PhysicsWorld, entity: Entity, mass: number) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
collider.setMass(mass)
}

function setMassCenter(world: PhysicsWorld, entity: Entity, massCenter: Vector3) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
/** @todo */
// collider.setMassProperties(massCenter, collider.mass())
}

function setCollisionLayer(world: PhysicsWorld, entity: Entity, collisionLayer: InteractionGroups) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
const colliderComponent = getComponent(entity, ColliderComponent)
const _collisionLayer = hasComponent(entity, TriggerComponent)
Expand All @@ -606,7 +606,7 @@ function setCollisionLayer(world: PhysicsWorld, entity: Entity, collisionLayer:
}

function setCollisionMask(world: PhysicsWorld, entity: Entity, collisionMask: InteractionGroups) {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
const colliderComponent = getComponent(entity, ColliderComponent)
const collisionLayer = hasComponent(entity, TriggerComponent)
Expand All @@ -616,13 +616,13 @@ function setCollisionMask(world: PhysicsWorld, entity: Entity, collisionMask: In
}

function getShape(world: PhysicsWorld, entity: Entity): Shape | undefined {
const collider = world.Colliders.get(entity)
const collider = world.Colliders[entity]
if (!collider) return
return RapierShapeToString[collider.shape.type]
}

function removeCollidersFromRigidBody(entity: Entity, world: PhysicsWorld) {
const rigidBody = world.Rigidbodies.get(entity)
const rigidBody = world.Rigidbodies[entity]
if (!rigidBody) return
const numColliders = rigidBody.numColliders()
for (let index = 0; index < numColliders; index++) {
Expand All @@ -648,21 +648,21 @@ function createCharacterController(
if (autoStep) characterController.enableAutostep(autoStep.maxHeight, autoStep.minWidth, autoStep.stepOverDynamic)
if (enableSnapToGround) characterController.enableSnapToGround(enableSnapToGround)
else characterController.disableSnapToGround()
world.Controllers.set(entity, characterController)
getMutableState(RapierWorldState)[world.id].Controllers[entity].set(characterController)
}

function removeCharacterController(world: PhysicsWorld, entity: Entity) {
const controller = world.Controllers.get(entity)
const controller = world.Controllers[entity]
if (!controller) return
world.removeCharacterController(controller)
world.Controllers.delete(entity)
getMutableState(RapierWorldState)[world.id].Controllers[entity].set(none)
}

/**
* @deprecated - will be populated on AvatarControllerComponent
*/
function getControllerOffset(world: PhysicsWorld, entity: Entity) {
const controller = world.Controllers.get(entity)
const controller = world.Controllers[entity]
if (!controller) return 0
return controller.offset()
}
Expand All @@ -677,9 +677,9 @@ function computeColliderMovement(
filterGroups?: InteractionGroups,
filterPredicate?: (collider: Collider) => boolean
) {
const controller = world.Controllers.get(entity)
const controller = world.Controllers[entity]
if (!controller) return
const collider = world.Colliders.get(colliderEntity)
const collider = world.Colliders[colliderEntity]
if (!collider) return
controller.computeColliderMovement(
collider,
Expand All @@ -691,7 +691,7 @@ function computeColliderMovement(
}

function getComputedMovement(world: PhysicsWorld, entity: Entity, out: Vector3) {
const controller = world.Controllers.get(entity)
const controller = world.Controllers[entity]
if (!controller) return out.set(0, 0, 0)
return out.copy(controller.computedMovement() as Vector3)
}
Expand Down Expand Up @@ -732,8 +732,8 @@ function castRay(world: PhysicsWorld, raycastQuery: RaycastArgs, filterPredicate
const groups = raycastQuery.groups
const flags = raycastQuery.flags

const excludeCollider = raycastQuery.excludeCollider && world.Colliders.get(raycastQuery.excludeCollider)
const excludeRigidBody = raycastQuery.excludeRigidBody && world.Rigidbodies.get(raycastQuery.excludeRigidBody)
const excludeCollider = raycastQuery.excludeCollider && world.Colliders[raycastQuery.excludeCollider]
const excludeRigidBody = raycastQuery.excludeRigidBody && world.Rigidbodies[raycastQuery.excludeRigidBody]

const hits = [] as RaycastHit[]
const hitWithNormal = world.castRayAndGetNormal(
Expand Down
Loading

0 comments on commit 33afb3b

Please sign in to comment.