From ecc7f26e3f01e160d97bb6b92e6330f72d36b2c0 Mon Sep 17 00:00:00 2001 From: Hanno Braun Date: Mon, 18 Mar 2024 13:56:28 +0100 Subject: [PATCH] Read half-edge path from geometry layer --- crates/fj-core/src/algorithms/approx/edge.rs | 10 ++++--- .../src/algorithms/bounding_volume/edge.rs | 9 ++++--- crates/fj-core/src/objects/kinds/cycle.rs | 4 +-- .../fj-core/src/operations/build/half_edge.rs | 4 +-- .../src/operations/geometry/half_edge.rs | 4 +-- crates/fj-core/src/operations/holes.rs | 19 ++++++++++--- .../fj-core/src/operations/reverse/cycle.rs | 4 +-- crates/fj-core/src/operations/reverse/edge.rs | 2 +- .../fj-core/src/operations/split/half_edge.rs | 12 ++++++--- crates/fj-core/src/operations/sweep/cycle.rs | 2 +- .../fj-core/src/operations/sweep/half_edge.rs | 7 ++++- .../fj-core/src/operations/transform/edge.rs | 2 +- crates/fj-core/src/validate/shell.rs | 27 ++++++++++++++----- .../validation/checks/half_edge_connection.rs | 7 +++-- 14 files changed, 79 insertions(+), 34 deletions(-) diff --git a/crates/fj-core/src/algorithms/approx/edge.rs b/crates/fj-core/src/algorithms/approx/edge.rs index 95a3fa52c..27a59c9da 100644 --- a/crates/fj-core/src/algorithms/approx/edge.rs +++ b/crates/fj-core/src/algorithms/approx/edge.rs @@ -46,7 +46,7 @@ impl Approx for (&Handle, &SurfaceGeometry) { let rest = { let approx = ( half_edge.curve(), - half_edge.path(), + core.layers.geometry.of_half_edge(half_edge).path, surface, half_edge.boundary(), ) @@ -57,8 +57,12 @@ impl Approx for (&Handle, &SurfaceGeometry) { ); approx.points.into_iter().map(|point| { - let point_surface = - half_edge.path().point_from_path_coords(point.local_form); + let point_surface = core + .layers + .geometry + .of_half_edge(half_edge) + .path + .point_from_path_coords(point.local_form); ApproxPoint::new(point_surface, point.global_form) }) diff --git a/crates/fj-core/src/algorithms/bounding_volume/edge.rs b/crates/fj-core/src/algorithms/bounding_volume/edge.rs index b902e33d5..ec7ed05f2 100644 --- a/crates/fj-core/src/algorithms/bounding_volume/edge.rs +++ b/crates/fj-core/src/algorithms/bounding_volume/edge.rs @@ -7,8 +7,8 @@ use crate::{ }; impl super::BoundingVolume<2> for Handle { - fn aabb(&self, _: &Geometry) -> Option> { - match self.path() { + fn aabb(&self, geometry: &Geometry) -> Option> { + match geometry.of_half_edge(self).path { SurfacePath::Circle(circle) => { // Just calculate the AABB of the whole circle. This is not the // most precise, but it should do for now. @@ -23,7 +23,10 @@ impl super::BoundingVolume<2> for Handle { } SurfacePath::Line(_) => { let points = self.boundary().inner.map(|point_curve| { - self.path().point_from_path_coords(point_curve) + geometry + .of_half_edge(self) + .path + .point_from_path_coords(point_curve) }); Some(Aabb::<2>::from_points(points)) diff --git a/crates/fj-core/src/objects/kinds/cycle.rs b/crates/fj-core/src/objects/kinds/cycle.rs index 2e0ac0cf9..69546f9a6 100644 --- a/crates/fj-core/src/objects/kinds/cycle.rs +++ b/crates/fj-core/src/objects/kinds/cycle.rs @@ -29,7 +29,7 @@ impl Cycle { /// Please note that this is not *the* winding of the cycle, only one of the /// two possible windings, depending on the direction you look at the /// surface that the cycle is defined on from. - pub fn winding(&self, _: &Geometry) -> Winding { + pub fn winding(&self, geometry: &Geometry) -> Winding { // The cycle could be made up of one or two circles. If that is the // case, the winding of the cycle is determined by the winding of the // first circle. @@ -43,7 +43,7 @@ impl Cycle { let [a, b] = first.boundary().inner; let edge_direction_positive = a < b; - let circle = match first.path() { + let circle = match geometry.of_half_edge(first).path { SurfacePath::Circle(circle) => circle, SurfacePath::Line(_) => unreachable!( "Invalid cycle: less than 3 edges, but not all are circles" diff --git a/crates/fj-core/src/operations/build/half_edge.rs b/crates/fj-core/src/operations/build/half_edge.rs index 45f497242..b674210ab 100644 --- a/crates/fj-core/src/operations/build/half_edge.rs +++ b/crates/fj-core/src/operations/build/half_edge.rs @@ -34,7 +34,7 @@ pub trait BuildHalfEdge { core: &mut Core, ) -> Handle { let half_edge = HalfEdge::new( - sibling.path(), + core.layers.geometry.of_half_edge(sibling).path, sibling.boundary().reverse(), sibling.curve().clone(), start_vertex, @@ -44,7 +44,7 @@ pub trait BuildHalfEdge { core.layers.geometry.define_half_edge( half_edge.clone(), HalfEdgeGeometry { - path: sibling.path(), + path: core.layers.geometry.of_half_edge(sibling).path, }, ); diff --git a/crates/fj-core/src/operations/geometry/half_edge.rs b/crates/fj-core/src/operations/geometry/half_edge.rs index 1ac2dd40b..ead3ddb82 100644 --- a/crates/fj-core/src/operations/geometry/half_edge.rs +++ b/crates/fj-core/src/operations/geometry/half_edge.rs @@ -50,7 +50,7 @@ impl UpdateHalfEdgeGeometry for Handle { update: impl FnOnce(SurfacePath) -> SurfacePath, core: &mut Core, ) -> Self { - let path = update(self.path()); + let path = update(core.layers.geometry.of_half_edge(self).path); let half_edge = HalfEdge::new( path, @@ -73,7 +73,7 @@ impl UpdateHalfEdgeGeometry for Handle { core: &mut Core, ) -> Self { HalfEdge::new( - self.path(), + core.layers.geometry.of_half_edge(self).path, update(self.boundary()), self.curve().clone(), self.start_vertex().clone(), diff --git a/crates/fj-core/src/operations/holes.rs b/crates/fj-core/src/operations/holes.rs index 6d878f2c7..3017c09ef 100644 --- a/crates/fj-core/src/operations/holes.rs +++ b/crates/fj-core/src/operations/holes.rs @@ -68,7 +68,10 @@ impl AddHole for Shell { [Cycle::empty().add_joined_edges( [( entry.clone(), - entry.path(), + core.layers + .geometry + .of_half_edge(&entry) + .path, entry.boundary(), )], core, @@ -139,7 +142,10 @@ impl AddHole for Shell { [Cycle::empty().add_joined_edges( [( entry.clone(), - entry.path(), + core.layers + .geometry + .of_half_edge(&entry) + .path, entry.boundary(), )], core, @@ -159,7 +165,14 @@ impl AddHole for Shell { |region, core| { region.add_interiors( [Cycle::empty().add_joined_edges( - [(exit.clone(), exit.path(), exit.boundary())], + [( + exit.clone(), + core.layers + .geometry + .of_half_edge(exit) + .path, + exit.boundary(), + )], core, )], core, diff --git a/crates/fj-core/src/operations/reverse/cycle.rs b/crates/fj-core/src/operations/reverse/cycle.rs index d4a990cbc..d579654de 100644 --- a/crates/fj-core/src/operations/reverse/cycle.rs +++ b/crates/fj-core/src/operations/reverse/cycle.rs @@ -14,7 +14,7 @@ impl Reverse for Cycle { .pairs() .map(|(current, next)| { let half_edge = HalfEdge::new( - current.path(), + core.layers.geometry.of_half_edge(current).path, current.boundary().reverse(), current.curve().clone(), next.start_vertex().clone(), @@ -25,7 +25,7 @@ impl Reverse for Cycle { core.layers.geometry.define_half_edge( half_edge.clone(), HalfEdgeGeometry { - path: current.path(), + path: core.layers.geometry.of_half_edge(current).path, }, ); diff --git a/crates/fj-core/src/operations/reverse/edge.rs b/crates/fj-core/src/operations/reverse/edge.rs index 4b992bec9..bba0b773a 100644 --- a/crates/fj-core/src/operations/reverse/edge.rs +++ b/crates/fj-core/src/operations/reverse/edge.rs @@ -10,7 +10,7 @@ use super::ReverseCurveCoordinateSystems; impl ReverseCurveCoordinateSystems for Handle { fn reverse_curve_coordinate_systems(&self, core: &mut Core) -> Self { - let path = self.path().reverse(); + let path = core.layers.geometry.of_half_edge(self).path.reverse(); let boundary = self.boundary().reverse(); let half_edge = HalfEdge::new( diff --git a/crates/fj-core/src/operations/split/half_edge.rs b/crates/fj-core/src/operations/split/half_edge.rs index 2ec5a5c1b..efc170571 100644 --- a/crates/fj-core/src/operations/split/half_edge.rs +++ b/crates/fj-core/src/operations/split/half_edge.rs @@ -44,14 +44,14 @@ impl SplitHalfEdge for Handle { let [start, end] = self.boundary().inner; let a = HalfEdge::new( - self.path(), + core.layers.geometry.of_half_edge(self).path, [start, point], self.curve().clone(), self.start_vertex().clone(), ) .insert(core); let b = HalfEdge::new( - self.path(), + core.layers.geometry.of_half_edge(self).path, [point, end], self.curve().clone(), Vertex::new().insert(core), @@ -60,11 +60,15 @@ impl SplitHalfEdge for Handle { core.layers.geometry.define_half_edge( a.clone(), - HalfEdgeGeometry { path: self.path() }, + HalfEdgeGeometry { + path: core.layers.geometry.of_half_edge(self).path, + }, ); core.layers.geometry.define_half_edge( b.clone(), - HalfEdgeGeometry { path: self.path() }, + HalfEdgeGeometry { + path: core.layers.geometry.of_half_edge(self).path, + }, ); [a, b] diff --git a/crates/fj-core/src/operations/sweep/cycle.rs b/crates/fj-core/src/operations/sweep/cycle.rs index b64ec1860..23f04051c 100644 --- a/crates/fj-core/src/operations/sweep/cycle.rs +++ b/crates/fj-core/src/operations/sweep/cycle.rs @@ -77,7 +77,7 @@ impl SweepCycle for Cycle { top_edges.push(( top_edge, - bottom_half_edge.path(), + core.layers.geometry.of_half_edge(bottom_half_edge).path, bottom_half_edge.boundary(), )); } diff --git a/crates/fj-core/src/operations/sweep/half_edge.rs b/crates/fj-core/src/operations/sweep/half_edge.rs index a1c07706a..d2db2834b 100644 --- a/crates/fj-core/src/operations/sweep/half_edge.rs +++ b/crates/fj-core/src/operations/sweep/half_edge.rs @@ -59,7 +59,12 @@ impl SweepHalfEdge for Handle { ) -> (Face, Handle) { let path = path.into(); - let surface = self.path().sweep_surface_path(surface, path, core); + let surface = core + .layers + .geometry + .of_half_edge(self) + .path + .sweep_surface_path(surface, path, core); // Next, we need to define the boundaries of the face. Let's start with // the global vertices and edges. diff --git a/crates/fj-core/src/operations/transform/edge.rs b/crates/fj-core/src/operations/transform/edge.rs index 50820b2e3..933ac520e 100644 --- a/crates/fj-core/src/operations/transform/edge.rs +++ b/crates/fj-core/src/operations/transform/edge.rs @@ -16,7 +16,7 @@ impl TransformObject for Handle { ) -> Self { // Don't need to transform the path, as that's defined in surface // coordinates. - let path = self.path(); + let path = core.layers.geometry.of_half_edge(self).path; let boundary = self.boundary(); let curve = self .curve() diff --git a/crates/fj-core/src/validate/shell.rs b/crates/fj-core/src/validate/shell.rs index 51e8d6dba..a8826ae42 100644 --- a/crates/fj-core/src/validate/shell.rs +++ b/crates/fj-core/src/validate/shell.rs @@ -104,6 +104,7 @@ impl ShellValidationError { surface_a: &SurfaceGeometry, edge_b: &Handle, surface_b: &SurfaceGeometry, + geometry: &Geometry, config: &ValidationConfig, mismatches: &mut Vec, ) { @@ -116,10 +117,14 @@ impl ShellValidationError { let c = a + (d - a) * 2. / 3.; for point_curve in [a, b, c, d] { - let a_surface = - edge_a.path().point_from_path_coords(point_curve); - let b_surface = - edge_b.path().point_from_path_coords(point_curve); + let a_surface = geometry + .of_half_edge(edge_a) + .path + .point_from_path_coords(point_curve); + let b_surface = geometry + .of_half_edge(edge_b) + .path + .point_from_path_coords(point_curve); let a_global = surface_a.point_from_surface_coords(a_surface); @@ -148,6 +153,7 @@ impl ShellValidationError { &geometry.of_surface(surface_a), edge_b, &geometry.of_surface(surface_b), + geometry, config, &mut mismatches, ); @@ -156,6 +162,7 @@ impl ShellValidationError { &geometry.of_surface(surface_b), edge_a, &geometry.of_surface(surface_a), + geometry, config, &mut mismatches, ); @@ -245,6 +252,7 @@ impl ShellValidationError { &geometry.of_surface(surface_a), half_edge_b.clone(), &geometry.of_surface(surface_b), + geometry, ) .all(|d| d < config.distinct_min_distance) { @@ -369,14 +377,19 @@ fn distances( surface_a: &SurfaceGeometry, edge_b: Handle, surface_b: &SurfaceGeometry, + geometry: &Geometry, ) -> impl Iterator { fn sample( percent: f64, (edge, surface): (&Handle, &SurfaceGeometry), + geometry: &Geometry, ) -> Point<3> { let [start, end] = edge.boundary().inner; let path_coords = start + (end - start) * percent; - let surface_coords = edge.path().point_from_path_coords(path_coords); + let surface_coords = geometry + .of_half_edge(edge) + .path + .point_from_path_coords(path_coords); surface.point_from_surface_coords(surface_coords) } @@ -389,8 +402,8 @@ fn distances( let mut distances = Vec::new(); for i in 0..sample_count { let percent = i as f64 * step; - let sample1 = sample(percent, (&edge_a, surface_a)); - let sample2 = sample(1.0 - percent, (&edge_b, surface_b)); + let sample1 = sample(percent, (&edge_a, surface_a), geometry); + let sample2 = sample(1.0 - percent, (&edge_b, surface_b), geometry); distances.push(sample1.distance_to(&sample2)) } distances.into_iter() diff --git a/crates/fj-core/src/validation/checks/half_edge_connection.rs b/crates/fj-core/src/validation/checks/half_edge_connection.rs index 2d65a92b9..8e2b8083b 100644 --- a/crates/fj-core/src/validation/checks/half_edge_connection.rs +++ b/crates/fj-core/src/validation/checks/half_edge_connection.rs @@ -41,13 +41,16 @@ pub struct AdjacentHalfEdgesNotConnected { impl ValidationCheck for AdjacentHalfEdgesNotConnected { fn check( object: &Cycle, - _: &Geometry, + geometry: &Geometry, config: &ValidationConfig, ) -> impl Iterator { object.half_edges().pairs().filter_map(|(first, second)| { let end_pos_of_first_half_edge = { let [_, end] = first.boundary().inner; - first.path().point_from_path_coords(end) + geometry + .of_half_edge(first) + .path + .point_from_path_coords(end) }; let start_pos_of_second_half_edge = second.start_position();