rapier3d/geometry/
broad_phase_qbvh.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
use crate::geometry::{BroadPhasePairEvent, ColliderHandle, ColliderPair, ColliderSet};
use parry::math::Real;
use parry::partitioning::Qbvh;
use parry::partitioning::QbvhUpdateWorkspace;
use parry::query::visitors::BoundingVolumeIntersectionsSimultaneousVisitor;

#[cfg_attr(feature = "serde-serialize", derive(Serialize, Deserialize))]
#[derive(Clone)]
pub struct BroadPhaseQbvh {
    qbvh: Qbvh<ColliderHandle>,
    stack: Vec<(u32, u32)>,
    #[cfg_attr(feature = "serde-serialize", serde(skip))]
    workspace: QbvhUpdateWorkspace,
}

impl Default for BroadPhaseQbvh {
    fn default() -> Self {
        Self::new()
    }
}

impl BroadPhaseQbvh {
    pub fn new() -> Self {
        Self {
            qbvh: Qbvh::new(),
            stack: vec![],
            workspace: QbvhUpdateWorkspace::default(),
        }
    }

    #[allow(dead_code)] // This broad-phase is just experimental right now.
    pub fn update(
        &mut self,
        prediction_distance: Real,
        colliders: &ColliderSet,
        modified_colliders: &[ColliderHandle],
        removed_colliders: &[ColliderHandle],
        events: &mut Vec<BroadPhasePairEvent>,
    ) {
        let margin = 0.01;

        if modified_colliders.is_empty() {
            return;
        }

        // Visitor to find collision pairs.
        let mut visitor = BoundingVolumeIntersectionsSimultaneousVisitor::new(
            |co1: &ColliderHandle, co2: &ColliderHandle| {
                events.push(BroadPhasePairEvent::AddPair(ColliderPair::new(*co1, *co2)));
                true
            },
        );

        let full_rebuild = self.qbvh.raw_nodes().is_empty();

        if full_rebuild {
            self.qbvh.clear_and_rebuild(
                colliders.iter().map(|(handle, collider)| {
                    (
                        handle,
                        collider.compute_collision_aabb(prediction_distance / 2.0),
                    )
                }),
                margin,
            );
            self.qbvh
                .traverse_bvtt_with_stack(&self.qbvh, &mut visitor, &mut self.stack);
        } else {
            for modified in modified_colliders {
                self.qbvh.pre_update_or_insert(*modified);
            }

            for removed in removed_colliders {
                self.qbvh.remove(*removed);
            }

            let _ = self.qbvh.refit(margin, &mut self.workspace, |handle| {
                colliders[*handle].compute_collision_aabb(prediction_distance / 2.0)
            });
            self.qbvh
                .traverse_modified_bvtt_with_stack(&self.qbvh, &mut visitor, &mut self.stack);
            self.qbvh.rebalance(margin, &mut self.workspace);
        }
    }
}