Skip to content

Commit

Permalink
working rtree intersector impl
Browse files Browse the repository at this point in the history
  • Loading branch information
michaelkirk committed May 10, 2022
1 parent 13e16c1 commit 2b45341
Show file tree
Hide file tree
Showing 2 changed files with 72 additions and 22 deletions.
4 changes: 4 additions & 0 deletions geo/CHANGES.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,10 @@

## Unreleased

* POSSIBLY BREAKING: `GeoFloat` types must now implement `num_traits::Signed` and `num_traits::Bounded`. This shouldn't
affect you if you are using a standard `Geometry<f64>` or `Geometry<f32>` or `geo::GeoFloat` generically.
* Speed up `Relate` and `Contains` traits for large LineStrings and Polygons by using an RTree to more efficiently
inspect edges in our topology graph.
* Speed up intersection checks by using a preliminary bbox check
* Remove unneeded reference for `*MapCoords*` closure parameter.
* <https://github.com/georust/geo/pull/810>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,41 +5,71 @@ use crate::GeoFloat;
use std::cell::RefCell;
use std::rc::Rc;

use rstar::RTree;

pub(crate) struct RstarEdgeSetIntersector;

impl RstarEdgeSetIntersector {
pub fn new() -> Self {
RstarEdgeSetIntersector
}
}

fn compute_intersects<F: GeoFloat>(
&mut self,
edge0: &Rc<RefCell<Edge<F>>>,
edge1: &Rc<RefCell<Edge<F>>>,
segment_intersector: &mut SegmentIntersector<F>,
) {
let edge0_coords_len = edge0.borrow().coords().len() - 1;
let edge1_coords_len = edge1.borrow().coords().len() - 1;
for i0 in 0..edge0_coords_len {
for i1 in 0..edge1_coords_len {
segment_intersector.add_intersections(edge0, i0, edge1, i1);
}
struct Segment<'a, F: GeoFloat + rstar::RTreeNum> {
i: usize,
edge: &'a RefCell<Edge<F>>,
envelope: rstar::AABB<crate::Coordinate<F>>,
}

impl<'a, F> Segment<'a, F>
where
F: GeoFloat + rstar::RTreeNum,
{
fn new(i: usize, edge: &'a RefCell<Edge<F>>) -> Self {
use crate::rstar::RTreeObject;
let p1 = edge.borrow().coords()[i];
let p2 = edge.borrow().coords()[i + 1];
Self {
i,
edge,
envelope: rstar::AABB::from_corners(p1, p2),
}
}
}

impl<F: GeoFloat> EdgeSetIntersector<F> for RstarEdgeSetIntersector {
impl<'a, F> rstar::RTreeObject for Segment<'a, F>
where
F: GeoFloat + rstar::RTreeNum,
{
type Envelope = rstar::AABB<crate::Coordinate<F>>;

fn envelope(&self) -> Self::Envelope {
self.envelope
}
}

impl<F> EdgeSetIntersector<F> for RstarEdgeSetIntersector
where
F: GeoFloat + rstar::RTreeNum,
{
fn compute_intersections_within_set(
&mut self,
edges: &[Rc<RefCell<Edge<F>>>],
check_for_self_intersecting_edges: bool,
segment_intersector: &mut SegmentIntersector<F>,
) {
for edge0 in edges.iter() {
for edge1 in edges.iter() {
if check_for_self_intersecting_edges || edge0.as_ptr() != edge1.as_ptr() {
self.compute_intersects(edge0, edge1, segment_intersector);
}
let segments: Vec<Segment<F>> = edges
.iter()
.flat_map(|edge| {
let start_of_final_segment: usize = RefCell::borrow(edge).coords().len() - 1;
(0..start_of_final_segment).map(|segment_i| Segment::new(segment_i, edge))
})
.collect();
let tree = RTree::bulk_load(segments);

for (edge0, edge1) in tree.intersection_candidates_with_other_tree(&tree) {
if check_for_self_intersecting_edges || edge0.edge.as_ptr() != edge1.edge.as_ptr() {
segment_intersector.add_intersections(edge0.edge, edge0.i, edge1.edge, edge1.i);
}
}
}
Expand All @@ -50,10 +80,26 @@ impl<F: GeoFloat> EdgeSetIntersector<F> for RstarEdgeSetIntersector {
edges1: &[Rc<RefCell<Edge<F>>>],
segment_intersector: &mut SegmentIntersector<F>,
) {
for edge0 in edges0 {
for edge1 in edges1 {
self.compute_intersects(edge0, edge1, segment_intersector);
}
let segments0: Vec<Segment<F>> = edges0
.iter()
.flat_map(|edge| {
let start_of_final_segment: usize = RefCell::borrow(edge).coords().len() - 1;
(0..start_of_final_segment).map(|segment_i| Segment::new(segment_i, edge))
})
.collect();
let tree_0 = RTree::bulk_load(segments0);

let segments1: Vec<Segment<F>> = edges1
.iter()
.flat_map(|edge| {
let start_of_final_segment: usize = RefCell::borrow(edge).coords().len() - 1;
(0..start_of_final_segment).map(|segment_i| Segment::new(segment_i, edge))
})
.collect();
let tree_1 = RTree::bulk_load(segments1);

for (edge0, edge1) in tree_0.intersection_candidates_with_other_tree(&tree_1) {
segment_intersector.add_intersections(edge0.edge, edge0.i, edge1.edge, edge1.i);
}
}
}

0 comments on commit 2b45341

Please sign in to comment.