use na::{DMatrix, Point3, RealField};
use crate::bounding_volume::AABB;
use crate::math::Vector;
use crate::query::{Contact, ContactKinematic, ContactPreprocessor};
use crate::shape::{FeatureId, Triangle};
bitflags! {
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[derive(Default)]
pub struct HeightFieldCellStatus: u8 {
const ZIGZAG_SUBDIVISION = 0b00000001;
const LEFT_TRIANGLE_REMOVED = 0b00000010;
const RIGHT_TRIANGLE_REMOVED = 0b00000100;
const CELL_REMOVED = Self::LEFT_TRIANGLE_REMOVED.bits | Self::RIGHT_TRIANGLE_REMOVED.bits;
}
}
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[derive(Clone, Debug)]
pub struct HeightField<N: RealField> {
heights: DMatrix<N>,
scale: Vector<N>,
aabb: AABB<N>,
num_triangles: usize,
status: DMatrix<HeightFieldCellStatus>,
}
impl<N: RealField> HeightField<N> {
pub fn new(heights: DMatrix<N>, scale: Vector<N>) -> Self {
assert!(
heights.nrows() > 1 && heights.ncols() > 1,
"A heightfield heights must have at least 2 rows and columns."
);
let max = heights.max();
let min = heights.min();
let hscale = scale * na::convert::<_, N>(0.5);
let aabb = AABB::new(
Point3::new(-hscale.x, min * scale.y, -hscale.z),
Point3::new(hscale.x, max * scale.y, hscale.z),
);
let num_triangles = (heights.nrows() - 1) * (heights.ncols() - 1) * 2;
let status = DMatrix::repeat(
heights.nrows() - 1,
heights.ncols() - 1,
HeightFieldCellStatus::default(),
);
HeightField {
heights,
scale,
aabb,
num_triangles,
status,
}
}
pub fn nrows(&self) -> usize {
self.heights.nrows() - 1
}
pub fn ncols(&self) -> usize {
self.heights.ncols() - 1
}
fn triangle_id(&self, i: usize, j: usize, left: bool) -> usize {
let tid = j * (self.heights.nrows() - 1) + i;
if left {
tid
} else {
tid + self.num_triangles / 2
}
}
fn face_id(&self, i: usize, j: usize, left: bool, front: bool) -> usize {
let tid = self.triangle_id(i, j, left);
if front {
tid
} else {
tid + self.num_triangles
}
}
fn quantize_floor(&self, val: N, cell_size: N, num_cells: usize) -> usize {
let _0_5: N = na::convert(0.5);
let i = na::clamp(
((val + _0_5) / cell_size).floor(),
N::zero(),
na::convert((num_cells - 1) as f64),
);
unsafe { na::convert_unchecked::<N, f64>(i) as usize }
}
fn quantize_ceil(&self, val: N, cell_size: N, num_cells: usize) -> usize {
let _0_5: N = na::convert(0.5);
let i = na::clamp(
((val + _0_5) / cell_size).ceil(),
N::zero(),
na::convert(num_cells as f64),
);
unsafe { na::convert_unchecked::<N, f64>(i) as usize }
}
pub fn cell_at_point(&self, pt: &Point3<N>) -> Option<(usize, usize)> {
let _0_5: N = na::convert(0.5);
let scaled_pt = pt.coords.component_div(&self.scale);
let cell_width = self.unit_cell_width();
let cell_height = self.unit_cell_height();
let ncells_x = self.ncols();
let ncells_z = self.nrows();
if scaled_pt.x < -_0_5 || scaled_pt.x > _0_5 || scaled_pt.z < -_0_5 || scaled_pt.z > _0_5 {
None
} else {
let j = self.quantize_floor(scaled_pt.x, cell_width, ncells_x);
let i = self.quantize_floor(scaled_pt.z, cell_height, ncells_z);
Some((i, j))
}
}
pub fn x_at(&self, j: usize) -> N {
let _0_5: N = na::convert(0.5);
(-_0_5 + self.unit_cell_width() * na::convert(j as f64)) * self.scale.x
}
pub fn z_at(&self, i: usize) -> N {
let _0_5: N = na::convert(0.5);
(-_0_5 + self.unit_cell_height() * na::convert(i as f64)) * self.scale.z
}
pub fn triangles<'a>(&'a self) -> impl Iterator<Item = Triangle<N>> + 'a {
HeightfieldTriangles {
heightfield: self,
curr: (0, 0),
tris: self.triangles_at(0, 0),
}
}
pub fn triangles_at(&self, i: usize, j: usize) -> (Option<Triangle<N>>, Option<Triangle<N>>) {
let status = self.status[(i, j)];
if status.contains(
HeightFieldCellStatus::LEFT_TRIANGLE_REMOVED
| HeightFieldCellStatus::RIGHT_TRIANGLE_REMOVED,
) {
return (None, None);
}
let cell_width = self.unit_cell_width();
let cell_height = self.unit_cell_height();
let _0_5: N = na::convert(0.5);
let z0 = -_0_5 + cell_height * na::convert(i as f64);
let z1 = z0 + cell_height;
let x0 = -_0_5 + cell_width * na::convert(j as f64);
let x1 = x0 + cell_width;
let y00 = self.heights[(i + 0, j + 0)];
let y10 = self.heights[(i + 1, j + 0)];
let y01 = self.heights[(i + 0, j + 1)];
let y11 = self.heights[(i + 1, j + 1)];
let mut p00 = Point3::new(x0, y00, z0);
let mut p10 = Point3::new(x0, y10, z1);
let mut p01 = Point3::new(x1, y01, z0);
let mut p11 = Point3::new(x1, y11, z1);
p00.coords.component_mul_assign(&self.scale);
p10.coords.component_mul_assign(&self.scale);
p01.coords.component_mul_assign(&self.scale);
p11.coords.component_mul_assign(&self.scale);
if status.contains(HeightFieldCellStatus::ZIGZAG_SUBDIVISION) {
let tri1 = if status.contains(HeightFieldCellStatus::LEFT_TRIANGLE_REMOVED) {
None
} else {
Some(Triangle::new(p00, p10, p11))
};
let tri2 = if status.contains(HeightFieldCellStatus::RIGHT_TRIANGLE_REMOVED) {
None
} else {
Some(Triangle::new(p00, p11, p01))
};
(tri1, tri2)
} else {
let tri1 = if status.contains(HeightFieldCellStatus::LEFT_TRIANGLE_REMOVED) {
None
} else {
Some(Triangle::new(p00, p10, p01))
};
let tri2 = if status.contains(HeightFieldCellStatus::RIGHT_TRIANGLE_REMOVED) {
None
} else {
Some(Triangle::new(p10, p11, p01))
};
(tri1, tri2)
}
}
pub fn cell_status(&self, i: usize, j: usize) -> HeightFieldCellStatus {
self.status[(i, j)]
}
pub fn set_cell_status(&mut self, i: usize, j: usize, status: HeightFieldCellStatus) {
self.status[(i, j)] = status
}
pub fn cells_statuses(&self) -> &DMatrix<HeightFieldCellStatus> {
&self.status
}
pub fn cells_statuses_mut(&mut self) -> &mut DMatrix<HeightFieldCellStatus> {
&mut self.status
}
pub fn heights(&self) -> &DMatrix<N> {
&self.heights
}
pub fn scale(&self) -> &Vector<N> {
&self.scale
}
pub fn cell_width(&self) -> N {
self.unit_cell_width() * self.scale.x
}
pub fn cell_height(&self) -> N {
self.unit_cell_height() * self.scale.z
}
pub fn unit_cell_width(&self) -> N {
N::one() / na::convert(self.heights.ncols() as f64 - 1.0)
}
pub fn unit_cell_height(&self) -> N {
N::one() / na::convert(self.heights.nrows() as f64 - 1.0)
}
pub fn aabb(&self) -> &AABB<N> {
&self.aabb
}
pub fn convert_triangle_feature_id(
&self,
i: usize,
j: usize,
left: bool,
fid: FeatureId,
) -> FeatureId {
match fid {
FeatureId::Vertex(ivertex) => {
let nrows = self.heights.nrows();
let ij = i + j * nrows;
if self.status[(i, j)].contains(HeightFieldCellStatus::ZIGZAG_SUBDIVISION) {
if left {
FeatureId::Vertex([ij, ij + 1, ij + 1 + nrows][ivertex])
} else {
FeatureId::Vertex([ij, ij + 1 + nrows, ij + nrows][ivertex])
}
} else {
if left {
FeatureId::Vertex([ij, ij + 1, ij + nrows][ivertex])
} else {
FeatureId::Vertex([ij + 1, ij + 1 + nrows, ij + nrows][ivertex])
}
}
}
FeatureId::Edge(iedge) => {
let (nrows, ncols) = self.heights.shape();
let vshift = 0;
let hshift = (nrows - 1) * ncols;
let dshift = hshift + nrows * (ncols - 1);
let idiag = dshift + i + j * (nrows - 1);
let itop = hshift + i + j * nrows;
let ibottom = itop + 1;
let ileft = vshift + i + j * (nrows - 1);
let iright = ileft + nrows - 1;
if self.status[(i, j)].contains(HeightFieldCellStatus::ZIGZAG_SUBDIVISION) {
if left {
FeatureId::Edge([ileft, ibottom, idiag][iedge])
} else {
FeatureId::Edge([idiag, iright, itop][iedge])
}
} else {
if left {
FeatureId::Edge([ileft, idiag, itop][iedge])
} else {
FeatureId::Edge([ibottom, iright, idiag][iedge])
}
}
}
FeatureId::Face(iface) => {
if iface == 0 {
FeatureId::Face(self.face_id(i, j, left, true))
} else {
FeatureId::Face(self.face_id(i, j, left, false))
}
}
FeatureId::Unknown => FeatureId::Unknown,
}
}
pub fn map_elements_in_local_aabb(
&self,
aabb: &AABB<N>,
f: &mut impl FnMut(usize, &Triangle<N>, &dyn ContactPreprocessor<N>),
) {
let _0_5: N = na::convert(0.5);
let ncells_x = self.ncols();
let ncells_z = self.nrows();
let ref_mins = aabb.mins().coords.component_div(&self.scale);
let ref_maxs = aabb.maxs().coords.component_div(&self.scale);
let cell_width = self.unit_cell_width();
let cell_height = self.unit_cell_height();
if ref_maxs.x <= -_0_5 || ref_maxs.z <= -_0_5 || ref_mins.x >= _0_5 || ref_mins.z >= _0_5 {
return;
}
let min_x = self.quantize_floor(ref_mins.x, cell_width, ncells_x);
let min_z = self.quantize_floor(ref_mins.z, cell_height, ncells_z);
let max_x = self.quantize_ceil(ref_maxs.x, cell_width, ncells_x);
let max_z = self.quantize_ceil(ref_maxs.z, cell_height, ncells_z);
for j in min_x..max_x {
for i in min_z..max_z {
let status = self.status[(i, j)];
if status.contains(HeightFieldCellStatus::CELL_REMOVED) {
continue;
}
let z0 = -_0_5 + cell_height * na::convert(i as f64);
let z1 = z0 + cell_height;
let x0 = -_0_5 + cell_width * na::convert(j as f64);
let x1 = x0 + cell_width;
let y00 = self.heights[(i + 0, j + 0)];
let y10 = self.heights[(i + 1, j + 0)];
let y01 = self.heights[(i + 0, j + 1)];
let y11 = self.heights[(i + 1, j + 1)];
if (y00 > ref_maxs.y && y10 > ref_maxs.y && y01 > ref_maxs.y && y11 > ref_maxs.y)
|| (y00 < ref_mins.y
&& y10 < ref_mins.y
&& y01 < ref_mins.y
&& y11 < ref_mins.y)
{
continue;
}
let mut p00 = Point3::new(x0, y00, z0);
let mut p10 = Point3::new(x0, y10, z1);
let mut p01 = Point3::new(x1, y01, z0);
let mut p11 = Point3::new(x1, y11, z1);
p00.coords.component_mul_assign(&self.scale);
p10.coords.component_mul_assign(&self.scale);
p01.coords.component_mul_assign(&self.scale);
p11.coords.component_mul_assign(&self.scale);
if !status.contains(HeightFieldCellStatus::LEFT_TRIANGLE_REMOVED) {
let tri1 = if status.contains(HeightFieldCellStatus::ZIGZAG_SUBDIVISION) {
Triangle::new(p00, p10, p11)
} else {
Triangle::new(p00, p10, p01)
};
let tid = self.triangle_id(i, j, true);
let proc1 = HeightFieldTriangleContactPreprocessor::new(self, tid);
f(tid, &tri1, &proc1);
}
if !status.contains(HeightFieldCellStatus::RIGHT_TRIANGLE_REMOVED) {
let tri2 = if status.contains(HeightFieldCellStatus::ZIGZAG_SUBDIVISION) {
Triangle::new(p00, p11, p01)
} else {
Triangle::new(p10, p11, p01)
};
let tid = self.triangle_id(i, j, false);
let proc2 = HeightFieldTriangleContactPreprocessor::new(self, tid);
f(tid, &tri2, &proc2);
}
}
}
}
}
#[allow(dead_code)]
pub struct HeightFieldTriangleContactPreprocessor<'a, N: RealField> {
heightfield: &'a HeightField<N>,
triangle: usize,
}
impl<'a, N: RealField> HeightFieldTriangleContactPreprocessor<'a, N> {
pub fn new(heightfield: &'a HeightField<N>, triangle: usize) -> Self {
HeightFieldTriangleContactPreprocessor {
heightfield,
triangle,
}
}
}
impl<'a, N: RealField> ContactPreprocessor<N> for HeightFieldTriangleContactPreprocessor<'a, N> {
fn process_contact(
&self,
_c: &mut Contact<N>,
_kinematic: &mut ContactKinematic<N>,
_is_first: bool,
) -> bool {
true
}
}
struct HeightfieldTriangles<'a, N: RealField> {
heightfield: &'a HeightField<N>,
curr: (usize, usize),
tris: (Option<Triangle<N>>, Option<Triangle<N>>),
}
impl<'a, N: RealField> Iterator for HeightfieldTriangles<'a, N> {
type Item = Triangle<N>;
fn next(&mut self) -> Option<Triangle<N>> {
loop {
if let Some(tri1) = self.tris.0.take() {
return Some(tri1);
} else if let Some(tri2) = self.tris.1.take() {
return Some(tri2);
} else {
self.curr.0 += 1;
if self.curr.0 >= self.heightfield.nrows() {
if self.curr.1 >= self.heightfield.ncols() - 1 {
return None;
}
self.curr.0 = 0;
self.curr.1 += 1;
}
self.tris = self.heightfield.triangles_at(self.curr.0, self.curr.1);
}
}
}
}