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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
use crate::bounding_volume::BoundingVolume;
use crate::math::Isometry;
use crate::partitioning::{BestFirstVisitStatus, BestFirstVisitor};
use crate::query::{PointQuery, Ray, RayCast, RayIntersection};
use na::RealField;
use std::any::Any;
use crate::pipeline::{BroadPhase, BroadPhaseProxyHandle};
pub struct RayIntersectionCostFnVisitor<'a, 'b, N: 'a + RealField, T, BV>
where
BV: BoundingVolume<N> + RayCast<N> + PointQuery<N> + Any + Send + Sync + Clone,
T: Any + Send + Sync,
{
ray: &'b Ray<N>,
max_toi: N,
broad_phase: &'a dyn BroadPhase<N, BV, T>,
cost_fn: &'a dyn Fn(T, &'b Ray<N>, N) -> Option<(T, RayIntersection<N>)>,
}
impl<'a, 'b, N: RealField, T, BV> RayIntersectionCostFnVisitor<'a, 'b, N, T, BV>
where
BV: BoundingVolume<N> + RayCast<N> + PointQuery<N> + Any + Send + Sync + Clone,
T: Any + Send + Sync,
{
#[inline]
pub fn new(
ray: &'b Ray<N>,
max_toi: N,
broad_phase: &'a dyn BroadPhase<N, BV, T>,
cost_fn: &'a dyn Fn(T, &'b Ray<N>, N) -> Option<(T, RayIntersection<N>)>,
) -> RayIntersectionCostFnVisitor<'a, 'b, N, T, BV> {
RayIntersectionCostFnVisitor {
ray,
max_toi,
broad_phase,
cost_fn,
}
}
}
impl<'a, 'b, N, BV, T> BestFirstVisitor<N, BroadPhaseProxyHandle, BV>
for RayIntersectionCostFnVisitor<'a, 'b, N, T, BV>
where
N: RealField,
BV: BoundingVolume<N> + RayCast<N> + PointQuery<N> + Any + Send + Sync + Clone,
T: Any + Send + Sync + Clone,
{
type Result = (T, RayIntersection<N>);
#[inline]
fn visit(
&mut self,
best_cost_so_far: N,
bv: &BV,
data: Option<&BroadPhaseProxyHandle>,
) -> BestFirstVisitStatus<N, Self::Result> {
if let Some(rough_toi) =
bv.toi_with_ray(&Isometry::identity(), self.ray, self.max_toi, true)
{
let mut res = BestFirstVisitStatus::Continue {
cost: rough_toi,
result: None,
};
if let Some(data_handle) = data {
if rough_toi < best_cost_so_far {
if let Some((_, leaf_data)) = self.broad_phase.proxy(*data_handle) {
if let Some(result) =
(self.cost_fn)(leaf_data.clone(), self.ray, self.max_toi)
{
res = BestFirstVisitStatus::Continue {
cost: result.1.toi,
result: Some(result),
};
}
}
};
}
res
} else {
BestFirstVisitStatus::Stop
}
}
}