rstar/primitives/
geom_with_data.rsuse crate::envelope::Envelope;
use crate::object::PointDistance;
use crate::{object::RTreeObject, point::Point};
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Default)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub struct GeomWithData<R: RTreeObject, T> {
geom: R,
pub data: T,
}
impl<R: RTreeObject, T> RTreeObject for GeomWithData<R, T> {
type Envelope = R::Envelope;
fn envelope(&self) -> Self::Envelope {
self.geom.envelope()
}
}
impl<R: PointDistance, T> PointDistance for GeomWithData<R, T> {
fn distance_2(
&self,
point: &<Self::Envelope as Envelope>::Point,
) -> <<Self::Envelope as Envelope>::Point as Point>::Scalar {
self.geom.distance_2(point)
}
fn contains_point(&self, p: &<Self::Envelope as Envelope>::Point) -> bool {
self.geom.contains_point(p)
}
fn distance_2_if_less_or_equal(
&self,
point: &<Self::Envelope as Envelope>::Point,
max_distance_2: <<Self::Envelope as Envelope>::Point as Point>::Scalar,
) -> Option<<<Self::Envelope as Envelope>::Point as Point>::Scalar> {
self.geom.distance_2_if_less_or_equal(point, max_distance_2)
}
}
impl<R: RTreeObject, T> GeomWithData<R, T> {
pub fn new(geom: R, data: T) -> Self {
Self { geom, data }
}
pub fn geom(&self) -> &R {
&self.geom
}
}
#[cfg(test)]
mod test {
use super::GeomWithData;
use crate::object::PointDistance;
use approx::*;
use crate::{primitives::Line, RTree};
#[test]
fn container_in_rtree() {
let line_1 = GeomWithData::new(Line::new([0.0, 0.0], [1.0, 1.0]), ());
let line_2 = GeomWithData::new(Line::new([0.0, 0.0], [-1.0, 1.0]), ());
let tree = RTree::bulk_load(vec![line_1, line_2]);
assert!(tree.contains(&line_1));
}
#[test]
fn container_edge_distance() {
let edge = GeomWithData::new(Line::new([0.5, 0.5], [0.5, 2.0]), 1usize);
assert_abs_diff_eq!(edge.distance_2(&[0.5, 0.5]), 0.0);
assert_abs_diff_eq!(edge.distance_2(&[0.0, 0.5]), 0.5 * 0.5);
assert_abs_diff_eq!(edge.distance_2(&[0.5, 1.0]), 0.0);
assert_abs_diff_eq!(edge.distance_2(&[0.0, 0.0]), 0.5);
assert_abs_diff_eq!(edge.distance_2(&[0.0, 1.0]), 0.5 * 0.5);
assert_abs_diff_eq!(edge.distance_2(&[1.0, 1.0]), 0.5 * 0.5);
assert_abs_diff_eq!(edge.distance_2(&[1.0, 3.0]), 0.5 * 0.5 + 1.0);
}
#[test]
fn container_length_2() {
let line = GeomWithData::new(Line::new([1, -1], [5, 5]), 1usize);
assert_eq!(line.geom().length_2(), 16 + 36);
}
#[test]
fn container_nearest_neighbour() {
let mut lines = RTree::new();
lines.insert(GeomWithData::new(
Line::new([0.0, 0.0], [1.0, 1.0]),
"Line A",
));
lines.insert(GeomWithData::new(
Line::new([0.0, 0.0], [-1.0, 1.0]),
"Line B",
));
let my_location = [0.0, 0.0];
let place = lines.nearest_neighbor(&my_location).unwrap();
assert_eq!(place.data, "Line A");
}
}