honeycomb_kernels/remeshing/swap.rs
1use honeycomb_core::{
2 cmap::{CMap2, DartIdType, EdgeIdType, NULL_DART_ID, NULL_EDGE_ID, SewError},
3 geometry::CoordsFloat,
4 stm::{Transaction, TransactionClosureResult, abort, try_or_coerce},
5};
6
7/// Error-modeling enum for edge swap routine.
8#[derive(thiserror::Error, Debug, PartialEq, Eq)]
9pub enum EdgeSwapError {
10 /// A core operation failed.
11 #[error("core operation failed: {0}")]
12 FailedCoreOp(#[from] SewError),
13 /// The edge passed as argument is null.
14 #[error("cannot swap null edge")]
15 NullEdge,
16 /// The edge passed as argument is made of single dart, hence doesn't have a cell on each side.
17 #[error("cannot swap an edge adjacent to a single cell")]
18 IncompleteEdge,
19 /// One or both of the cells adjacent to the edge are not triangles.
20 #[error("cannot swap an edge adjacent to a non-triangular cell")]
21 BadTopology,
22}
23
24/// Tip over an edge shared by two triangles.
25///
26/// The edge is tipped in the clockwise direction. Vertices that were shared become exclusive
27/// to each new triangle and vice versa:
28///
29/// ```text
30///
31/// + +
32/// / \ /|\
33/// / \ / | \
34/// / \ / | \
35/// / \ / | \
36/// / \ / | \
37/// +-----------+ --> + | +
38/// \ e / \ | /
39/// \ / \ | /
40/// \ / \ | /
41/// \ / \ | /
42/// \ / \|/
43/// + +
44///
45/// ```
46///
47/// This function expects to operate on a triangular mesh. At the moment, calling it on another type
48/// of mesh may result in non-explicit errors (e.g. an internal sew operation will consistently fail
49/// due to a dart being non-free) as there is no check on each faces' degree.
50///
51/// # Arguments
52///
53/// - `t: &mut Transaction` -- Associated transaction.
54/// - `map: &mut CMap2` -- Edited map.
55/// - `e: EdgeIdType` -- Edge to move.
56///
57/// # Panics
58///
59/// This function will panic if there is no cell on one side of the edge.
60///
61/// # Errors
62///
63/// This function will abort and raise an error if:
64/// - the transaction cannot be completed,
65/// - one internal sew operation fails.
66///
67/// The returned error can be used in conjunction with transaction control to avoid any
68/// modifications in case of failure at attribute level. The user can then choose to retry or
69/// abort as he wishes using `Transaction::with_control_and_err`.
70#[inline]
71pub fn swap_edge<T: CoordsFloat>(
72 t: &mut Transaction,
73 map: &CMap2<T>,
74 e: EdgeIdType,
75) -> TransactionClosureResult<(), EdgeSwapError> {
76 if e == NULL_EDGE_ID {
77 abort(EdgeSwapError::NullEdge)?;
78 }
79 let (l, r) = (e as DartIdType, map.beta_transac::<2>(t, e as DartIdType)?);
80 if r == NULL_DART_ID {
81 abort(EdgeSwapError::IncompleteEdge)?;
82 }
83
84 let (b1l, b1r) = (map.beta_transac::<1>(t, l)?, map.beta_transac::<1>(t, r)?);
85 let (b0l, b0r) = (map.beta_transac::<0>(t, l)?, map.beta_transac::<0>(t, r)?);
86 if map.beta_transac::<1>(t, b1l)? != b0l || map.beta_transac::<1>(t, b1r)? != b0r {
87 abort(EdgeSwapError::BadTopology)?;
88 }
89
90 try_or_coerce!(map.unsew::<1>(t, l), EdgeSwapError);
91 try_or_coerce!(map.unsew::<1>(t, r), EdgeSwapError);
92 try_or_coerce!(map.unsew::<1>(t, b0l), EdgeSwapError);
93 try_or_coerce!(map.unsew::<1>(t, b0r), EdgeSwapError);
94 try_or_coerce!(map.unsew::<1>(t, b1l), EdgeSwapError);
95 try_or_coerce!(map.unsew::<1>(t, b1r), EdgeSwapError);
96
97 try_or_coerce!(map.sew::<1>(t, l, b0r), EdgeSwapError);
98 try_or_coerce!(map.sew::<1>(t, b0r, b1l), EdgeSwapError);
99 try_or_coerce!(map.sew::<1>(t, b1l, l), EdgeSwapError);
100 try_or_coerce!(map.sew::<1>(t, r, b0l), EdgeSwapError);
101 try_or_coerce!(map.sew::<1>(t, b0l, b1r), EdgeSwapError);
102 try_or_coerce!(map.sew::<1>(t, b1r, r), EdgeSwapError);
103
104 Ok(())
105}