From aea681a9a9c44c6066b3566cfa4f4dff817429bb Mon Sep 17 00:00:00 2001 From: Stefan Altmayer Date: Tue, 14 May 2024 18:25:06 +0200 Subject: [PATCH] Implements CDT bulk loading. Fixes some edge cases for regular DT bulk loading. --- CHANGELOG.md | 6 + fuzz/Cargo.toml | 5 + fuzz/fuzz_targets/bulk_load_cdt_fuzz.rs | 53 ++ fuzz/fuzz_targets/bulk_load_fuzz.rs | 24 +- fuzz/fuzz_targets/fuzz_shared.rs | 20 + images/shape_iterator_circle_vertices.svg | 2 +- src/cdt.rs | 128 ++++- src/delaunay_core/bulk_load.rs | 601 ++++++++++++++++++---- src/delaunay_core/bulk_load_fuzz_tests.rs | 82 +++ src/delaunay_core/mod.rs | 2 +- src/delaunay_core/refinement.rs | 2 +- src/delaunay_core/triangulation_ext.rs | 14 +- src/delaunay_triangulation.rs | 18 +- src/triangulation.rs | 4 +- 14 files changed, 826 insertions(+), 135 deletions(-) create mode 100644 fuzz/fuzz_targets/bulk_load_cdt_fuzz.rs create mode 100644 fuzz/fuzz_targets/fuzz_shared.rs diff --git a/CHANGELOG.md b/CHANGELOG.md index f2bc928..b9f44dc 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,12 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](http://keepachangelog.com/) and this project adheres to [Semantic Versioning](http://semver.org/). +## [2.7.0] - 2024-MM-DD + +### Added + - Implements `ConstrainedDelaunayTriangulation::bulk_load_cdt` + - Implements `ConstrainedDelaunayTriangulation::bulk_load_cdt_stable` + ## [2.6.0] - 2024-01-14 ### Added diff --git a/fuzz/Cargo.toml b/fuzz/Cargo.toml index 756ec71..272353f 100644 --- a/fuzz/Cargo.toml +++ b/fuzz/Cargo.toml @@ -31,3 +31,8 @@ path = "fuzz_targets/bulk_load_int_fuzz.rs" test = false doc = false +[[bin]] +name = "bulk_load_cdt_fuzz" +path = "fuzz_targets/bulk_load_cdt_fuzz.rs" +test = false +doc = false diff --git a/fuzz/fuzz_targets/bulk_load_cdt_fuzz.rs b/fuzz/fuzz_targets/bulk_load_cdt_fuzz.rs new file mode 100644 index 0000000..aacb7b1 --- /dev/null +++ b/fuzz/fuzz_targets/bulk_load_cdt_fuzz.rs @@ -0,0 +1,53 @@ +#![no_main] +mod fuzz_shared; +use fuzz_shared::FuzzPoint; +use libfuzzer_sys::fuzz_target; + +use spade::{ConstrainedDelaunayTriangulation, HasPosition, Triangulation}; + +fuzz_target!(|input: (Vec, Vec<[usize; 2]>)| { + let (data, mut edges) = input; + for p in &data { + if spade::validate_coordinate(p.x).is_err() || spade::validate_coordinate(p.y).is_err() { + return; + } + if p.x.abs() > 20.0 || p.y.abs() > 20.0 { + return; + } + } + + for &[from, to] in &edges { + if from >= data.len() || to >= data.len() || from == to { + return; + } + } + + let mut reference_cdt = + ConstrainedDelaunayTriangulation::::bulk_load(data.clone()).unwrap(); + + let mut last_index = 0; + for (index, [from, to]) in edges.iter().copied().enumerate() { + let from = reference_cdt + .locate_vertex(data[from].position()) + .unwrap() + .fix(); + let to = reference_cdt + .locate_vertex(data[to].position()) + .unwrap() + .fix(); + + if reference_cdt.can_add_constraint(from, to) { + reference_cdt.add_constraint(from, to); + } else { + last_index = index; + break; + } + } + + edges.truncate(last_index); + + let bulk_loaded = + ConstrainedDelaunayTriangulation::::bulk_load_cdt(data, edges).unwrap(); + + bulk_loaded.cdt_sanity_check(); +}); diff --git a/fuzz/fuzz_targets/bulk_load_fuzz.rs b/fuzz/fuzz_targets/bulk_load_fuzz.rs index 0d5dbc8..a34e94a 100644 --- a/fuzz/fuzz_targets/bulk_load_fuzz.rs +++ b/fuzz/fuzz_targets/bulk_load_fuzz.rs @@ -1,7 +1,8 @@ #![no_main] +mod fuzz_shared; +use fuzz_shared::FuzzPoint; use libfuzzer_sys::fuzz_target; - -use spade::{DelaunayTriangulation, Point2, Triangulation, TriangulationExt}; +use spade::{DelaunayTriangulation, Triangulation, TriangulationExt}; fuzz_target!(|data: Vec| { for p in &data { @@ -12,23 +13,6 @@ fuzz_target!(|data: Vec| { return; } } - let triangulation = DelaunayTriangulation::<_>::bulk_load( - data.iter() - .map(|point| Point2::new(point.x as f64, point.y as f64)) - .collect(), - ) - .unwrap(); + let triangulation = DelaunayTriangulation::<_>::bulk_load(data.clone()).unwrap(); triangulation.sanity_check(); }); - -#[derive(Clone, Copy, arbitrary::Arbitrary)] -pub struct FuzzPoint { - x: f64, - y: f64, -} - -impl core::fmt::Debug for FuzzPoint { - fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { - f.write_fmt(format_args!("Point2::new({:?}, {:?})", self.x, self.y)) - } -} diff --git a/fuzz/fuzz_targets/fuzz_shared.rs b/fuzz/fuzz_targets/fuzz_shared.rs new file mode 100644 index 0000000..92d5938 --- /dev/null +++ b/fuzz/fuzz_targets/fuzz_shared.rs @@ -0,0 +1,20 @@ +use spade::{HasPosition, Point2}; + +#[derive(Clone, Copy, arbitrary::Arbitrary)] +pub struct FuzzPoint { + pub x: f64, + pub y: f64, +} + +impl HasPosition for FuzzPoint { + type Scalar = f64; + fn position(&self) -> Point2 { + Point2::new(self.x, self.y) + } +} + +impl core::fmt::Debug for FuzzPoint { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + f.write_fmt(format_args!("Point2::new({:?}, {:?})", self.x, self.y)) + } +} diff --git a/images/shape_iterator_circle_vertices.svg b/images/shape_iterator_circle_vertices.svg index 2e044a0..aba9c8b 100644 --- a/images/shape_iterator_circle_vertices.svg +++ b/images/shape_iterator_circle_vertices.svg @@ -86,8 +86,8 @@ - + diff --git a/src/cdt.rs b/src/cdt.rs index 7980a49..b1a79a5 100644 --- a/src/cdt.rs +++ b/src/cdt.rs @@ -1,9 +1,11 @@ +use crate::delaunay_core::{bulk_load_cdt, bulk_load_stable}; use crate::{delaunay_core::Dcel, intersection_iterator::LineIntersectionIterator}; use crate::{handles::*, intersection_iterator::Intersection}; use crate::{ DelaunayTriangulation, HasPosition, HintGenerator, InsertionError, LastUsedVertexHintGenerator, Point2, Triangulation, TriangulationExt, }; + #[cfg(feature = "serde")] use serde::{Deserialize, Serialize}; @@ -235,8 +237,9 @@ where ) -> ( Dcel, Self::HintGenerator, + usize, ) { - todo!() + (self.dcel, self.hint_generator, self.num_constraints) } } @@ -270,6 +273,114 @@ where F: Default, L: HintGenerator<::Scalar>, { + /// Efficient bulk loading of a constraint delaunay triangulation, including both vertices and constraint edges. + /// + /// The edges are given as pairs of vertex indices. + /// + /// Note that the vertex order is not preserved by this function - iterating through all vertices will not result in + /// the same sequence as the input vertices. Use [ConstrainedDelaunayTriangulation::bulk_load_cdt_stable] for a + /// slower but order preserving variant. + /// + /// Input vertices may have the same position. However, only one vertex for each position will be kept. Edges + /// that go to a discarded vertex are rerouted and still inserted. + /// It is arbitrary which duplicated vertex remains. + /// + /// # Example + /// ``` + /// # fn main() -> Result<(), spade::InsertionError> { + /// use spade::{ConstrainedDelaunayTriangulation, Point2, Triangulation}; + /// let mut vertices = vec![ + /// Point2::new(0.0, 1.0), + /// Point2::new(1.0, 2.0), + /// Point2::new(3.0, -3.0), + /// Point2::new(-1.0, -2.0), + /// Point2::new(-4.0, -5.0), + /// ]; + /// let mut edges = vec![[0, 1], [1, 2], [2, 3], [3, 4]]; + /// let cdt = ConstrainedDelaunayTriangulation::<_>::bulk_load_cdt(vertices.clone(), edges)?; + /// + /// assert_eq!(cdt.num_vertices(), 5); + /// assert_eq!(cdt.num_constraints(), 4); + /// // The order will usually change + /// assert_ne!(cdt.vertices().map(|v| v.position()).collect::>(), vertices); + /// # Ok(()) + /// # } + /// ``` + /// + /// # Panics + /// Panics if any constraint edges overlap. Panics if the edges contain an invalid index (out of range). + pub fn bulk_load_cdt(vertices: Vec, edges: Vec<[usize; 2]>) -> Result { + let mut result = bulk_load_cdt(vertices, edges)?; + *result.hint_generator_mut() = L::initialize_from_triangulation(&result); + Ok(result) + } + + /// Stable bulk load variant that preserves the input vertex order + /// + /// The resulting vertex set will be equal to the input vertex set if their positions are all distinct. + /// See [ConstrainedDelaunayTriangulation::bulk_load_cdt] for additional details like panic behavior and duplicate + /// handling. + /// + /// # Example + /// ``` + /// # fn main() -> Result<(), spade::InsertionError> { + /// use spade::{ConstrainedDelaunayTriangulation, Point2, Triangulation}; + /// let mut vertices = vec![ + /// Point2::new(0.0, 1.0), + /// Point2::new(1.0, 2.0), + /// Point2::new(3.0, -3.0), + /// Point2::new(-1.0, -2.0), + /// Point2::new(-4.0, -5.0), + /// ]; + /// let mut edges = vec![[0, 1], [1, 2], [2, 3], [3, 4]]; + /// let cdt = ConstrainedDelaunayTriangulation::<_>::bulk_load_cdt_stable(vertices.clone(), edges)?; + /// + /// // The ordered will be preserved: + /// assert_eq!(cdt.vertices().map(|v| v.position()).collect::>(), vertices); + /// # Ok(()) + /// # } + /// ``` + /// + /// It is fine to include vertex positions multiple times. The resulting order will be the same as if + /// the duplicates were removed prior to insertion. However, it is unclear *which* duplicates are + /// removed - e.g. do not assume that always the first duplicated vertex remains. + /// + /// ``` + /// # fn main() -> Result<(), spade::InsertionError> { + /// use spade::{ConstrainedDelaunayTriangulation, Point2, Triangulation}; + /// let mut vertices = vec![ + /// Point2::new(0.0, 1.0), + /// Point2::new(1.0, 2.0), // Duplicate + /// Point2::new(1.0, 2.0), + /// Point2::new(3.0, -3.0), + /// Point2::new(3.0, -3.0), // Duplicate + /// Point2::new(-4.0, -5.0), + /// ]; + /// let mut edges = vec![[0, 1], [2, 3], [4, 5]]; + /// let cdt = ConstrainedDelaunayTriangulation::<_>::bulk_load_cdt_stable(vertices.clone(), edges)?; + /// + /// // The choice of deduplicated vertices is arbitrary. In this example, dedup[1] and dedup[2] could + /// // have been swapped + /// let dedup = [ + /// Point2::new(0.0, 1.0), + /// Point2::new(1.0, 2.0), + /// Point2::new(3.0, -3.0), + /// Point2::new(-4.0, -5.0), + /// ]; + /// assert_eq!(cdt.vertices().map(|v| v.position()).collect::>(), dedup); + /// # Ok(()) + /// # } + /// ``` + pub fn bulk_load_cdt_stable( + vertices: Vec, + edges: Vec<[usize; 2]>, + ) -> Result { + let mut result: Self = + bulk_load_stable(move |vertices| bulk_load_cdt(vertices, edges), vertices)?; + *result.hint_generator_mut() = L::initialize_from_triangulation(&result); + Ok(result) + } + /// Removes a vertex from the triangulation. /// /// This operation runs in O(n²), where n is the degree of the @@ -622,8 +733,13 @@ where } } - #[cfg(test)] + #[cfg(any(test, fuzzing))] pub fn cdt_sanity_check(&self) { + self.cdt_sanity_check_with_params(true); + } + + #[cfg(any(test, fuzzing))] + pub fn cdt_sanity_check_with_params(&self, check_convexity: bool) { let num_undirected_edges = self .dcel .undirected_edges() @@ -631,12 +747,18 @@ where .count(); assert_eq!(num_undirected_edges, self.num_constraints()); - self.basic_sanity_check(); + + if self.num_constraints() == 0 && check_convexity { + self.sanity_check(); + } else { + self.basic_sanity_check(check_convexity); + } } } #[cfg(test)] mod test { + use super::ConstrainedDelaunayTriangulation; use crate::test_utilities::*; use crate::{DelaunayTriangulation, InsertionError, Point2, Triangulation}; diff --git a/src/delaunay_core/bulk_load.rs b/src/delaunay_core/bulk_load.rs index 7fb181e..139ea90 100644 --- a/src/delaunay_core/bulk_load.rs +++ b/src/delaunay_core/bulk_load.rs @@ -1,10 +1,15 @@ -use crate::{HasPosition, InsertionError, Point2, Triangulation, TriangulationExt}; +use crate::{ + ConstrainedDelaunayTriangulation, HasPosition, HintGenerator, InsertionError, Point2, + Triangulation, TriangulationExt, +}; use core::cmp::{Ordering, Reverse}; +use num_traits::Zero; use super::{ dcel_operations, FixedDirectedEdgeHandle, FixedUndirectedEdgeHandle, FixedVertexHandle, }; +use alloc::vec; use alloc::vec::Vec; /// An `f64` wrapper implementing `Ord` and `Eq`. @@ -13,16 +18,19 @@ use alloc::vec::Vec; /// All input coordinates are checked with `validate_coordinate` before they are used, hence /// `Ord` and `Eq` should always be well defined. #[derive(Debug, PartialEq, PartialOrd, Clone, Copy)] -struct FloatOrd(f64); +struct FloatOrd(S); #[allow(clippy::derive_ord_xor_partial_ord)] -impl Ord for FloatOrd { +impl Ord for FloatOrd +where + S: PartialOrd, +{ fn cmp(&self, other: &Self) -> Ordering { self.partial_cmp(other).unwrap() } } -impl Eq for FloatOrd {} +impl Eq for FloatOrd where S: PartialOrd {} /// Implements a circle-sweep bulk loading algorithm for efficient initialization of Delaunay /// triangulations. @@ -65,10 +73,6 @@ impl Eq for FloatOrd {} /// /// "angle" does not refer to an actual angle in radians but rather to an approximation that doesn't /// require trigonometry for calculation. See method `pseudo_angle` for more information. -/// -/// In rare cases, step 6 is not able to insert a vertex properly. It will be skipped and inserted -/// regularly at the end (slow path). This may happen especially for very skewed triangulations -/// and might be a good point for investigation if some point sets takes surprisingly long to load. pub fn bulk_load(mut elements: Vec) -> Result where V: HasPosition, @@ -104,70 +108,235 @@ where Reverse(FloatOrd(initial_center.distance_2(e.position().to_f64()))) }); - while let Some(next) = elements.pop() { + let mut hull = loop { + let Some(next) = elements.pop() else { + return Ok(result); + }; + result.insert(next)?; - if !result.all_vertices_on_line() && result.num_vertices() >= 4 { - // We'll need 4 vertices to calculate a center position with good precision. - // Otherwise, dividing by 3.0 can introduce precision loss and errors. - break; + + if let Some(hull) = try_get_hull_center(&result) + .and_then(|center| Hull::from_triangulation(&result, center)) + { + hull_sanity_check(&result, &hull); + + break hull; } - } + }; if elements.is_empty() { return Ok(result); } - // Get new center that is guaranteed to be within the convex hull - let center_positions = || { - result - .vertices() - .rev() - .take(4) - .map(|v| v.position().to_f64()) + let mut buffer = Vec::new(); + let mut skipped_elements = Vec::::new(); + + while let Some(next) = elements.pop() { + skipped_elements.extend( + single_bulk_insertion_step(&mut result, false, &mut hull, next, &mut buffer).err(), + ); + } + + if cfg!(any(fuzzing, test)) { + hull_sanity_check(&result, &hull); + } + + fix_convexity(&mut result); + + for element in skipped_elements { + result.insert(element)?; + } + + Ok(result) +} + +pub fn bulk_load_cdt( + elements: Vec, + mut edges: Vec<[usize; 2]>, +) -> Result, InsertionError> +where + V: HasPosition, + DE: Default, + UE: Default, + F: Default, + L: HintGenerator<::Scalar>, +{ + if elements.is_empty() { + return Ok(ConstrainedDelaunayTriangulation::new()); + } + + if edges.is_empty() { + return bulk_load(elements); + } + + let mut point_sum = Point2::::new(0.0, 0.0); + + for element in &elements { + crate::validate_vertex(element)?; + let position = element.position(); + + point_sum = point_sum.add(position.to_f64()); + } + + // Set the initial center to the average of all positions. This should be a good choice for most triangulations. + // + // The research paper uses a different approach by taking the center of the points' bounding box. + // However, this position might be far off the center off mass if the triangulation has just a few outliers. + // This could lead to a very uneven angle distribution as nearly all points are might be in a very small angle segment + // around the center. This degrades the hull-structure's lookup and insertion performance. + // For this reason, taking the average appears to be a safer option as most vertices should be distributed around the + // initial center. + let initial_center = point_sum.mul(1.0 / (elements.len() as f64)); + + let mut result = ConstrainedDelaunayTriangulation::with_capacity( + elements.len(), + elements.len() * 3, + elements.len() * 2, + ); + + let distance_fn = |position: Point2<::Scalar>| { + ( + Reverse(FloatOrd(initial_center.distance_2(position.to_f64()))), + FloatOrd(position.x), + FloatOrd(position.y), + ) }; - let sum_x = center_positions().map(|p| p.x).sum(); - let sum_y = center_positions().map(|p| p.y).sum(); + for edge in &mut edges { + let [d1, d2] = edge.map(|vertex| distance_fn(elements[vertex].position())); + if d1 > d2 { + edge.reverse(); + } + } + + edges.sort_by_cached_key(|[from, _]| distance_fn(elements[*from].position())); - // Note that we don't re-sort the elements according to their distance to the newest center. This doesn't seem to - // be required for the algorithms performance, probably due to the `center` being close to `initial_center`. - // As of now, it's a unclear how to construct point sets that result in a `center` being farther off - // `initial center` and what the impact of this would be. - let center = Point2::new(sum_x, sum_y).mul(0.25); + let mut elements = elements.into_iter().enumerate().collect::>(); - let mut hull = loop { - if let Some(hull) = Hull::from_triangulation(&result, center) { - break hull; + // Sort by distance, smallest values last. This allows to pop values depending on their distance. + elements.sort_unstable_by_key(|(_, e)| distance_fn(e.position())); + + let mut old_to_new = vec![usize::MAX; elements.len()]; + let mut last_position = None; + let mut last_index = 0; + for (old_index, e) in elements.iter().rev() { + let position = e.position(); + if last_position.is_some() && Some(position) != last_position { + last_index += 1; } + old_to_new[*old_index] = last_index; - // The hull cannot be constructed in some rare cases for very degenerate - // triangulations. Just insert another vertex and try again. Usually hull generation should succeed eventually. - if let Some(next) = elements.pop() { - result.insert(next).unwrap(); - } else { + last_position = Some(position); + } + + let mut next_constraint = edges.pop(); + + let mut buffer = Vec::new(); + + let mut add_constraints_for_new_vertex = + |result: &mut ConstrainedDelaunayTriangulation, index| { + while let Some([from, to]) = next_constraint { + // Check if next creates any constraint edge + if old_to_new[from] == old_to_new[index] { + let [new_from, new_to] = + [from, to].map(|v| FixedVertexHandle::new(old_to_new[v])); + // Insert constraint edge + result.add_constraint(new_from, new_to); + next_constraint = edges.pop(); + } else { + break; + } + } + }; + + let mut hull = loop { + let Some((old_index, next)) = elements.pop() else { return Ok(result); + }; + result.insert(next)?; + add_constraints_for_new_vertex(&mut result, old_index); + + if let Some(hull) = try_get_hull_center(&result) + .and_then(|center| Hull::from_triangulation(&result, center)) + { + break hull; } }; - let mut buffer = Vec::new(); - let mut skipped_elements = Vec::new(); - while let Some(next) = elements.pop() { - skipped_elements.extend( - single_bulk_insertion_step(&mut result, center, &mut hull, next, &mut buffer).err(), - ); + while let Some((old_index, next)) = elements.pop() { + if let Err(skipped) = + single_bulk_insertion_step(&mut result, true, &mut hull, next, &mut buffer) + { + // Sometimes the bulk insertion step fails due to floating point inaccuracies. + // The easiest way to handle these rare occurrences is by skipping them. However, this doesn't + // work as CDT vertices **must** be inserted in their predefined order (after sorting for distance) + // to keep `old_to_new` lookup accurate. + // Instead, this code leverages that the triangulation for CDTs is always convex: This + // means that `result.insert` should work. Unfortunately, using `insert` will invalidate + // the hull structure. We'll recreate it with a loop similar to the initial hull creation. + // + // This process is certainly confusing and inefficient but, luckily, rarely required for real inputs. + + // Push the element again, it will popped directly. This seems to be somewhat simpler than + // the alternatives. + elements.push((old_index, skipped)); + hull = loop { + let Some((old_index, next)) = elements.pop() else { + return Ok(result); + }; + result.insert(next)?; + add_constraints_for_new_vertex(&mut result, old_index); + + if let Some(hull) = Hull::from_triangulation(&result, hull.center) { + break hull; + }; + }; + } else { + add_constraints_for_new_vertex(&mut result, old_index); + } } + assert_eq!(edges.len(), 0); + if cfg!(any(fuzzing, test)) { hull_sanity_check(&result, &hull); } - fix_convexity(&mut result); + Ok(result) +} - for element in skipped_elements { - result.insert(element)?; +fn try_get_hull_center(result: &T) -> Option> +where + V: HasPosition, + T: Triangulation, +{ + let zero = ::Scalar::zero(); + if !result.all_vertices_on_line() && result.num_vertices() >= 4 { + // We'll need 4 vertices to calculate a center position with good precision. + // Otherwise, dividing by 3.0 can introduce precision loss and errors. + + // Get new center that is usually within the convex hull + let center_positions = || result.vertices().rev().take(4).map(|v| v.position()); + + let sum_x = center_positions() + .map(|p| p.x) + .fold(zero, |num, acc| num + acc); + let sum_y = center_positions() + .map(|p| p.y) + .fold(zero, |num, acc| num + acc); + + // Note that we don't re-sort the elements according to their distance to the newest center. This doesn't seem to + // be required for the algorithms performance, probably due to the `center` being close to `initial_center`. + // As of now, it is a unclear how to construct point sets that result in a `center` being farther off + // `initial center` and what the impact of this would be. + let center = Point2::new(sum_x, sum_y).mul(::Scalar::from(0.25f32)); + + if let crate::PositionInTriangulation::OnFace(_) = result.locate(center) { + return Some(center.to_f64()); + } } - Ok(result) + None } pub struct PointWithIndex { @@ -185,7 +354,10 @@ where } } -pub fn bulk_load_stable(elements: Vec) -> Result +pub fn bulk_load_stable( + constructor: Constructor, + elements: Vec, +) -> Result where V: HasPosition, T: Triangulation, @@ -196,6 +368,7 @@ where Face = T::Face, HintGenerator = T::HintGenerator, >, + Constructor: FnOnce(Vec>) -> Result, { let elements = elements .into_iter() @@ -205,7 +378,7 @@ where let num_original_elements = elements.len(); - let mut with_indices = bulk_load::, T2>(elements)?; + let mut with_indices = constructor(elements)?; if with_indices.num_vertices() != num_original_elements { // Handling duplicates is more complicated - we cannot simply swap the elements into @@ -249,6 +422,10 @@ where // since gaps are eliminated in the step above. let mut current_index = 0; loop { + if current_index >= with_indices.num_vertices() { + break; + } + // Example: The permutation [0 -> 2, 1 -> 0, 2 -> 1, 3 -> 3, 4 -> 4] // (written as [2, 0, 1, 3, 4]) will lead to the following swaps: // Swap 2, 0 (leading to [1, 0, 2, 3, 4]) @@ -263,22 +440,18 @@ where .s_mut() .swap_vertices(FixedVertexHandle::new(old_index), new_index); } - - if current_index >= with_indices.num_vertices() { - break; - } } - let (dcel, hint_generator) = with_indices.into_parts(); + let (dcel, hint_generator, num_constraints) = with_indices.into_parts(); let dcel = dcel.map_vertices(|point_with_index| point_with_index.data); - Ok(T::from_parts(dcel, hint_generator, 0)) + Ok(T::from_parts(dcel, hint_generator, num_constraints)) } #[inline(never)] // Prevent inlining for better profiling data fn single_bulk_insertion_step( result: &mut TR, - center: Point2, + require_convexity: bool, hull: &mut Hull, element: T, buffer_for_edge_legalization: &mut Vec, @@ -288,11 +461,17 @@ where TR: Triangulation, { let next_position = element.position(); - let current_angle = pseudo_angle(next_position.to_f64(), center); + let current_angle = pseudo_angle(next_position.to_f64(), hull.center); let edge_hint = hull.get(current_angle); let edge = result.directed_edge(edge_hint); + + let [from, to] = edge.positions(); + if next_position == from || next_position == to { + return Ok(()); + } + if edge.side_query(next_position).is_on_right_side_or_on_line() { // The position is, for some reason, not on the left side of the edge. This can e.g. happen // if the vertices have the same angle. The safest way to include these elements appears to @@ -334,19 +513,31 @@ where // After: // *if* the angle between v->x1 and x1->x2 is smaller than 90°, the edge x2->v and its new // adjacent face is created. + // + // This only applies to DTs: For CDTs, regular convexity is needed at all points to prevent + // constraint edges to leave the convex hull. let mut current_edge = ccw_walk_start; loop { let handle = result.directed_edge(current_edge); let prev = handle.prev(); let handle = handle.fix(); - let point_projection = - super::math::project_point(next_position, prev.to().position(), prev.from().position()); + let [prev_from, prev_to] = prev.positions(); + // `!point_projection.is_behind_edge` is used to identify if the new face's angle will be less + // than 90° + let angle_condition = require_convexity + || !super::math::project_point(next_position, prev_to, prev_from).is_behind_edge(); + current_edge = prev.fix(); - // `!point_projection.is_after_edge` is used to identify if the new face's angle will be less - // than 90°. - if !point_projection.is_behind_edge() && prev.side_query(next_position).is_on_left_side() { + if angle_condition && prev.side_query(next_position).is_on_left_side() { + let prev_prev = prev.prev(); + if prev + .side_query(prev_prev.from().position()) + .is_on_left_side_or_on_line() + { + assert!(prev_prev.side_query(next_position).is_on_left_side()); + } let new_edge = dcel_operations::create_single_face_between_edge_and_next( result.s_mut(), current_edge, @@ -365,17 +556,24 @@ where let mut current_edge = cw_walk_start; // Same as before: Create faces if they will have inner angles less than 90 degrees. This loop - // goes in the other direction (clockwise). + // goes in the other direction (clockwise). Refer to the code above for more comments. loop { let handle = result.directed_edge(current_edge); let next = handle.next(); let handle = handle.fix(); - let point_projection = - super::math::project_point(next_position, next.from().position(), next.to().position()); + let angle_condition = require_convexity + || !super::math::project_point( + next.from().position(), + next_position, + next.to().position(), + ) + .is_behind_edge(); let next_fix = next.fix(); - if !point_projection.is_behind_edge() && next.side_query(next_position).is_on_left_side() { + let is_on_left_side = next.side_query(next_position).is_on_left_side(); + + if angle_condition && is_on_left_side { let new_edge = dcel_operations::create_single_face_between_edge_and_next( result.s_mut(), current_edge, @@ -399,8 +597,8 @@ where if let Some(second_edge) = outgoing_ch_edge { let first_edge = second_edge.prev(); - let first_angle = pseudo_angle(first_edge.from().position().to_f64(), center); - let second_angle = pseudo_angle(second_edge.to().position().to_f64(), center); + let first_angle = pseudo_angle(first_edge.from().position().to_f64(), hull.center); + let second_angle = pseudo_angle(second_edge.to().position().to_f64(), hull.center); hull.insert( first_angle, @@ -461,13 +659,14 @@ where } } +#[derive(Debug, Copy, Clone)] struct Segment { - from: FloatOrd, - to: FloatOrd, + from: FloatOrd, + to: FloatOrd, } impl Segment { - fn new(from: FloatOrd, to: FloatOrd) -> Self { + fn new(from: FloatOrd, to: FloatOrd) -> Self { assert_ne!(from, to); Self { from, to } } @@ -479,7 +678,7 @@ impl Segment { self.from < self.to } - fn contains_angle(&self, angle: FloatOrd) -> bool { + fn contains_angle(&self, angle: FloatOrd) -> bool { if self.is_non_wrapping_segment() { self.from <= angle && angle < self.to } else { @@ -491,7 +690,7 @@ impl Segment { #[derive(Clone, Copy, Debug)] struct Node { /// Pseudo-angle of this hull entry - angle: FloatOrd, + angle: FloatOrd, /// An edge leaving at this hull entry. edge: FixedDirectedEdgeHandle, @@ -523,6 +722,8 @@ pub struct Hull { buckets: Vec, data: Vec, + center: Point2, + /// Unused indices in data which might be reclaimed later empty: Vec, } @@ -539,15 +740,25 @@ impl Hull { let mut prev_index = hull_size - 1; + let mut last_segment: Option = None; for (current_index, edge) in triangulation.convex_hull().enumerate() { let angle_from = pseudo_angle(edge.from().position().to_f64(), center); let angle_to = pseudo_angle(edge.to().position().to_f64(), center); + if let Some(segment) = last_segment { + if segment.contains_angle(angle_to) { + // In rare cases angle_from will be larger than angle_to due to inaccuracies. + return None; + } + } + if angle_from == angle_to || angle_from.0.is_nan() || angle_to.0.is_nan() { // Should only be possible for very degenerate triangulations return None; } + last_segment = Some(Segment::new(angle_from, angle_to)); + let next_index = (current_index + 1) % hull_size; data.push(Node { @@ -560,6 +771,7 @@ impl Hull { } let mut result = Self { buckets: Vec::new(), + center, data, empty: Vec::new(), }; @@ -624,9 +836,9 @@ impl Hull { /// calling this method will result in an endless loop. fn insert( &mut self, - left_angle: FloatOrd, - middle_angle: FloatOrd, - mut right_angle: FloatOrd, + left_angle: FloatOrd, + middle_angle: FloatOrd, + mut right_angle: FloatOrd, left_edge: FixedDirectedEdgeHandle, mut right_edge: FixedDirectedEdgeHandle, ) { @@ -741,7 +953,7 @@ impl Hull { /// /// An edge is considered to cover an input angle if the input angle is contained in the angle /// segment spanned by `pseudo_angle(edge.from()) .. pseudo_angle(edge.from())` - fn get(&self, angle: FloatOrd) -> FixedDirectedEdgeHandle { + fn get(&self, angle: FloatOrd) -> FixedDirectedEdgeHandle { let mut current_handle = self.buckets[self.floored_bucket(angle)]; loop { let current_node = self.data[current_handle]; @@ -757,11 +969,11 @@ impl Hull { } /// Looks up what bucket a given pseudo-angle will fall into. - fn floored_bucket(&self, angle: FloatOrd) -> usize { + fn floored_bucket(&self, angle: FloatOrd) -> usize { ((angle.0 * (self.buckets.len()) as f64).floor() as usize) % self.buckets.len() } - fn ceiled_bucket(&self, angle: FloatOrd) -> usize { + fn ceiled_bucket(&self, angle: FloatOrd) -> usize { ((angle.0 * (self.buckets.len()) as f64).ceil() as usize) % self.buckets.len() } @@ -804,7 +1016,7 @@ impl Hull { /// 0.75 /// ``` #[inline] -fn pseudo_angle(a: Point2, center: Point2) -> FloatOrd { +fn pseudo_angle(a: Point2, center: Point2) -> FloatOrd { let diff = a.sub(center); let p = diff.x / (diff.x.abs() + diff.y.abs()); @@ -854,12 +1066,17 @@ mod test { use float_next_after::NextAfter; use rand::{seq::SliceRandom, SeedableRng}; + use crate::handles::FixedVertexHandle; use crate::test_utilities::{random_points_with_seed, SEED2}; - use crate::{DelaunayTriangulation, InsertionError, Point2, Triangulation, TriangulationExt}; + use crate::{ + ConstrainedDelaunayTriangulation, DelaunayTriangulation, InsertionError, Point2, + Triangulation, TriangulationExt, + }; use super::Hull; + use alloc::vec; use alloc::vec::Vec; #[test] @@ -898,38 +1115,66 @@ mod test { Ok(()) } - #[test] - fn test_bulk_load_on_epsilon_grid() -> Result<(), InsertionError> { - // Inserts vertices on a grid spaced a part by the smallest possible f64 step - let mut rng = rand::rngs::StdRng::from_seed(*SEED2); - const TEST_REPETITIONS: usize = 30; - const GRID_SIZE: usize = 20; - + fn get_epsilon_grid(grid_size: usize) -> Vec> { // Contains The first GRID_SIZE f64 values that are >= 0.0 - let mut possible_f64: Vec<_> = Vec::with_capacity(GRID_SIZE); + let mut possible_f64: Vec<_> = Vec::with_capacity(grid_size); let mut current_float = crate::MIN_ALLOWED_VALUE; - for _ in 0..GRID_SIZE / 2 { + + for _ in 0..grid_size / 2 { possible_f64.push(current_float); possible_f64.push(-current_float); current_float = current_float.next_after(f64::INFINITY); } - for _ in 0..TEST_REPETITIONS { - let mut vertices = Vec::with_capacity(GRID_SIZE * GRID_SIZE); - for x in 0..GRID_SIZE { - for y in 0..GRID_SIZE { - vertices.push(Point2::new(possible_f64[x], possible_f64[y])); - } + possible_f64.sort_by(|l, r| l.partial_cmp(r).unwrap()); + + let mut vertices = Vec::with_capacity(grid_size * grid_size); + for x in 0..grid_size { + for y in 0..grid_size { + vertices.push(Point2::new(possible_f64[x], possible_f64[y])); } + } + + vertices + } + + #[test] + fn test_bulk_load_on_epsilon_grid() -> Result<(), InsertionError> { + const GRID_SIZE: usize = 20; + let mut rng = rand::rngs::StdRng::from_seed(*SEED2); + const TEST_REPETITIONS: usize = 30; + + let vertices = get_epsilon_grid(GRID_SIZE); + + for _ in 0..TEST_REPETITIONS { + let mut vertices = vertices.clone(); vertices.shuffle(&mut rng); let triangulation = DelaunayTriangulation::<_>::bulk_load(vertices)?; - assert_eq!(triangulation.num_vertices(), GRID_SIZE * GRID_SIZE); + triangulation.sanity_check(); + assert_eq!(triangulation.num_vertices(), GRID_SIZE * GRID_SIZE); } Ok(()) } + #[test] + fn test_cdt_bulk_load_on_epsilon_grid() -> Result<(), InsertionError> { + const GRID_SIZE: usize = 20; + + let vertices = get_epsilon_grid(GRID_SIZE); + // Creates a zig zag pattern + let edges = (0..vertices.len() - 1).map(|i| [i, i + 1]).collect(); + + let triangulation = + ConstrainedDelaunayTriangulation::<_>::bulk_load_cdt_stable(vertices, edges)?; + + triangulation.cdt_sanity_check(); + assert_eq!(triangulation.num_vertices(), GRID_SIZE * GRID_SIZE); + assert_eq!(triangulation.num_constraints(), GRID_SIZE * GRID_SIZE - 1); + Ok(()) + } + #[test] fn test_bulk_load_stable() -> Result<(), InsertionError> { const SIZE: usize = 200; @@ -965,6 +1210,96 @@ mod test { Ok(()) } + fn small_cdt_vertices() -> Vec> { + vec![ + Point2::new(1.0, 1.0), + Point2::new(1.0, -1.0), + Point2::new(-1.0, 0.0), + Point2::new(-0.9, -0.9), + Point2::new(0.0, 2.0), + Point2::new(2.0, 0.4), + Point2::new(-0.2, -1.9), + Point2::new(-2.0, 0.1), + ] + } + + fn check_bulk_load_cdt(edges: Vec<[usize; 2]>) -> Result<(), InsertionError> { + let vertices = small_cdt_vertices(); + + let num_constraints = edges.len(); + let num_vertices = vertices.len(); + let cdt = + ConstrainedDelaunayTriangulation::<_>::bulk_load_cdt_stable(vertices, edges.clone())?; + + cdt.cdt_sanity_check(); + assert_eq!(cdt.num_vertices(), num_vertices); + assert_eq!(cdt.num_constraints(), num_constraints); + + for [from, to] in edges { + let from = FixedVertexHandle::from_index(from); + let to = FixedVertexHandle::from_index(to); + assert_eq!( + cdt.get_edge_from_neighbors(from, to) + .map(|h| h.is_constraint_edge()), + Some(true) + ); + } + + Ok(()) + } + + #[test] + fn test_cdt_bulk_load_small() -> Result<(), InsertionError> { + let edges = vec![[4, 5], [5, 6], [6, 7], [7, 4]]; + check_bulk_load_cdt(edges) + } + + #[test] + fn test_cdt_bulk_load_with_constraint_edges_in_center() -> Result<(), InsertionError> { + let edges = vec![[0, 1], [1, 3], [3, 2], [2, 0]]; + + check_bulk_load_cdt(edges) + } + + #[test] + fn test_cdt_bulk_load_with_duplicates() -> Result<(), InsertionError> { + let mut vertices = small_cdt_vertices(); + vertices.extend(small_cdt_vertices()); + let edges = vec![[0, 1], [9, 3], [11, 2], [10, 0]]; + + let num_constraints = edges.len(); + let cdt = ConstrainedDelaunayTriangulation::<_>::bulk_load_cdt_stable(vertices, edges)?; + assert_eq!(cdt.num_constraints(), num_constraints); + Ok(()) + } + + #[test] + fn test_cdt_bulk_load() -> Result<(), InsertionError> { + const SIZE: usize = 500; + let vertices = random_points_with_seed(SIZE, SEED2); + + let edge_vertices = vertices[0..SIZE / 10].to_vec(); + + let edge_triangulation = DelaunayTriangulation::<_>::bulk_load_stable(edge_vertices)?; + // Take a random subsample of edges + let edges = edge_triangulation + .undirected_edges() + // This should return roughly SIZE / 20 undirected edges + .step_by(edge_triangulation.num_undirected_edges() * 20 / SIZE) + .map(|edge| edge.vertices().map(|v| v.index())) + .collect::>(); + + let num_constraints = edges.len(); + + let cdt = ConstrainedDelaunayTriangulation::<_>::bulk_load_cdt(vertices, edges)?; + + cdt.cdt_sanity_check(); + assert_eq!(cdt.num_vertices(), SIZE); + assert_eq!(cdt.num_constraints(), num_constraints); + + Ok(()) + } + #[test] fn test_bulk_load_stable_with_duplicates() -> Result<(), InsertionError> { const SIZE: usize = 200; @@ -987,6 +1322,20 @@ mod test { Ok(()) } + #[test] + fn test_empty() -> Result<(), InsertionError> { + let cdt = ConstrainedDelaunayTriangulation::>::bulk_load_cdt_stable( + Vec::new(), + Vec::new(), + )?; + assert_eq!(cdt.num_vertices(), 0); + assert_eq!(cdt.num_constraints(), 0); + + let dt = DelaunayTriangulation::>::bulk_load_stable(Vec::new())?; + assert_eq!(dt.num_vertices(), 0); + Ok(()) + } + #[test] fn test_bulk_load() -> Result<(), InsertionError> { const SIZE: usize = 9000; @@ -1041,7 +1390,6 @@ mod test { let mut hull = Hull::from_triangulation(&triangulation, Point2::new(0.0, 0.0)).unwrap(); super::hull_sanity_check(&triangulation, &hull); - let center = Point2::new(0.0, 0.0); let additional_elements = [ Point2::new(0.4, 2.0), Point2::new(-0.4, 3.0), @@ -1052,7 +1400,7 @@ mod test { for (index, element) in additional_elements.iter().enumerate() { super::single_bulk_insertion_step( &mut triangulation, - center, + false, &mut hull, *element, &mut Vec::new(), @@ -1064,4 +1412,61 @@ mod test { } Ok(()) } + + #[test] + fn test_cdt_fuzz_1() -> Result<(), InsertionError> { + let data = vec![ + Point2::new(-2.7049442424493675e-11f64, -2.7049442424493268e-11), + Point2::new(-2.7049442424493268e-11, -2.704944239760038e-11), + Point2::new(-2.704944242438945e-11, -2.704943553980988e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424388623e-11), + Point2::new(-2.7049442424493268e-11, -2.704944239760038e-11), + Point2::new(-2.7049442424493675e-11, 0.0), + ]; + + let mut edges = Vec::<[usize; 2]>::new(); + + for p in &data { + if crate::validate_coordinate(p.x).is_err() || crate::validate_coordinate(p.y).is_err() + { + return Ok(()); + } + if p.x.abs() > 20.0 || p.y.abs() > 20.0 { + return Ok(()); + } + } + + for &[from, to] in &edges { + if from >= data.len() || to >= data.len() || from == to { + return Ok(()); + } + } + + let mut reference_cdt = + ConstrainedDelaunayTriangulation::>::bulk_load(data.clone()).unwrap(); + + let mut last_index = 0; + for (index, [from, to]) in edges + .iter() + .copied() + .map(|e| e.map(FixedVertexHandle::from_index)) + .enumerate() + { + if reference_cdt.can_add_constraint(from, to) { + reference_cdt.add_constraint(from, to); + } else { + last_index = index; + break; + } + } + + edges.truncate(last_index); + + let bulk_loaded = + ConstrainedDelaunayTriangulation::>::bulk_load_cdt(data, edges).unwrap(); + + bulk_loaded.cdt_sanity_check(); + + Ok(()) + } } diff --git a/src/delaunay_core/bulk_load_fuzz_tests.rs b/src/delaunay_core/bulk_load_fuzz_tests.rs index 4ad0efd..40e363e 100644 --- a/src/delaunay_core/bulk_load_fuzz_tests.rs +++ b/src/delaunay_core/bulk_load_fuzz_tests.rs @@ -307,3 +307,85 @@ fn test_bulk_load_fuzz_18() { Point2::new(2.0, -2.0004883110523224), ]); } + +#[test] +fn test_bulk_load_fuzz_19() { + fuzz_test(vec![ + Point2::new(-2.7049442424493675e-11, -2.7049442424493268e-11), + Point2::new(-2.7049442424493268e-11, -2.704944239760038e-11), + Point2::new(-2.704944242438945e-11, -2.704943553980988e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424388623e-11), + Point2::new(-2.7049442424493268e-11, -2.704944239760038e-11), + Point2::new(-2.7049442424493675e-11, 0.0), + ]); +} + +#[test] +fn test_bulk_load_fuzz_20() { + fuzz_test(vec![ + Point2::new(2.2526911184108794e-23f64, 2.252526342575774e-23), + Point2::new(2.252526342575774e-23, 2.252526342575774e-23), + Point2::new(2.252526342575774e-23, 2.252526342575774e-23), + Point2::new(2.252526342575774e-23, 2.252526342575774e-23), + Point2::new(2.252526342575774e-23, 2.252526342575774e-23), + Point2::new(2.252526342575774e-23, 2.252526342575774e-23), + Point2::new(0.11616869159773284, -2.7049442424493675e-11), + Point2::new(-2.704944220765324e-11, -2.7049442438046202e-11), + Point2::new(-2.7035231569778473e-11, -1.1276731182827932e-13), + Point2::new(-2.704944242449326e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-4.545365786677739e-10, 1.1648885083023547e-11), + Point2::new(-2.4306747464459356e-11, -2.7049442424493675e-11), + Point2::new(-2.704944242448623e-11, -2.7049442424493675e-11), + ]); +} + +#[test] +fn test_bulk_load_fuzz_21() { + fuzz_test(vec![ + Point2::new(0.11617647291654166f64, -2.705000112790462e-11), + Point2::new(-2.7049442384471368e-11f64, -2.6555608406219246e-11), + Point2::new(-9.387412660936057e-12, -2.707786413392408e-11), + Point2::new(-4.6189490530823523e-10, -2.7049442424337338e-11), + Point2::new(-2.7049434889184558e-11, -2.696027628257575e-11), + Point2::new(-2.7049379378136938e-11, -2.7049442384471368e-11), + Point2::new(-2.7049442397865077e-11, -2.70494424244932e-11), + Point2::new(-2.704944242258785e-11, -2.704944242279961e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442384314453e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.6992599005632867e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-2.693575558677206e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.704943548559977e-11), + Point2::new(-2.7049442424493662e-11, -2.7049442424493675e-11), + Point2::new(-2.6811408873290566e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424499465e-11, -2.7049442424493675e-11), + Point2::new(-2.659469507360721e-11, -2.7049442424493675e-11), + Point2::new(-2.704944242258785e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424501947e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493882e-11, -2.7049442424493675e-11), + Point2::new(-2.7049317524403405e-11, -2.7049442424493675e-11), + Point2::new(-2.7045889710814875e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493672e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.704944241771741e-11), + Point2::new(-2.7049441936602697e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.70494406897702e-11), + Point2::new(-2.7049442424493675e-11, -2.704943548559977e-11), + Point2::new(-2.704944242407016e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.704944242258785e-11), + Point2::new(-2.7049435729968776e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493672e-11), + Point2::new(-2.7049442424467205e-11, -2.7049441936602697e-11), + Point2::new(-2.7049442424493675e-11, -2.704944239865917e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-2.704944242258785e-11, -2.7049442424493675e-11), + Point2::new(-2.7049442398980546e-11, -2.7049442395376918e-11), + Point2::new(-2.7049442424493675e-11, -2.7049442424493675e-11), + Point2::new(-1.600710678326356e-10, -2.7049442424493675e-11), + Point2::new(-2.7049442424909747e-11, -2.7049442424493675e-11), + ]); +} diff --git a/src/delaunay_core/mod.rs b/src/delaunay_core/mod.rs index b17129a..2509653 100644 --- a/src/delaunay_core/mod.rs +++ b/src/delaunay_core/mod.rs @@ -15,7 +15,7 @@ pub mod refinement; pub mod interpolation; pub mod math; -pub use bulk_load::{bulk_load, bulk_load_stable}; +pub use bulk_load::{bulk_load, bulk_load_cdt, bulk_load_stable}; pub use triangulation_ext::{RemovalResult, TriangulationExt}; diff --git a/src/delaunay_core/refinement.rs b/src/delaunay_core/refinement.rs index d6dc44c..4443c37 100644 --- a/src/delaunay_core/refinement.rs +++ b/src/delaunay_core/refinement.rs @@ -1085,7 +1085,7 @@ mod test { cdt.refine(Default::default()); - cdt.cdt_sanity_check(); + cdt.cdt_sanity_check_with_params(false); Ok(()) } diff --git a/src/delaunay_core/triangulation_ext.rs b/src/delaunay_core/triangulation_ext.rs index 047f38d..d12e3aa 100644 --- a/src/delaunay_core/triangulation_ext.rs +++ b/src/delaunay_core/triangulation_ext.rs @@ -788,7 +788,7 @@ pub trait TriangulationExt: Triangulation { } #[cfg(any(test, fuzzing))] - fn basic_sanity_check(&self) { + fn basic_sanity_check(&self, check_convexity: bool) { self.s().sanity_check(); let all_vertices_on_line = self.s().num_faces() <= 1; @@ -828,12 +828,22 @@ pub trait TriangulationExt: Triangulation { let num_inner_faces = self.s().num_faces() - 1; assert_eq!(num_inner_faces * 3, num_inner_edges); + + if check_convexity { + for edge in self.convex_hull() { + for vert in self.vertices() { + assert!(edge + .side_query(vert.position()) + .is_on_right_side_or_on_line(),); + } + } + } } } #[cfg(any(test, fuzzing))] fn sanity_check(&self) { - self.basic_sanity_check(); + self.basic_sanity_check(true); for edge in self.undirected_edges() { let edge = edge.as_directed(); diff --git a/src/delaunay_triangulation.rs b/src/delaunay_triangulation.rs index f5c233d..6ece0f2 100644 --- a/src/delaunay_triangulation.rs +++ b/src/delaunay_triangulation.rs @@ -1,7 +1,7 @@ use super::delaunay_core::Dcel; use crate::{ - handles::VertexHandle, HasPosition, HintGenerator, InsertionError, LastUsedVertexHintGenerator, - NaturalNeighbor, Point2, Triangulation, TriangulationExt, + delaunay_core::bulk_load, handles::VertexHandle, HasPosition, HintGenerator, InsertionError, + LastUsedVertexHintGenerator, NaturalNeighbor, Point2, Triangulation, TriangulationExt, }; use alloc::vec::Vec; @@ -345,10 +345,13 @@ where /// # Ok(()) } /// ``` pub fn bulk_load_stable(elements: Vec) -> Result { - let result: Self = - crate::delaunay_core::bulk_load_stable::<_, _, DelaunayTriangulation<_, _, _, _, _>>( - elements, - )?; + let mut result: Self = crate::delaunay_core::bulk_load_stable::< + _, + _, + DelaunayTriangulation<_, _, _, _, _>, + _, + >(bulk_load, elements)?; + *result.hint_generator_mut() = L::initialize_from_triangulation(&result); Ok(result) } } @@ -432,8 +435,9 @@ where ) -> ( Dcel, Self::HintGenerator, + usize, ) { - (self.dcel, self.hint_generator) + (self.dcel, self.hint_generator, 0) } } diff --git a/src/triangulation.rs b/src/triangulation.rs index bc8000f..838817a 100644 --- a/src/triangulation.rs +++ b/src/triangulation.rs @@ -95,6 +95,7 @@ pub trait Triangulation: Default { ) -> ( Dcel, Self::HintGenerator, + usize, ); /// Creates a new triangulation. @@ -169,8 +170,7 @@ pub trait Triangulation: Default { #[doc = include_str!("../images/bulk_load_vs_incremental_graph.svg")] fn bulk_load(elements: Vec) -> Result { let mut result: Self = crate::delaunay_core::bulk_load(elements)?; - let hint_generator = Self::HintGenerator::initialize_from_triangulation(&result); - *result.hint_generator_mut() = hint_generator; + *result.hint_generator_mut() = Self::HintGenerator::initialize_from_triangulation(&result); Ok(result) }