use crate::bounding_volume::{self, BoundingVolume, AABB};
use crate::math::{Isometry, Point, Vector, DIM};
use crate::partitioning::{BVHImpl, BVT};
use crate::query::{
Contact, ContactKinematic, ContactPrediction, ContactPreprocessor, LocalShapeApproximation,
NeighborhoodGeometry,
};
use crate::shape::{CompositeShape, DeformableShape, DeformationsType, FeatureId, Segment, Shape};
use na::{self, Point2, RealField, Unit};
use std::iter;
use std::ops::Range;
use std::slice;
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[derive(Clone)]
struct DeformationInfos<N: RealField + Copy> {
margin: N,
curr_timestamp: usize,
timestamps: Vec<usize>,
ref_vertices: Vec<Point<N>>,
seg_to_update: Vec<usize>,
}
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[derive(Clone)]
pub struct PolylineEdge<N: RealField + Copy> {
pub indices: Point2<usize>,
bvt_leaf: usize,
pub normal: Option<Unit<Vector<N>>>, }
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[derive(Clone)]
pub struct PolylineVertex {
pub adj_edges: Range<usize>,
pub adj_vertices: Range<usize>,
}
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[derive(Clone)]
pub struct Polyline<N: RealField + Copy> {
bvt: BVT<usize, AABB<N>>,
points: Vec<Point<N>>,
vertices: Vec<PolylineVertex>,
edges: Vec<PolylineEdge<N>>,
adj_edge_list: Vec<usize>,
adj_vertex_list: Vec<usize>,
deformations: DeformationInfos<N>,
oriented: bool, }
impl<N: RealField + Copy> Polyline<N> {
pub fn new(points: Vec<Point<N>>, indices: Option<Vec<Point2<usize>>>) -> Polyline<N> {
let indices = indices.unwrap_or(
(0..)
.map(|i| Point2::new(i, i + 1))
.take(points.len() - 1)
.collect(),
);
let mut leaves = Vec::with_capacity(indices.len());
let mut vertices: Vec<PolylineVertex> = iter::repeat(PolylineVertex {
adj_edges: 0..0,
adj_vertices: 0..0,
})
.take(points.len())
.collect();
let mut edges = Vec::with_capacity(indices.len());
let adj_edge_list = Self::adj_edge_list(&indices, &mut vertices);
let adj_vertex_list = Self::adj_vertex_list(&indices, &mut vertices);
{
let is = &*indices;
for (i, is) in is.iter().enumerate() {
let segment = Segment::new(points[is.x], points[is.y]);
let normal = segment.normal();
let bv = segment.local_aabb();
leaves.push((i, bv.clone()));
edges.push(PolylineEdge {
indices: *is,
bvt_leaf: 0, normal,
})
}
}
let bvt = BVT::new_balanced(leaves);
for (i, leaf) in bvt.leaves().iter().enumerate() {
edges[*leaf.data()].bvt_leaf = i;
}
let deformations = DeformationInfos {
margin: na::convert(0.1), curr_timestamp: 0,
timestamps: Vec::new(),
ref_vertices: Vec::new(),
seg_to_update: Vec::new(),
};
Polyline {
bvt,
points,
deformations,
vertices,
edges,
adj_edge_list,
adj_vertex_list,
oriented: false,
}
}
fn adj_vertex_list(edges: &[Point2<usize>], vertices: &mut [PolylineVertex]) -> Vec<usize> {
let mut num_neighbors: Vec<usize> = iter::repeat(0).take(vertices.len()).collect();
for e in edges {
num_neighbors[e.x] += 1;
num_neighbors[e.y] += 1;
}
let mut total_num_nbh = 0;
for (num_nbh, vtx) in num_neighbors.iter().zip(vertices.iter_mut()) {
vtx.adj_vertices = total_num_nbh..total_num_nbh + num_nbh;
total_num_nbh += num_nbh;
}
let mut adj_vertex_list: Vec<usize> = iter::repeat(0).take(total_num_nbh).collect();
for n in &mut num_neighbors {
*n = 0;
}
for e in edges.iter() {
adj_vertex_list[vertices[e.x].adj_vertices.start + num_neighbors[e.x]] = e.y;
adj_vertex_list[vertices[e.y].adj_vertices.start + num_neighbors[e.y]] = e.x;
num_neighbors[e.x] += 1;
num_neighbors[e.y] += 1;
}
adj_vertex_list
}
fn adj_edge_list(edges: &[Point2<usize>], vertices: &mut [PolylineVertex]) -> Vec<usize> {
let mut num_neighbors: Vec<usize> = iter::repeat(0).take(vertices.len()).collect();
for idx in edges {
num_neighbors[idx.x] += 1;
num_neighbors[idx.y] += 1;
}
let mut total_num_nbh = 0;
for (num_nbh, vtx) in num_neighbors.iter().zip(vertices.iter_mut()) {
vtx.adj_edges = total_num_nbh..total_num_nbh + num_nbh;
total_num_nbh += num_nbh;
}
let mut adj_edge_list: Vec<usize> = iter::repeat(0).take(total_num_nbh).collect();
for n in &mut num_neighbors {
*n = 0;
}
for (i, idx) in edges.iter().enumerate() {
adj_edge_list[vertices[idx.x].adj_edges.start + num_neighbors[idx.x]] = i;
adj_edge_list[vertices[idx.y].adj_edges.start + num_neighbors[idx.y]] = i;
num_neighbors[idx.x] += 1;
num_neighbors[idx.y] += 1;
}
adj_edge_list
}
pub fn quad(nx: usize, ny: usize) -> Self {
let mut vertices = Vec::new();
let step_x = N::one() / na::convert(nx as f64);
let step_y = N::one() / na::convert(ny as f64);
let _0_5: N = na::convert(0.5);
for i in 0..=nx {
vertices.push(xy_point(step_x * na::convert(i as f64) - _0_5, -_0_5));
}
for j in 1..=ny {
vertices.push(xy_point(_0_5, step_y * na::convert(j as f64) - _0_5));
}
for i in 1..=nx {
vertices.push(xy_point(_0_5 - step_x * na::convert(i as f64), _0_5));
}
for j in 1..ny {
vertices.push(xy_point(-_0_5, _0_5 - step_y * na::convert(j as f64)));
}
let mut indices: Vec<_> = (0..)
.map(|i| Point2::new(i, i + 1))
.take(vertices.len() - 1)
.collect();
indices.push(Point2::new(vertices.len() - 1, 0));
Polyline::new(vertices, Some(indices))
}
#[inline]
pub fn aabb(&self) -> &AABB<N> {
self.bvt
.root_bounding_volume()
.expect("An empty Polyline has no AABB.")
}
#[inline]
pub fn points(&self) -> &[Point<N>] {
&self.points
}
#[inline]
pub fn edges(&self) -> &[PolylineEdge<N>] {
&self.edges
}
#[inline]
pub fn oriented(&self) -> bool {
self.oriented
}
#[inline]
pub fn set_oriented(&mut self, oriented: bool) {
self.oriented = oriented
}
#[inline]
pub fn edge_containing_feature(&self, id: FeatureId) -> usize {
match id {
FeatureId::Vertex(i) => self.adj_edge_list[self.vertices[i].adj_edges.start],
#[cfg(feature = "dim3")]
FeatureId::Edge(i) => i,
FeatureId::Face(i) => i % self.edges.len(),
_ => panic!("Feature ID cannot be unknown."),
}
}
#[inline]
pub fn segment_feature_to_polyline_feature(
&self,
edge_id: usize,
feature: FeatureId,
) -> FeatureId {
let edge = &self.edges[edge_id];
match feature {
FeatureId::Vertex(i) => FeatureId::Vertex(edge.indices[i]),
#[cfg(feature = "dim3")]
FeatureId::Edge(_) => FeatureId::Edge(edge_id),
FeatureId::Face(i) => {
if i == 0 {
FeatureId::Face(edge_id)
} else {
FeatureId::Face(edge_id + self.edges.len())
}
}
FeatureId::Unknown => FeatureId::Unknown,
}
}
#[inline]
pub fn edge_segment(&self, i: usize) -> Segment<N> {
let edge = &self.edges[i];
Segment::new(self.points[edge.indices.x], self.points[edge.indices.y])
}
#[inline]
pub fn segment_at(&self, i: usize) -> Segment<N> {
let idx = self.edges[i].indices;
Segment::new(self.points[idx.x], self.points[idx.y])
}
#[inline]
pub fn bvt(&self) -> &BVT<usize, AABB<N>> {
&self.bvt
}
#[cfg(feature = "dim3")]
pub fn vertex_tangent_cone_contains_dir(
&self,
_i: usize,
_deformations: Option<&[N]>,
_dir: &Unit<Vector<N>>,
) -> bool {
return false;
}
#[cfg(feature = "dim2")]
pub fn vertex_tangent_cone_contains_dir(
&self,
i: usize,
deformations: Option<&[N]>,
dir: &Unit<Vector<N>>,
) -> bool {
if !self.oriented {
return false;
}
let v = &self.vertices[i];
if let Some(coords) = deformations {
for adj_edge in &self.adj_edge_list[v.adj_edges.clone()] {
let indices = self.edges[*adj_edge].indices * DIM;
let seg = Segment::new(
Point::from_slice(&coords[indices.x..indices.x + DIM]),
Point::from_slice(&coords[indices.y..indices.y + DIM]),
);
if seg.scaled_normal().dot(dir) > N::zero() {
return false;
}
}
} else {
for adj_edge in &self.adj_edge_list[v.adj_edges.clone()] {
let edge = &self.edges[*adj_edge];
if let Some(ref n) = edge.normal {
if n.dot(dir) > N::zero() {
return false;
}
}
}
}
true
}
pub fn transform_by(&mut self, transform: &Isometry<N>) {
for pt in &mut self.points {
*pt = transform * *pt
}
}
#[inline]
pub fn transformed(mut self, t: &Isometry<N>) -> Self {
self.transform_by(t);
self
}
pub fn scale_by(&mut self, scale: &Vector<N>) {
for pt in &mut self.points {
pt.coords.component_mul_assign(scale)
}
}
#[inline]
pub fn scaled(mut self, s: &Vector<N>) -> Self {
self.scale_by(s);
self
}
#[inline]
pub fn is_backface(&self, feature: FeatureId) -> bool {
if let FeatureId::Face(i) = feature {
i >= self.edges.len()
} else {
false
}
}
pub fn vertex_tangent_cone_polar_contains_dir(
&self,
i: usize,
dir: &Unit<Vector<N>>,
sin_ang_tol: N,
) -> bool {
let v = &self.vertices[i];
for adj_vtx in &self.adj_vertex_list[v.adj_vertices.clone()] {
let edge_dir = self.points[i] - self.points[*adj_vtx];
if edge_dir.dot(dir) < -sin_ang_tol * edge_dir.norm() {
return false;
}
}
true
}
#[cfg(feature = "dim3")]
pub fn edge_tangent_cone_contains_dir(
&self,
_i: usize,
_deformations: Option<&[N]>,
_dir: &Unit<Vector<N>>,
) -> bool {
return false;
}
#[cfg(feature = "dim2")]
pub fn edge_tangent_cone_contains_dir(
&self,
i: usize,
deformations: Option<&[N]>,
dir: &Unit<Vector<N>>,
) -> bool {
if !self.oriented {
return false;
}
let normal;
if let Some(coords) = deformations {
let indices = self.edges[i % self.edges.len()].indices * DIM;
let seg = Segment::new(
Point::from_slice(&coords[indices.x..indices.x + DIM]),
Point::from_slice(&coords[indices.y..indices.y + DIM]),
);
if i >= self.edges.len() {
normal = -seg.scaled_normal();
} else {
normal = seg.scaled_normal();
}
} else {
if i >= self.edges.len() {
normal = -self.edges[i - self.edges.len()]
.normal
.map(|n| n.into_inner())
.unwrap_or(Vector::zeros());
} else {
normal = self.edges[i]
.normal
.map(|n| n.into_inner())
.unwrap_or(Vector::zeros());
}
}
normal.dot(dir) <= N::zero()
}
pub fn edge_tangent_cone_polar_contains_dir(
&self,
i: usize,
dir: &Unit<Vector<N>>,
cos_ang_tol: N,
) -> bool {
let normal;
if i >= self.edges.len() {
normal = -self.edges[i - self.edges.len()]
.normal
.map(|n| n.into_inner())
.unwrap_or(Vector::zeros());
} else {
normal = self.edges[i]
.normal
.map(|n| n.into_inner())
.unwrap_or(Vector::zeros());
}
normal.dot(dir) >= cos_ang_tol
}
pub fn tangent_cone_polar_contains_dir(
&self,
_feature: FeatureId,
_dir: &Unit<Vector<N>>,
_sin_ang_tol: N,
_cos_ang_tol: N,
) -> bool {
unimplemented!()
}
fn init_deformation_infos(&mut self) -> bool {
if self.deformations.ref_vertices.is_empty() {
self.deformations.timestamps = iter::repeat(0).take(self.edges.len()).collect();
self.deformations.ref_vertices = self.points.clone();
true
} else {
false
}
}
}
impl<N: RealField + Copy> CompositeShape<N> for Polyline<N> {
#[inline]
fn nparts(&self) -> usize {
self.edges.len()
}
#[inline(always)]
fn map_part_at(
&self,
i: usize,
m: &Isometry<N>,
f: &mut dyn FnMut(&Isometry<N>, &dyn Shape<N>),
) {
let element = self.segment_at(i);
f(m, &element)
}
fn map_part_and_preprocessor_at(
&self,
i: usize,
m: &Isometry<N>,
prediction: &ContactPrediction<N>,
f: &mut dyn FnMut(&Isometry<N>, &dyn Shape<N>, &dyn ContactPreprocessor<N>),
) {
let element = self.segment_at(i);
let proc = PolylineContactProcessor::new(self, m, i, prediction);
f(m, &element, &proc)
}
#[inline]
fn aabb_at(&self, i: usize) -> AABB<N> {
self.bvt
.leaf(self.edges[i].bvt_leaf)
.bounding_volume()
.clone()
}
#[inline]
fn bvh(&self) -> BVHImpl<N, usize, AABB<N>> {
BVHImpl::BVT(&self.bvt)
}
}
impl<N: RealField + Copy> DeformableShape<N> for Polyline<N> {
fn deformations_type(&self) -> DeformationsType {
DeformationsType::Vectors
}
fn set_deformations(&mut self, coords: &[N]) {
assert!(
coords.len() >= self.points.len() * DIM,
"Set deformations error: dimension mismatch."
);
let is_first_init = self.init_deformation_infos();
self.deformations.curr_timestamp += 1;
unsafe {
let len = self.points.len();
let coords_ptr = coords.as_ptr() as *const Point<N>;
let coords_pt: &[Point<N>] = slice::from_raw_parts(coords_ptr, len);
self.points.copy_from_slice(coords_pt);
}
for (target, pt) in self.points.iter_mut().enumerate() {
let ref_pt = &mut self.deformations.ref_vertices[target];
let sq_dist_to_ref = na::distance_squared(pt, ref_pt);
if is_first_init || sq_dist_to_ref > self.deformations.margin * self.deformations.margin
{
let ids = self.vertices[target].adj_edges.clone();
self.deformations
.seg_to_update
.extend_from_slice(&self.adj_edge_list[ids]);
*ref_pt = *pt;
}
}
for e in &mut self.edges {
let seg = Segment::new(self.points[e.indices.x], self.points[e.indices.y]);
e.normal = seg.normal();
}
for seg_id in self.deformations.seg_to_update.drain(..) {
if self.deformations.timestamps[seg_id] != self.deformations.curr_timestamp {
let idx = &self.edges[seg_id].indices;
let mut new_bv = bounding_volume::local_point_cloud_aabb(&[
self.points[idx.x],
self.points[idx.y],
]);
new_bv.loosen(self.deformations.margin);
self.bvt
.set_leaf_bounding_volume(self.edges[seg_id].bvt_leaf, new_bv, false);
self.deformations.timestamps[seg_id] = self.deformations.curr_timestamp;
}
}
self.bvt.refit(N::zero())
}
fn update_local_approximation(&self, coords: &[N], approx: &mut LocalShapeApproximation<N>) {
match approx.feature {
FeatureId::Vertex(i) => {
approx.point = Point::from_slice(&coords[i * DIM..(i + 1) * DIM]);
approx.geometry = NeighborhoodGeometry::Point;
}
#[cfg(feature = "dim3")]
FeatureId::Edge(i) => {
let edge = &self.edges[i];
let pid1 = edge.indices.x * DIM;
let pid2 = edge.indices.y * DIM;
let seg = Segment::new(
Point::from_slice(&coords[pid1..pid1 + DIM]),
Point::from_slice(&coords[pid2..pid2 + DIM]),
);
approx.point = seg.a;
if let Some(dir) = seg.direction() {
approx.geometry = NeighborhoodGeometry::Line(dir);
} else {
approx.geometry = NeighborhoodGeometry::Point;
}
}
#[cfg(feature = "dim3")]
FeatureId::Face(_) => unreachable!(),
#[cfg(feature = "dim2")]
FeatureId::Face(mut i) => {
let is_backface = i >= self.edges.len();
if is_backface {
i -= self.edges.len();
}
let edge = &self.edges[i];
let pid1 = edge.indices.x * DIM;
let pid2 = edge.indices.y * DIM;
let seg = Segment::new(
Point::from_slice(&coords[pid1..pid1 + DIM]),
Point::from_slice(&coords[pid2..pid2 + DIM]),
);
approx.point = seg.a;
if let Some(n) = seg.normal() {
if !is_backface {
approx.geometry = NeighborhoodGeometry::Plane(n);
} else {
approx.geometry = NeighborhoodGeometry::Plane(-n);
}
} else {
approx.geometry = NeighborhoodGeometry::Point;
}
}
_ => panic!(
"Encountered invalid triangle feature: {:?}.",
approx.feature
),
}
}
}
#[allow(dead_code)]
struct PolylineContactProcessor<'a, N: RealField + Copy> {
polyline: &'a Polyline<N>,
pos: &'a Isometry<N>,
edge_id: usize,
prediction: &'a ContactPrediction<N>,
}
impl<'a, N: RealField + Copy> PolylineContactProcessor<'a, N> {
pub fn new(
polyline: &'a Polyline<N>,
pos: &'a Isometry<N>,
edge_id: usize,
prediction: &'a ContactPrediction<N>,
) -> Self {
PolylineContactProcessor {
polyline,
pos,
edge_id,
prediction,
}
}
}
impl<'a, N: RealField + Copy> ContactPreprocessor<N> for PolylineContactProcessor<'a, N> {
fn process_contact(
&self,
_c: &mut Contact<N>,
kinematic: &mut ContactKinematic<N>,
is_first: bool,
) -> bool {
let feature = if is_first {
kinematic.feature1()
} else {
kinematic.feature2()
};
let edge = &self.polyline.edges()[self.edge_id];
let actual_feature = match feature {
FeatureId::Vertex(i) => FeatureId::Vertex(edge.indices[i]),
#[cfg(feature = "dim3")]
FeatureId::Edge(_) => FeatureId::Edge(self.edge_id),
FeatureId::Face(i) => {
if i == 0 {
FeatureId::Face(self.edge_id)
} else {
FeatureId::Face(self.edge_id + self.polyline.edges().len())
}
}
FeatureId::Unknown => FeatureId::Unknown,
};
if is_first {
kinematic.set_feature1(actual_feature);
} else {
kinematic.set_feature2(actual_feature);
}
true
}
}
#[cfg(feature = "dim2")]
fn xy_point<N: RealField + Copy>(x: N, y: N) -> Point<N> {
Point::new(x, y)
}
#[cfg(feature = "dim3")]
fn xy_point<N: RealField + Copy>(x: N, y: N) -> Point<N> {
Point::new(x, y, N::zero())
}