bevy_rapier2d/plugin/systems/
multiple_rapier_contexts.rsuse crate::dynamics::{
RapierImpulseJointHandle, RapierMultibodyJointHandle, RapierRigidBodyHandle,
};
use crate::geometry::RapierColliderHandle;
use crate::plugin::context::RapierRigidBodySet;
use crate::plugin::context::{
RapierContextColliders, RapierContextEntityLink, RapierContextJoints,
};
use bevy::prelude::*;
pub fn on_add_entity_with_parent(
q_add_entity_without_parent: Query<
(Entity, &Parent),
(
With<RapierContextEntityLink>,
Or<(Changed<RapierContextEntityLink>, Changed<Parent>)>,
),
>,
q_parent: Query<&Parent>,
q_physics_world: Query<&RapierContextEntityLink>,
mut commands: Commands,
) {
for (ent, parent) in &q_add_entity_without_parent {
let mut parent = Some(parent.get());
while let Some(parent_entity) = parent {
if let Ok(pw) = q_physics_world.get(parent_entity) {
if q_physics_world.get(ent).map(|x| x != pw).unwrap_or(true) {
remove_old_physics(ent, &mut commands);
commands.entity(ent).insert(*pw);
}
break;
}
parent = q_parent.get(parent_entity).ok().map(|x| x.get());
}
}
}
fn remove_old_physics(entity: Entity, commands: &mut Commands) {
commands
.entity(entity)
.remove::<RapierColliderHandle>()
.remove::<RapierRigidBodyHandle>()
.remove::<RapierMultibodyJointHandle>()
.remove::<RapierImpulseJointHandle>();
}
pub fn on_change_context(
q_changed_contexts: Query<
(Entity, Ref<RapierContextEntityLink>),
Changed<RapierContextEntityLink>,
>,
q_children: Query<&Children>,
q_physics_context: Query<&RapierContextEntityLink>,
q_context: Query<(
&RapierContextColliders,
&RapierContextJoints,
&RapierRigidBodySet,
)>,
mut commands: Commands,
) {
for (entity, new_physics_context) in &q_changed_contexts {
let context = q_context.get(new_physics_context.0);
if !context
.map(|(colliders, joints, rigidbody_set)| {
colliders.entity2collider.contains_key(&entity)
|| rigidbody_set.entity2body.contains_key(&entity)
|| joints.entity2impulse_joint.contains_key(&entity)
|| joints.entity2multibody_joint.contains_key(&entity)
})
.unwrap_or(false)
{
remove_old_physics(entity, &mut commands);
bubble_down_context_change(
&mut commands,
entity,
&q_children,
*new_physics_context,
&q_physics_context,
);
}
}
}
fn bubble_down_context_change(
commands: &mut Commands,
entity: Entity,
q_children: &Query<&Children>,
new_physics_context: RapierContextEntityLink,
q_physics_context: &Query<&RapierContextEntityLink>,
) {
let Ok(children) = q_children.get(entity) else {
return;
};
children.iter().for_each(|&child| {
if q_physics_context
.get(child)
.map(|x| *x == new_physics_context)
.unwrap_or(false)
{
return;
}
remove_old_physics(child, commands);
commands.entity(child).insert(new_physics_context);
bubble_down_context_change(
commands,
child,
q_children,
new_physics_context,
q_physics_context,
);
});
}
#[cfg(test)]
mod test {
use crate::plugin::{
context::{RapierContextEntityLink, RapierContextSimulation},
NoUserData, PhysicsSet, RapierPhysicsPlugin,
};
use crate::prelude::{ActiveEvents, Collider, ContactForceEventThreshold, RigidBody, Sensor};
use bevy::prelude::*;
use bevy::time::{TimePlugin, TimeUpdateStrategy};
use rapier::math::Real;
#[test]
pub fn multi_context_hierarchy_update() {
let mut app = App::new();
app.add_plugins((
TransformPlugin,
TimePlugin,
RapierPhysicsPlugin::<NoUserData>::default(),
))
.add_systems(
PostUpdate,
setup_physics
.run_if(run_once)
.before(PhysicsSet::SyncBackend),
);
app.insert_resource(TimeUpdateStrategy::ManualDuration(
std::time::Duration::from_secs_f32(1f32 / 60f32),
));
app.finish();
app.update();
let world = app.world_mut();
let mut query = world.query_filtered::<Entity, With<Marker<'R'>>>();
for entity in query.iter(&world) {
world
.get::<RapierContextEntityLink>(entity)
.unwrap_or_else(|| panic!("no link to rapier context entity from {entity}."));
}
let new_rapier_context = world.spawn((RapierContextSimulation::default(),)).id();
app.update();
let mut world = app.world_mut();
let mut query = world.query_filtered::<&mut RapierContextEntityLink, With<Marker<'P'>>>();
let mut link_parent = query.get_single_mut(&mut world).unwrap();
link_parent.0 = new_rapier_context;
app.update();
let mut world = app.world_mut();
let mut query = world.query_filtered::<&RapierContextEntityLink, With<Marker<'C'>>>();
let link_child = query.get_single_mut(&mut world).unwrap();
assert_eq!(link_child.0, new_rapier_context);
return;
#[derive(Component)]
pub struct Marker<const MARKER: char>;
#[cfg(feature = "dim3")]
fn cuboid(hx: Real, hy: Real, hz: Real) -> Collider {
Collider::cuboid(hx, hy, hz)
}
#[cfg(feature = "dim2")]
fn cuboid(hx: Real, hy: Real, _hz: Real) -> Collider {
Collider::cuboid(hx, hy)
}
pub fn setup_physics(mut commands: Commands) {
commands.spawn((
Transform::from_xyz(0.0, -1.2, 0.0),
cuboid(4.0, 1.0, 1.0),
Marker::<'R'>,
));
commands.spawn((
Transform::from_xyz(0.0, 5.0, 0.0),
cuboid(4.0, 1.5, 1.0),
Sensor,
Marker::<'R'>,
));
commands
.spawn((
Transform::from_xyz(0.0, 13.0, 0.0),
RigidBody::Dynamic,
cuboid(0.5, 0.5, 0.5),
ActiveEvents::COLLISION_EVENTS,
ContactForceEventThreshold(30.0),
Marker::<'P'>,
Marker::<'R'>,
))
.with_children(|child_builder| {
child_builder.spawn((
Transform::from_xyz(0.0, -1.2, 0.0),
cuboid(4.0, 1.0, 1.0),
Marker::<'C'>,
Marker::<'R'>,
));
});
}
}
}