From a40b6c009c5112b9854150812cd8a8fdd2409a6c Mon Sep 17 00:00:00 2001 From: Andre Esteve Date: Sun, 31 Oct 2021 01:02:09 -0700 Subject: [PATCH] Fixes invalid hull on collinear points Fixes #24 --- Cargo.toml | 2 +- src/lib.rs | 8 ++++-- tests/fixtures/issue24.json | 1 + tests/tests.rs | 53 ++++++++++++++++++++++++------------- 4 files changed, 42 insertions(+), 22 deletions(-) create mode 100644 tests/fixtures/issue24.json diff --git a/Cargo.toml b/Cargo.toml index da6f003..30aee62 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "delaunator" -version = "1.0.0" +version = "1.0.1" edition = "2018" description = "A very fast 2D Delaunay triangulation library." documentation = "https://docs.rs/delaunator" diff --git a/src/lib.rs b/src/lib.rs index b3fede2..208ff34 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -62,7 +62,11 @@ impl Point { dx * dx + dy * dy } + /// Returns a **negative** value if ```self```, ```q``` and ```r``` occur in counterclockwise order (```r``` is to the left of the directed line ```self``` --> ```q```) + /// Returns a **positive** value if they occur in clockwise order(```r``` is to the right of the directed line ```self``` --> ```q```) + /// Returns zero is they are collinear fn orient(&self, q: &Self, r: &Self) -> f64 { + // robust-rs orients Y-axis upwards, our convention is Y downwards. This means that the interpretation of the result must be flipped orient2d(self.into(), q.into(), r.into()) } @@ -526,7 +530,7 @@ pub fn triangulate(points: &[Point]) -> Triangulation { let mut n = hull.next[e]; loop { let q = hull.next[n]; - if p.orient(&points[n], &points[q]) < 0. { + if p.orient(&points[n], &points[q]) <= 0. { break; } let t = triangulation.add_triangle(n, i, q, hull.tri[i], EMPTY, hull.tri[n]); @@ -539,7 +543,7 @@ pub fn triangulate(points: &[Point]) -> Triangulation { if walk_back { loop { let q = hull.prev[e]; - if p.orient(&points[q], &points[e]) < 0. { + if p.orient(&points[q], &points[e]) <= 0. { break; } let t = triangulation.add_triangle(q, i, e, EMPTY, hull.tri[e], hull.tri[q]); diff --git a/tests/fixtures/issue24.json b/tests/fixtures/issue24.json new file mode 100644 index 0000000..1faa21c --- /dev/null +++ b/tests/fixtures/issue24.json @@ -0,0 +1 @@ +[[0,0],[1,-1],[1,0],[1,1]] \ No newline at end of file diff --git a/tests/tests.rs b/tests/tests.rs index 3d4bb20..684c290 100644 --- a/tests/tests.rs +++ b/tests/tests.rs @@ -149,6 +149,15 @@ fn unordered_collinear_points_input() { ); } +#[test] +fn hull_collinear_issue24() { + let points = load_fixture("tests/fixtures/issue24.json"); + validate(&points); + + let t = triangulate(&points); + assert_eq!(t.hull, &[0, 3, 2, 1], "Invalid hull"); +} + fn scale_points(points: &[Point], scale: f64) -> Vec { let scaled: Vec = points .iter() @@ -170,8 +179,23 @@ fn orient(p: &Point, q: &Point, r: &Point) -> f64 { robust::orient2d(p.into(), q.into(), r.into()) } -fn convex(r: &Point, q: &Point, p: &Point) -> bool { - orient(p, r, q) <= 0. || orient(r, q, p) <= 0. || orient(q, p, r) < 0. +/// make sure hull is convex and counter-clockwise (p1 is to the right of the directed line p0 --> p2) +/// in case of collinear points, make sure they are ordered (p1 between p0 and p2) +// p-1 p3 +// \ ^ +// > p0 ---------------> p2 / +// p1 +fn assert_convex(p0: &Point, p1: &Point, p2: &Point) { + let l = orient(p0, p2, p1); + assert!(l >= 0., "p1 ({:?}) is to the left of the directed line p0 ({:?}) --> p2 ({:?}). Hull is not convex.", p1, p0, p2); + + if l == 0. { + // if p0, p1 and p2 are collinear, they must be ordered + // that means that p1 - p0 = c * (p2 - p0), where c is (0..1) but not inclusive (linear combination) + let c = ((p1.x - p0.x) / (p2.x - p0.x)).max((p1.y - p0.y) / (p2.y - p0.y)); + assert!(c > 0., "incorrect ordering, found p1, p0, p2, expected p0 ({:?}), p1 ({:?}), p2 ({:?}). Invalid hull.", p0, p1, p2); + assert!(c < 1., "incorrect ordering, found p0, p2, p1, expected p0 ({:?}), p1 ({:?}), p2 ({:?}). Invalid hull.", p0, p1, p2); + } } fn validate(points: &[Point]) { @@ -191,24 +215,15 @@ fn validate(points: &[Point]) { // validate triangulation let hull_area = { let mut hull_areas = Vec::new(); - let mut i = 0; - let mut j = hull.len() - 1; - while i < hull.len() { - let p0 = &points[hull[j]]; - let p = &points[hull[i]]; - - if !convex( - p0, - &points[hull[(j + 1) % hull.len()]], - &points[hull[(j + 3) % hull.len()]], - ) { - panic!("Hull is not convex at {}", j); - } - - hull_areas.push((p.x - p0.x) * (p.y + p0.y)); - j = i; - i += 1; + + for i in 0..hull.len() { + let p0 = &points[hull[i]]; + let p1 = &points[hull[(i + 1) % hull.len()]]; + let p2 = &points[hull[(i + 2) % hull.len()]]; + assert_convex(p0, p1, p2); + hull_areas.push((p1.x - p0.x) * (p1.y + p0.y)); } + sum(&hull_areas) }; let triangles_area = {