1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
use std::slice;
use fj_interop::ext::SliceExt;
use fj_math::{Scalar, Winding};
use crate::{
geometry::path::SurfacePath,
objects::{HalfEdge, Surface},
storage::Handle,
};
#[derive(Clone, Debug, Eq, PartialEq, Hash, Ord, PartialOrd)]
pub struct Cycle {
half_edges: Vec<Handle<HalfEdge>>,
}
impl Cycle {
pub fn new(half_edges: impl IntoIterator<Item = Handle<HalfEdge>>) -> Self {
let half_edges = half_edges.into_iter().collect::<Vec<_>>();
assert!(
!half_edges.is_empty(),
"Cycle must contain at least one half-edge"
);
Self { half_edges }
}
pub fn surface(&self) -> &Handle<Surface> {
if let Some(half_edge) = self.half_edges.first() {
return half_edge.surface();
}
unreachable!(
"Cycle has no half-edges, which the constructor should prevent."
)
}
pub fn half_edges(&self) -> HalfEdgesOfCycle {
self.half_edges.iter()
}
pub fn winding(&self) -> Winding {
if self.half_edges.len() < 3 {
let first = self
.half_edges()
.next()
.expect("Invalid cycle: expected at least one half-edge");
let [a, b] = first.vertices();
let edge_direction_positive = a.position() < b.position();
let circle = match first.curve().path() {
SurfacePath::Circle(circle) => circle,
SurfacePath::Line(_) => unreachable!(
"Invalid cycle: less than 3 edges, but not all are circles"
),
};
let cross_positive = circle.a().cross2d(&circle.b()) > Scalar::ZERO;
if edge_direction_positive == cross_positive {
return Winding::Ccw;
} else {
return Winding::Cw;
}
}
let mut sum = Scalar::ZERO;
for [a, b] in self.half_edges.as_slice().array_windows_ext() {
let [a, b] = [a, b].map(|half_edge| {
let [vertex, _] = half_edge.vertices();
vertex.surface_form().position()
});
sum += (b.u - a.u) * (b.v + a.v);
}
if sum > Scalar::ZERO {
return Winding::Cw;
}
if sum < Scalar::ZERO {
return Winding::Ccw;
}
unreachable!("Encountered invalid cycle: {self:#?}");
}
}
pub type HalfEdgesOfCycle<'a> = slice::Iter<'a, Handle<HalfEdge>>;