From 754cdf70d8887c00bc8963c88b055c2132ca2f3c Mon Sep 17 00:00:00 2001 From: Hanno Braun Date: Thu, 29 Sep 2022 11:42:13 +0200 Subject: [PATCH 1/6] Don't rely on `GlobalEdge` vertex order I'm about to make `GlobalEdge` undirected, so the vertex order can no longer be relied on. --- crates/fj-kernel/src/algorithms/sweep/vertex.rs | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/crates/fj-kernel/src/algorithms/sweep/vertex.rs b/crates/fj-kernel/src/algorithms/sweep/vertex.rs index eb36cef85b..7556b624cc 100644 --- a/crates/fj-kernel/src/algorithms/sweep/vertex.rs +++ b/crates/fj-kernel/src/algorithms/sweep/vertex.rs @@ -53,7 +53,8 @@ impl Sweep for (Vertex, Surface) { // With that out of the way, let's start by creating the `GlobalEdge`, // as that is the most straight-forward part of this operations, and // we're going to need it soon anyway. - let edge_global = vertex.global_form().sweep(path, stores); + let (edge_global, vertices_global) = + vertex.global_form().sweep(path, stores); // Next, let's compute the surface coordinates of the two vertices of // the output `Edge`, as we're going to need these for the rest of this @@ -87,15 +88,13 @@ impl Sweep for (Vertex, Surface) { // And now the vertices. Again, nothing wild here. let vertices = { - let vertices_global = edge_global.vertices(); - // Can be cleaned up, once `zip` is stable: // https://doc.rust-lang.org/std/primitive.array.html#method.zip let [a_surface, b_surface] = points_surface; let [a_global, b_global] = vertices_global; let vertices_surface = [(a_surface, a_global), (b_surface, b_global)].map( - |(point_surface, &vertex_global)| { + |(point_surface, vertex_global)| { SurfaceVertex::new( point_surface, surface, @@ -110,7 +109,7 @@ impl Sweep for (Vertex, Surface) { let [a_global, b_global] = vertices_global; let vertices = [(a_surface, a_global), (b_surface, b_global)]; - vertices.map(|(vertex_surface, &vertex_global)| { + vertices.map(|(vertex_surface, vertex_global)| { Vertex::new( [vertex_surface.position().v], curve.clone(), @@ -127,7 +126,7 @@ impl Sweep for (Vertex, Surface) { } impl Sweep for GlobalVertex { - type Swept = GlobalEdge; + type Swept = (GlobalEdge, [GlobalVertex; 2]); fn sweep(self, path: impl Into>, stores: &Stores) -> Self::Swept { let curve = GlobalCurve::new(stores); @@ -135,7 +134,10 @@ impl Sweep for GlobalVertex { let a = self; let b = GlobalVertex::from_position(self.position() + path.into()); - GlobalEdge::new(curve, [a, b]) + let vertices = [a, b]; + let global_edge = GlobalEdge::new(curve, vertices); + + (global_edge, vertices) } } From 86e7c23c0a18645f21bd250ed9b2674fbbffc7c2 Mon Sep 17 00:00:00 2001 From: Hanno Braun Date: Fri, 9 Sep 2022 16:35:24 +0200 Subject: [PATCH 2/6] Make `GlobalEdge` undirected --- crates/fj-kernel/src/objects/edge.rs | 41 +++++++++++++++++++++++++++- 1 file changed, 40 insertions(+), 1 deletion(-) diff --git a/crates/fj-kernel/src/objects/edge.rs b/crates/fj-kernel/src/objects/edge.rs index b3bce8e1b0..9f3912ad32 100644 --- a/crates/fj-kernel/src/objects/edge.rs +++ b/crates/fj-kernel/src/objects/edge.rs @@ -52,7 +52,9 @@ impl HalfEdge { the half-edge's global form" ); assert_eq!( - &vertices.clone().map(|vertex| *vertex.global_form()), + &normalize_vertex_order( + vertices.clone().map(|vertex| *vertex.global_form()) + ), global_form.vertices(), "The global forms of a half-edge's vertices must match the \ vertices of the half-edge's global form" @@ -121,6 +123,7 @@ impl GlobalEdge { vertices: [GlobalVertex; 2], ) -> Self { let curve = curve.into(); + let vertices = normalize_vertex_order(vertices); Self { curve, vertices } } @@ -142,3 +145,39 @@ impl GlobalEdge { &self.vertices } } + +fn normalize_vertex_order([a, b]: [GlobalVertex; 2]) -> [GlobalVertex; 2] { + if a < b { + [a, b] + } else { + [b, a] + } +} + +#[cfg(test)] +mod tests { + use pretty_assertions::assert_eq; + + use crate::{objects::Surface, partial::HasPartial, stores::Stores}; + + use super::HalfEdge; + + #[test] + fn global_edge_equality() { + let stores = Stores::new(); + + let surface = Surface::xy_plane(); + + let a = [0., 0.]; + let b = [1., 0.]; + + let a_to_b = HalfEdge::partial() + .as_line_segment_from_points(surface, [a, b]) + .build(&stores); + let b_to_a = HalfEdge::partial() + .as_line_segment_from_points(surface, [b, a]) + .build(&stores); + + assert_eq!(a_to_b.global_form(), b_to_a.global_form()); + } +} From 40d0efb0ebe7e7a084b216cc654ceb1725767903 Mon Sep 17 00:00:00 2001 From: Hanno Braun Date: Thu, 29 Sep 2022 11:47:32 +0200 Subject: [PATCH 3/6] Add comment --- crates/fj-kernel/src/algorithms/sweep/vertex.rs | 3 +++ 1 file changed, 3 insertions(+) diff --git a/crates/fj-kernel/src/algorithms/sweep/vertex.rs b/crates/fj-kernel/src/algorithms/sweep/vertex.rs index 7556b624cc..73a083e534 100644 --- a/crates/fj-kernel/src/algorithms/sweep/vertex.rs +++ b/crates/fj-kernel/src/algorithms/sweep/vertex.rs @@ -137,6 +137,9 @@ impl Sweep for GlobalVertex { let vertices = [a, b]; let global_edge = GlobalEdge::new(curve, vertices); + // The vertices of the returned `GlobalEdge` are in normalized order, + // which means the order can't be relied upon by the caller. Return the + // ordered vertices in addition. (global_edge, vertices) } } From 772489cc90d6fe0fed3fc921180a8173e43cbb07 Mon Sep 17 00:00:00 2001 From: Hanno Braun Date: Thu, 29 Sep 2022 11:49:31 +0200 Subject: [PATCH 4/6] Rename `GlobalEdge::vertices` The new name is very verbose, but I think that is necessary in order to make mistakes exceedingly unlikely. --- crates/fj-kernel/src/algorithms/transform/edge.rs | 2 +- crates/fj-kernel/src/objects/edge.rs | 4 ++-- crates/fj-kernel/src/partial/objects/edge.rs | 2 +- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/crates/fj-kernel/src/algorithms/transform/edge.rs b/crates/fj-kernel/src/algorithms/transform/edge.rs index 03dbb2f01b..4ff8324a3e 100644 --- a/crates/fj-kernel/src/algorithms/transform/edge.rs +++ b/crates/fj-kernel/src/algorithms/transform/edge.rs @@ -38,7 +38,7 @@ impl TransformObject for GlobalEdge { fn transform(self, transform: &Transform, stores: &Stores) -> Self { let curve = self.curve().clone().transform(transform, stores); let vertices = self - .vertices() + .vertices_in_normalized_order() .map(|vertex| vertex.transform(transform, stores)); Self::new(curve, vertices) diff --git a/crates/fj-kernel/src/objects/edge.rs b/crates/fj-kernel/src/objects/edge.rs index 9f3912ad32..3113bd9772 100644 --- a/crates/fj-kernel/src/objects/edge.rs +++ b/crates/fj-kernel/src/objects/edge.rs @@ -55,7 +55,7 @@ impl HalfEdge { &normalize_vertex_order( vertices.clone().map(|vertex| *vertex.global_form()) ), - global_form.vertices(), + global_form.vertices_in_normalized_order(), "The global forms of a half-edge's vertices must match the \ vertices of the half-edge's global form" ); @@ -141,7 +141,7 @@ impl GlobalEdge { /// An edge has either two bounding vertices or none. The latter is possible /// if the edge's curve is continuous (i.e. connects to itself), and defines /// the whole edge. - pub fn vertices(&self) -> &[GlobalVertex; 2] { + pub fn vertices_in_normalized_order(&self) -> &[GlobalVertex; 2] { &self.vertices } } diff --git a/crates/fj-kernel/src/partial/objects/edge.rs b/crates/fj-kernel/src/partial/objects/edge.rs index d884e60204..ea5591ff95 100644 --- a/crates/fj-kernel/src/partial/objects/edge.rs +++ b/crates/fj-kernel/src/partial/objects/edge.rs @@ -254,7 +254,7 @@ impl From<&GlobalEdge> for PartialGlobalEdge { fn from(global_edge: &GlobalEdge) -> Self { Self { curve: Some(global_edge.curve().clone().into()), - vertices: Some(*global_edge.vertices()), + vertices: Some(*global_edge.vertices_in_normalized_order()), } } } From cea18091e9b330b60c25a8c8a8af855a5e2b5493 Mon Sep 17 00:00:00 2001 From: Hanno Braun Date: Thu, 29 Sep 2022 11:23:07 +0200 Subject: [PATCH 5/6] Remove `Reverse` implementation of `GlobalEdge` `GlobalEdge` is undirected now, so it no longer makes any sense. --- crates/fj-kernel/src/algorithms/reverse/edge.rs | 15 ++------------- 1 file changed, 2 insertions(+), 13 deletions(-) diff --git a/crates/fj-kernel/src/algorithms/reverse/edge.rs b/crates/fj-kernel/src/algorithms/reverse/edge.rs index adf0b4237c..f8bcb5f27e 100644 --- a/crates/fj-kernel/src/algorithms/reverse/edge.rs +++ b/crates/fj-kernel/src/algorithms/reverse/edge.rs @@ -1,4 +1,4 @@ -use crate::objects::{GlobalEdge, HalfEdge}; +use crate::objects::HalfEdge; use super::Reverse; @@ -12,18 +12,7 @@ impl Reverse for HalfEdge { HalfEdge::new( self.curve().clone(), vertices, - self.global_form().clone().reverse(), + self.global_form().clone(), ) } } - -impl Reverse for GlobalEdge { - fn reverse(self) -> Self { - let vertices = { - let &[a, b] = self.vertices(); - [b, a] - }; - - GlobalEdge::new(self.curve().clone(), vertices) - } -} From d7dc21a08c0b1ed7f8a8157a5d4bc9d0a386a175 Mon Sep 17 00:00:00 2001 From: Hanno Braun Date: Mon, 12 Sep 2022 12:21:25 +0200 Subject: [PATCH 6/6] Update documentation of `GlobalEdge` --- crates/fj-kernel/src/objects/edge.rs | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/crates/fj-kernel/src/objects/edge.rs b/crates/fj-kernel/src/objects/edge.rs index 3113bd9772..143041d1e1 100644 --- a/crates/fj-kernel/src/objects/edge.rs +++ b/crates/fj-kernel/src/objects/edge.rs @@ -110,6 +110,11 @@ impl fmt::Display for HalfEdge { } /// An edge, defined in global (3D) coordinates +/// +/// In contract to [`HalfEdge`], `GlobalEdge` is undirected, meaning it has no +/// defined direction, and its vertices have no defined order. This means it can +/// be used to determine whether two [`HalfEdge`]s map to the same `GlobalEdge`, +/// regardless of their direction. #[derive(Clone, Debug, Eq, PartialEq, Hash, Ord, PartialOrd)] pub struct GlobalEdge { curve: HandleWrapper, @@ -118,6 +123,10 @@ pub struct GlobalEdge { impl GlobalEdge { /// Create a new instance + /// + /// The order of `vertices` is irrelevant. Two `GlobalEdge`s with the same + /// `curve` and `vertices` will end up being equal, regardless of the order + /// of `vertices` here. pub fn new( curve: impl Into>, vertices: [GlobalVertex; 2], @@ -138,9 +147,10 @@ impl GlobalEdge { /// Access the vertices that bound the edge on the curve /// - /// An edge has either two bounding vertices or none. The latter is possible - /// if the edge's curve is continuous (i.e. connects to itself), and defines - /// the whole edge. + /// As the name indicates, the order of the returned vertices is normalized + /// and might not match the order of the vertices that were passed to + /// [`GlobalEdge::new`]. You must not rely on the vertices being in any + /// specific order. pub fn vertices_in_normalized_order(&self) -> &[GlobalVertex; 2] { &self.vertices }