Skip to content

Commit

Permalink
Merge branch 'main' into feat/frechet-in-various-spaces
Browse files Browse the repository at this point in the history
  • Loading branch information
grevgeny authored Dec 3, 2024
2 parents 314c0e2 + 41e5ed0 commit f5dce87
Show file tree
Hide file tree
Showing 6 changed files with 79 additions and 328 deletions.
8 changes: 8 additions & 0 deletions geo/CHANGES.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,14 @@

## Unreleased

- Loosen bounds on `RemoveRepeatedPoints` trait (`num_traits::FromPrimitive` isn't required)
- <https://github.com/georust/geo/pull/1278>

## 0.29.3 - 2024.12.03

- Fix crash in `BoolOps` by updating `i_overlay` to 1.9.0.
- <https://github.com/georust/geo/pull/1275>

## 0.29.2 - 2024.11.15

- Pin `i_overlay` to < 1.8.0 to work around [recursion bug](https://github.com/georust/geo/issues/1270).
Expand Down
4 changes: 2 additions & 2 deletions geo/Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
[package]
name = "geo"
description = "Geospatial primitives and algorithms"
version = "0.29.2"
version = "0.29.3"
license = "MIT OR Apache-2.0"
repository = "https://github.com/georust/geo"
documentation = "https://docs.rs/geo/"
Expand Down Expand Up @@ -31,7 +31,7 @@ proj = { version = "0.27.0", optional = true }
robust = "1.1.0"
rstar = "0.12.0"
serde = { version = "1.0", optional = true, features = ["derive"] }
i_overlay = { version = "1.7.2, < 1.8.0", default-features = false }
i_overlay = { version = "1.9.0, < 1.10.0", default-features = false }

[dev-dependencies]
approx = ">= 0.4.0, < 0.6.0"
Expand Down
277 changes: 29 additions & 248 deletions geo/src/algorithm/bool_ops/i_overlay_integration.rs
Original file line number Diff line number Diff line change
@@ -1,290 +1,71 @@
use crate::geometry::Coord;
use crate::GeoNum;
use i_overlay::core::fill_rule::FillRule;
use i_overlay::core::overlay::ShapeType;
use i_overlay::core::overlay_rule::OverlayRule;
use i_overlay::string::clip::ClipRule;
use i_overlay::i_float::float::compatible::FloatPointCompatible;
use i_overlay::i_float::float::number::FloatNumber;

pub trait BoolOpsCoord<T>: Copy {
fn new(x: T, y: T) -> Self;
fn x(&self) -> T;
fn y(&self) -> T;
}

/// A geometry coordinate number suitable for performing geometric boolean operations.
pub trait BoolOpsNum: GeoNum {
type CoordType: BoolOpsCoord<Self>;
type OverlayType: BoolOpsOverlay<CoordType = Self::CoordType>;
type StringOverlayType: BoolOpsStringOverlay<CoordType = Self::CoordType>;

fn to_bops_coord(geo_coord: Coord<Self>) -> Self::CoordType {
Self::CoordType::new(geo_coord.x, geo_coord.y)
}

fn to_geo_coord(bops_coord: Self::CoordType) -> Coord<Self> {
Coord {
x: bops_coord.x(),
y: bops_coord.y(),
}
}
}

pub trait BoolOpsOverlay {
type CoordType;
type OverlayGraph: BoolOpsOverlayGraph<CoordType = Self::CoordType>;
fn new() -> Self;
fn add_path(&mut self, path: Vec<Self::CoordType>, shape_type: ShapeType);
fn into_graph(self, fill_rule: FillRule) -> Self::OverlayGraph;
}

pub(super) trait BoolOpsOverlayGraph {
type CoordType;
fn extract_shapes(&self, overlay_rule: OverlayRule) -> Vec<Vec<Vec<Self::CoordType>>>;
}

pub trait BoolOpsStringOverlay {
type CoordType;
type StringGraph: BoolOpsStringGraph<CoordType = Self::CoordType>;
fn new() -> Self;
fn add_shape_path(&mut self, path: Vec<Self::CoordType>);
fn add_string_line(&mut self, path: [Self::CoordType; 2]);
fn into_graph(self, fill_rule: FillRule) -> Self::StringGraph;
}

pub(super) trait BoolOpsStringGraph {
type CoordType;
fn clip_string_lines(&self, clip_rule: ClipRule) -> Vec<Vec<Self::CoordType>>;
}

mod f64 {
use super::{ClipRule, FillRule, OverlayRule, ShapeType};
use i_overlay::f64::{
graph::F64OverlayGraph,
overlay::F64Overlay,
string::{F64StringGraph, F64StringOverlay},
};
use i_overlay::i_float::f64_point::F64Point;

impl super::BoolOpsNum for f64 {
type CoordType = F64Point;
type OverlayType = F64Overlay;
type StringOverlayType = F64StringOverlay;
}

impl super::BoolOpsCoord<f64> for F64Point {
#[inline]
fn new(x: f64, y: f64) -> Self {
Self::new(x, y)
}

#[inline]
fn x(&self) -> f64 {
self.x
}

#[inline]
fn y(&self) -> f64 {
self.y
}
}

impl super::BoolOpsOverlay for F64Overlay {
type CoordType = F64Point;
type OverlayGraph = F64OverlayGraph;

#[inline]
fn new() -> Self {
Self::new()
}

#[inline]
fn add_path(&mut self, path: Vec<F64Point>, shape_type: ShapeType) {
self.add_path(path, shape_type)
}

#[inline]
fn into_graph(self, fill_rule: FillRule) -> Self::OverlayGraph {
self.into_graph(fill_rule)
}
}

impl super::BoolOpsOverlayGraph for F64OverlayGraph {
type CoordType = F64Point;

#[inline]
fn extract_shapes(&self, overlay_rule: OverlayRule) -> Vec<Vec<Vec<F64Point>>> {
self.extract_shapes(overlay_rule)
}
}

impl super::BoolOpsStringOverlay for F64StringOverlay {
type CoordType = F64Point;
type StringGraph = F64StringGraph;

#[inline]
fn new() -> Self {
Self::new()
}
/// A geometry coordinate scalar suitable for performing geometric boolean operations.
pub trait BoolOpsNum: GeoNum + FloatNumber {}
impl<T: GeoNum + FloatNumber> BoolOpsNum for T {}

#[inline]
fn add_shape_path(&mut self, path: Vec<Self::CoordType>) {
self.add_shape_path(path)
}

#[inline]
fn add_string_line(&mut self, path: [Self::CoordType; 2]) {
self.add_string_line(path)
}
/// New type for `Coord` that implements `FloatPointCompatible` for `BoolOpsNum` to
/// circumvent orphan rule, since Coord is defined in geo_types.
#[derive(Copy, Clone, Debug)]
pub struct BoolOpsCoord<T: BoolOpsNum>(pub(crate) Coord<T>);

#[inline]
fn into_graph(self, fill_rule: FillRule) -> Self::StringGraph {
self.into_graph(fill_rule)
}
impl<T: BoolOpsNum> FloatPointCompatible<T> for BoolOpsCoord<T> {
fn from_xy(x: T, y: T) -> Self {
Self(Coord { x, y })
}

impl super::BoolOpsStringGraph for F64StringGraph {
type CoordType = F64Point;

#[inline]
fn clip_string_lines(&self, clip_rule: ClipRule) -> Vec<Vec<Self::CoordType>> {
self.clip_string_lines(clip_rule)
}
fn x(&self) -> T {
self.0.x
}
}

mod f32 {
use i_overlay::core::fill_rule::FillRule;
use i_overlay::core::overlay::ShapeType;
use i_overlay::core::overlay_rule::OverlayRule;
use i_overlay::f32::graph::F32OverlayGraph;
use i_overlay::f32::overlay::F32Overlay;
use i_overlay::f32::string::{F32StringGraph, F32StringOverlay};
use i_overlay::i_float::f32_point::F32Point;
use i_overlay::string::clip::ClipRule;

impl super::BoolOpsNum for f32 {
type CoordType = F32Point;
type OverlayType = F32Overlay;
type StringOverlayType = F32StringOverlay;
}

impl super::BoolOpsCoord<f32> for F32Point {
#[inline]
fn new(x: f32, y: f32) -> Self {
Self::new(x, y)
}
#[inline]
fn x(&self) -> f32 {
self.x
}
#[inline]
fn y(&self) -> f32 {
self.y
}
}

impl super::BoolOpsOverlay for F32Overlay {
type CoordType = F32Point;
type OverlayGraph = F32OverlayGraph;

#[inline]
fn new() -> Self {
Self::new()
}

#[inline]
fn add_path(&mut self, path: Vec<Self::CoordType>, shape_type: ShapeType) {
self.add_path(path, shape_type)
}

#[inline]
fn into_graph(self, fill_rule: FillRule) -> Self::OverlayGraph {
self.into_graph(fill_rule)
}
}

impl super::BoolOpsOverlayGraph for F32OverlayGraph {
type CoordType = F32Point;

#[inline]
fn extract_shapes(&self, overlay_rule: OverlayRule) -> Vec<Vec<Vec<F32Point>>> {
self.extract_shapes(overlay_rule)
}
}

impl super::BoolOpsStringOverlay for F32StringOverlay {
type CoordType = F32Point;
type StringGraph = F32StringGraph;

#[inline]
fn new() -> Self {
Self::new()
}

#[inline]
fn add_shape_path(&mut self, path: Vec<Self::CoordType>) {
self.add_shape_path(path)
}

#[inline]
fn add_string_line(&mut self, path: [Self::CoordType; 2]) {
self.add_string_line(path)
}

#[inline]
fn into_graph(self, fill_rule: FillRule) -> Self::StringGraph {
self.into_graph(fill_rule)
}
}

impl super::BoolOpsStringGraph for F32StringGraph {
type CoordType = F32Point;

#[inline]
fn clip_string_lines(&self, clip_rule: ClipRule) -> Vec<Vec<Self::CoordType>> {
self.clip_string_lines(clip_rule)
}
fn y(&self) -> T {
self.0.y
}
}

pub(super) mod convert {
use super::super::OpType;
use super::{BoolOpsNum, OverlayRule};
use super::BoolOpsNum;
use crate::bool_ops::i_overlay_integration::BoolOpsCoord;
use crate::geometry::{LineString, MultiLineString, MultiPolygon, Polygon};
use i_overlay::core::overlay_rule::OverlayRule;

pub fn line_string_from_path<T: BoolOpsNum>(path: Vec<T::CoordType>) -> LineString<T> {
let coords = path.into_iter().map(T::to_geo_coord);
LineString(coords.collect())
pub fn line_string_from_path<T: BoolOpsNum>(path: Vec<BoolOpsCoord<T>>) -> LineString<T> {
let coords = path.into_iter().map(|bops_coord| bops_coord.0).collect();
LineString(coords)
}

pub fn multi_line_string_from_paths<T: BoolOpsNum>(
paths: Vec<Vec<T::CoordType>>,
paths: Vec<Vec<BoolOpsCoord<T>>>,
) -> MultiLineString<T> {
let line_strings = paths.into_iter().map(|p| line_string_from_path(p));
MultiLineString(line_strings.collect())
}

pub fn polygon_from_shape<T: BoolOpsNum>(shape: Vec<Vec<T::CoordType>>) -> Polygon<T> {
pub fn polygon_from_shape<T: BoolOpsNum>(shape: Vec<Vec<BoolOpsCoord<T>>>) -> Polygon<T> {
let mut rings = shape.into_iter().map(|p| line_string_from_path(p));
let exterior = rings.next().unwrap_or(LineString::new(vec![]));
Polygon::new(exterior, rings.collect())
}

pub fn multi_polygon_from_shapes<T: BoolOpsNum>(
shapes: Vec<Vec<Vec<T::CoordType>>>,
shapes: Vec<Vec<Vec<BoolOpsCoord<T>>>>,
) -> MultiPolygon<T> {
let polygons = shapes.into_iter().map(|s| polygon_from_shape(s));
MultiPolygon(polygons.collect())
}

pub fn ring_to_shape_path<T: BoolOpsNum>(line_string: &LineString<T>) -> Vec<T::CoordType> {
pub fn ring_to_shape_path<T: BoolOpsNum>(line_string: &LineString<T>) -> Vec<BoolOpsCoord<T>> {
if line_string.0.is_empty() {
return vec![];
}
// In geo, Polygon rings are explicitly closed LineStrings — their final coordinate is the same as their first coordinate,
// however in i_overlay, shape paths are implicitly closed, so we skip the last coordinate.
let coords = &line_string.0[..line_string.0.len() - 1];
coords.iter().copied().map(T::to_bops_coord).collect()
coords.iter().copied().map(BoolOpsCoord).collect()
}

impl From<OpType> for OverlayRule {
Expand All @@ -308,15 +89,15 @@ mod tests {
#[test]
fn two_empty_polygons() {
let p1: Polygon = wkt!(POLYGON EMPTY);
let p2 = wkt!(POLYGON EMPTY);
let p2: Polygon = wkt!(POLYGON EMPTY);
assert_eq!(&p1.union(&p2), &wkt!(MULTIPOLYGON EMPTY));
assert_eq!(&p1.intersection(&p2), &wkt!(MULTIPOLYGON EMPTY));
}

#[test]
fn one_empty_polygon() {
let p1: Polygon = wkt!(POLYGON((0. 0., 0. 1., 1. 1., 1. 0., 0. 0.)));
let p2 = wkt!(POLYGON EMPTY);
let p2: Polygon = wkt!(POLYGON EMPTY);
assert_eq!(&p1.union(&p2), &MultiPolygon(vec![p1.clone()]));
assert_eq!(&p1.intersection(&p2), &wkt!(MULTIPOLYGON EMPTY));
}
Expand Down
Loading

0 comments on commit f5dce87

Please sign in to comment.