Skip to content

Commit

Permalink
Merge #290
Browse files Browse the repository at this point in the history
290: Implement spade::SpatialObject trait for LineString r=frewsxcv a=chrisbrue

A continuation of the spade r-tree integration work in progress.

Continued pattern of duplicating necessary algorithm code in algorithms.rs for now until a consensus can be reached/solution can be provided on how to deal with the circular dependency created now that the rust-geo crates are split.

Co-authored-by: Chris Brue <[email protected]>
  • Loading branch information
bors[bot] and chrisbrue committed Jul 4, 2018
2 parents aa4e66a + e0bcc1c commit 88012e2
Show file tree
Hide file tree
Showing 2 changed files with 166 additions and 1 deletion.
129 changes: 128 additions & 1 deletion geo-types/src/algorithms.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,56 @@
// wouldn't have this duplication

use num_traits::{Float, ToPrimitive};
use {CoordinateType, Line, Point};
use {Coordinate, CoordinateType, Line, LineString, Point};

pub static COORD_PRECISION: f32 = 1e-1; // 0.1m

pub trait Contains<Rhs = Self> {
fn contains(&self, rhs: &Rhs) -> bool;
}

impl<T> Contains<Point<T>> for Point<T>
where
T: Float + ToPrimitive,
{
fn contains(&self, p: &Point<T>) -> bool {
self.euclidean_distance(p).to_f32().unwrap() < COORD_PRECISION
}
}

impl<T> Contains<Point<T>> for LineString<T>
where
T: Float,
{
fn contains(&self, p: &Point<T>) -> bool {
// LineString without points
if self.0.is_empty() {
return false;
}
// LineString with one point equal p
if self.0.len() == 1 {
return Point(self.0[0]).contains(p);
}
// check if point is a vertex
if self.0.contains(&p.0) {
return true;
}
for line in self.lines() {
if ((line.start.y == line.end.y)
&& (line.start.y == p.y())
&& (p.x() > line.start.x.min(line.end.x))
&& (p.x() < line.start.x.max(line.end.x)))
|| ((line.start.x == line.end.x)
&& (line.start.x == p.x())
&& (p.y() > line.start.y.min(line.end.y))
&& (p.y() < line.start.y.max(line.end.y)))
{
return true;
}
}
false
}
}

pub trait EuclideanDistance<T, Rhs = Self> {
fn euclidean_distance(&self, rhs: &Rhs) -> T;
Expand Down Expand Up @@ -39,6 +88,33 @@ where
}
}

impl<T> EuclideanDistance<T, LineString<T>> for Point<T>
where
T: Float,
{
/// Minimum distance from a Point to a LineString
fn euclidean_distance(&self, linestring: &LineString<T>) -> T {
// No need to continue if the point is on the LineString, or it's empty
if linestring.contains(self) || linestring.0.is_empty() {
return T::zero();
}
linestring
.lines()
.map(|line| line_segment_distance(*self, line.start_point(), line.end_point()))
.fold(T::max_value(), |accum, val| accum.min(val))
}
}

impl<T> EuclideanDistance<T, Point<T>> for LineString<T>
where
T: Float,
{
/// Minimum distance from a LineString to a Point
fn euclidean_distance(&self, point: &Point<T>) -> T {
point.euclidean_distance(self)
}
}

impl<T> EuclideanDistance<T, Point<T>> for Line<T>
where
T: Float,
Expand All @@ -55,6 +131,43 @@ pub trait BoundingBox<T: CoordinateType> {
fn bbox(&self) -> Self::Output;
}

fn get_min_max<T>(p: T, min: T, max: T) -> (T, T)
where
T: CoordinateType,
{
if p > max {
(min, p)
} else if p < min {
(p, max)
} else {
(min, max)
}
}

fn get_bbox<I, T>(collection: I) -> Option<Bbox<T>>
where
T: CoordinateType,
I: IntoIterator<Item = Coordinate<T>>,
{
let mut iter = collection.into_iter();
if let Some(pnt) = iter.next() {
let mut xrange = (pnt.x, pnt.x);
let mut yrange = (pnt.y, pnt.y);
for pnt in iter {
let (px, py) = pnt.x_y();
xrange = get_min_max(px, xrange.0, xrange.1);
yrange = get_min_max(py, yrange.0, yrange.1);
}
return Some(Bbox {
xmin: xrange.0,
xmax: xrange.1,
ymin: yrange.0,
ymax: yrange.1,
});
}
None
}

impl<T> BoundingBox<T> for Line<T>
where
T: CoordinateType,
Expand All @@ -75,6 +188,20 @@ where
}
}

impl<T> BoundingBox<T> for LineString<T>
where
T: CoordinateType,
{
type Output = Option<Bbox<T>>;

///
/// Return the BoundingBox for a LineString
///
fn bbox(&self) -> Self::Output {
get_bbox(self.0.iter().cloned())
}
}

#[derive(PartialEq, Clone, Copy, Debug)]
pub struct Bbox<T>
where
Expand Down
38 changes: 38 additions & 0 deletions geo-types/src/line_string.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
use std::iter::FromIterator;
use {Coordinate, CoordinateType, Line, Point, Triangle};

#[cfg(feature = "spade")]
use algorithms::{BoundingBox, EuclideanDistance};

/// An ordered collection of two or more [`Coordinate`s](struct.Coordinate.html), representing a
/// path between locations.
///
Expand Down Expand Up @@ -162,3 +165,38 @@ impl<T: CoordinateType> IntoIterator for LineString<T> {
self.0.into_iter()
}
}

#[cfg(feature = "spade")]
impl<T> ::spade::SpatialObject for LineString<T>
where
T: ::num_traits::Float + ::spade::SpadeNum + ::std::fmt::Debug,
{
type Point = Point<T>;

fn mbr(&self) -> ::spade::BoundingRect<Self::Point> {
let bbox = self.bbox();
match bbox {
None => {
::spade::BoundingRect::from_corners(
&Point::new(T::min_value(), T::min_value()),
&Point::new(T::max_value(), T::max_value()),
)
},
Some(b) => {
::spade::BoundingRect::from_corners(
&Point::new(b.xmin, b.ymin),
&Point::new(b.xmax, b.ymax),
)
},
}
}

fn distance2(&self, point: &Self::Point) -> <Self::Point as ::spade::PointN>::Scalar {
let d = self.euclidean_distance(point);
if d == T::zero() {
d
} else {
d.powi(2)
}
}
}

0 comments on commit 88012e2

Please sign in to comment.