diff --git a/honeycomb-core/src/attributes/manager.rs b/honeycomb-core/src/attributes/manager.rs
index 21a70e91a..d8691beab 100644
--- a/honeycomb-core/src/attributes/manager.rs
+++ b/honeycomb-core/src/attributes/manager.rs
@@ -21,9 +21,11 @@ use std::{any::TypeId, collections::HashMap};
macro_rules! get_storage {
($slf: ident, $id: ident) => {
let probably_storage = match A::BIND_POLICY {
- OrbitPolicy::Vertex => $slf.vertices.get(&TypeId::of::()),
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
+ $slf.vertices.get(&TypeId::of::())
+ }
OrbitPolicy::Edge => $slf.edges.get(&TypeId::of::()),
- OrbitPolicy::Face => $slf.faces.get(&TypeId::of::()),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => $slf.faces.get(&TypeId::of::()),
OrbitPolicy::Custom(_) => $slf.others.get(&TypeId::of::()),
};
let $id = probably_storage
@@ -35,9 +37,11 @@ macro_rules! get_storage {
macro_rules! get_storage_mut {
($slf: ident, $id: ident) => {
let probably_storage = match A::BIND_POLICY {
- OrbitPolicy::Vertex => $slf.vertices.get_mut(&TypeId::of::()),
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
+ $slf.vertices.get_mut(&TypeId::of::())
+ }
OrbitPolicy::Edge => $slf.edges.get_mut(&TypeId::of::()),
- OrbitPolicy::Face => $slf.faces.get_mut(&TypeId::of::()),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => $slf.faces.get_mut(&TypeId::of::()),
OrbitPolicy::Custom(_) => $slf.others.get_mut(&TypeId::of::()),
};
let $id = probably_storage
@@ -162,9 +166,13 @@ impl AttrStorageManager {
let typeid = TypeId::of::();
let new_storage = ::StorageType::new(size);
if match A::BIND_POLICY {
- OrbitPolicy::Vertex => self.vertices.insert(typeid, Box::new(new_storage)),
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
+ self.vertices.insert(typeid, Box::new(new_storage))
+ }
OrbitPolicy::Edge => self.edges.insert(typeid, Box::new(new_storage)),
- OrbitPolicy::Face => self.faces.insert(typeid, Box::new(new_storage)),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => {
+ self.faces.insert(typeid, Box::new(new_storage))
+ }
OrbitPolicy::Custom(_) => self.others.insert(typeid, Box::new(new_storage)),
}
.is_some()
@@ -212,9 +220,9 @@ impl AttrStorageManager {
#[must_use = "unused getter result - please remove this method call"]
pub fn get_storage(&self) -> Option<&::StorageType> {
let probably_storage = match A::BIND_POLICY {
- OrbitPolicy::Vertex => &self.vertices[&TypeId::of::()],
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => &self.vertices[&TypeId::of::()],
OrbitPolicy::Edge => &self.edges[&TypeId::of::()],
- OrbitPolicy::Face => &self.faces[&TypeId::of::()],
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => &self.faces[&TypeId::of::()],
OrbitPolicy::Custom(_) => &self.others[&TypeId::of::()],
};
probably_storage.downcast_ref::<::StorageType>()
@@ -231,9 +239,11 @@ impl AttrStorageManager {
pub fn remove_storage(&mut self) {
// we could return it ?
let _ = match A::BIND_POLICY {
- OrbitPolicy::Vertex => &self.vertices.remove(&TypeId::of::()),
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
+ &self.vertices.remove(&TypeId::of::())
+ }
OrbitPolicy::Edge => &self.edges.remove(&TypeId::of::()),
- OrbitPolicy::Face => &self.faces.remove(&TypeId::of::()),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => &self.faces.remove(&TypeId::of::()),
OrbitPolicy::Custom(_) => &self.others.remove(&TypeId::of::()),
};
}
@@ -256,9 +266,13 @@ impl AttrStorageManager {
id_in_rhs: DartIdType,
) {
match orbit_policy {
- OrbitPolicy::Vertex => self.force_merge_vertex_attributes(id_out, id_in_lhs, id_in_rhs),
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
+ self.force_merge_vertex_attributes(id_out, id_in_lhs, id_in_rhs);
+ }
OrbitPolicy::Edge => self.force_merge_edge_attributes(id_out, id_in_lhs, id_in_rhs),
- OrbitPolicy::Face => self.force_merge_face_attributes(id_out, id_in_lhs, id_in_rhs),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => {
+ self.force_merge_face_attributes(id_out, id_in_lhs, id_in_rhs);
+ }
OrbitPolicy::Custom(_) => unimplemented!(),
}
}
@@ -335,11 +349,13 @@ impl AttrStorageManager {
id_in_rhs: DartIdType,
) -> StmResult<()> {
match orbit_policy {
- OrbitPolicy::Vertex => {
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
self.merge_vertex_attributes(trans, id_out, id_in_lhs, id_in_rhs)
}
OrbitPolicy::Edge => self.merge_edge_attributes(trans, id_out, id_in_lhs, id_in_rhs),
- OrbitPolicy::Face => self.merge_face_attributes(trans, id_out, id_in_lhs, id_in_rhs),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => {
+ self.merge_face_attributes(trans, id_out, id_in_lhs, id_in_rhs)
+ }
OrbitPolicy::Custom(_) => unimplemented!(),
}
}
@@ -448,13 +464,13 @@ impl AttrStorageManager {
id_in_rhs: DartIdType,
) -> CMapResult<()> {
match orbit_policy {
- OrbitPolicy::Vertex => {
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
self.try_merge_vertex_attributes(trans, id_out, id_in_lhs, id_in_rhs)
}
OrbitPolicy::Edge => {
self.try_merge_edge_attributes(trans, id_out, id_in_lhs, id_in_rhs)
}
- OrbitPolicy::Face => {
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => {
self.try_merge_face_attributes(trans, id_out, id_in_lhs, id_in_rhs)
}
OrbitPolicy::Custom(_) => unimplemented!(),
@@ -637,11 +653,13 @@ impl AttrStorageManager {
id_in: DartIdType,
) {
match orbit_policy {
- OrbitPolicy::Vertex => {
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
self.force_split_vertex_attributes(id_out_lhs, id_out_rhs, id_in);
}
OrbitPolicy::Edge => self.force_split_edge_attributes(id_out_lhs, id_out_rhs, id_in),
- OrbitPolicy::Face => self.force_split_face_attributes(id_out_lhs, id_out_rhs, id_in),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => {
+ self.force_split_face_attributes(id_out_lhs, id_out_rhs, id_in);
+ }
OrbitPolicy::Custom(_) => unimplemented!(),
}
}
@@ -721,11 +739,13 @@ impl AttrStorageManager {
id_in: DartIdType,
) -> StmResult<()> {
match orbit_policy {
- OrbitPolicy::Vertex => {
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
self.split_vertex_attributes(trans, id_out_lhs, id_out_rhs, id_in)
}
OrbitPolicy::Edge => self.split_edge_attributes(trans, id_out_lhs, id_out_rhs, id_in),
- OrbitPolicy::Face => self.split_face_attributes(trans, id_out_lhs, id_out_rhs, id_in),
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => {
+ self.split_face_attributes(trans, id_out_lhs, id_out_rhs, id_in)
+ }
OrbitPolicy::Custom(_) => unimplemented!(),
}
}
@@ -835,13 +855,13 @@ impl AttrStorageManager {
id_in: DartIdType,
) -> CMapResult<()> {
match orbit_policy {
- OrbitPolicy::Vertex => {
+ OrbitPolicy::Vertex | OrbitPolicy::VertexLinear => {
self.try_split_vertex_attributes(trans, id_out_lhs, id_out_rhs, id_in)
}
OrbitPolicy::Edge => {
self.try_split_edge_attributes(trans, id_out_lhs, id_out_rhs, id_in)
}
- OrbitPolicy::Face => {
+ OrbitPolicy::Face | OrbitPolicy::FaceLinear => {
self.try_split_face_attributes(trans, id_out_lhs, id_out_rhs, id_in)
}
OrbitPolicy::Custom(_) => unimplemented!(),
diff --git a/honeycomb-core/src/cmap/builder/grid/tests.rs b/honeycomb-core/src/cmap/builder/grid/tests.rs
index 3f278220c..5fc9baad2 100644
--- a/honeycomb-core/src/cmap/builder/grid/tests.rs
+++ b/honeycomb-core/src/cmap/builder/grid/tests.rs
@@ -112,11 +112,12 @@ fn square_cmap2_correctness() {
assert_eq!(cmap.face_id(3), 1);
assert_eq!(cmap.face_id(4), 1);
+ // i-cell uses beta 0 to ensure correctness, so the iterator is BFS-like
let mut face = cmap.i_cell::<2>(1);
assert_eq!(face.next(), Some(1));
- assert_eq!(face.next(), Some(2));
- assert_eq!(face.next(), Some(3));
- assert_eq!(face.next(), Some(4));
+ assert_eq!(face.next(), Some(2)); // b1
+ assert_eq!(face.next(), Some(4)); // b0
+ assert_eq!(face.next(), Some(3)); // b1b1
assert_eq!(face.next(), None);
assert_eq!(cmap.beta::<1>(1), 2);
@@ -138,8 +139,8 @@ fn square_cmap2_correctness() {
let mut face = cmap.i_cell::<2>(5);
assert_eq!(face.next(), Some(5));
assert_eq!(face.next(), Some(6));
- assert_eq!(face.next(), Some(7));
assert_eq!(face.next(), Some(8));
+ assert_eq!(face.next(), Some(7));
assert_eq!(face.next(), None);
assert_eq!(cmap.beta::<1>(5), 6);
@@ -161,8 +162,8 @@ fn square_cmap2_correctness() {
let mut face = cmap.i_cell::<2>(9);
assert_eq!(face.next(), Some(9));
assert_eq!(face.next(), Some(10));
- assert_eq!(face.next(), Some(11));
assert_eq!(face.next(), Some(12));
+ assert_eq!(face.next(), Some(11));
assert_eq!(face.next(), None);
assert_eq!(cmap.beta::<1>(9), 10);
@@ -184,8 +185,8 @@ fn square_cmap2_correctness() {
let mut face = cmap.i_cell::<2>(13);
assert_eq!(face.next(), Some(13));
assert_eq!(face.next(), Some(14));
- assert_eq!(face.next(), Some(15));
assert_eq!(face.next(), Some(16));
+ assert_eq!(face.next(), Some(15));
assert_eq!(face.next(), None);
assert_eq!(cmap.beta::<1>(13), 14);
diff --git a/honeycomb-core/src/cmap/dim2/orbits.rs b/honeycomb-core/src/cmap/dim2/orbits.rs
index e50c5aae6..4d7989c16 100644
--- a/honeycomb-core/src/cmap/dim2/orbits.rs
+++ b/honeycomb-core/src/cmap/dim2/orbits.rs
@@ -20,10 +20,14 @@ use std::collections::{BTreeSet, VecDeque};
pub enum OrbitPolicy {
/// 0-cell orbit.
Vertex,
+ /// 0-cell orbit, without using beta 0. This requires the cell to be complete / closed.
+ VertexLinear,
/// 1-cell orbit.
Edge,
/// 2-cell orbit.
Face,
+ /// 2-cell orbit, without using beta 0. This requires the cell to be complete / closed.
+ FaceLinear,
/// Ordered array of beta functions that define the orbit.
Custom(&'static [u8]),
}
@@ -125,7 +129,6 @@ impl<'a, T: CoordsFloat> Iterator for Orbit2<'a, T> {
match self.orbit_policy {
OrbitPolicy::Vertex => {
// THIS CODE IS ONLY VALID IN 2D
-
let image1 = self.map_handle.beta::<1>(self.map_handle.beta::<2>(d));
if self.marked.insert(image1) {
// if true, we did not see this dart yet
@@ -139,6 +142,15 @@ impl<'a, T: CoordsFloat> Iterator for Orbit2<'a, T> {
self.pending.push_back(image2);
}
}
+ OrbitPolicy::VertexLinear => {
+ // THIS CODE IS ONLY VALID IN 2D
+ let image = self.map_handle.beta::<1>(self.map_handle.beta::<2>(d));
+ if self.marked.insert(image) {
+ // if true, we did not see this dart yet
+ // i.e. we need to visit it later
+ self.pending.push_back(image);
+ }
+ }
OrbitPolicy::Edge => {
// THIS CODE IS ONLY VALID IN 2D
let image = self.map_handle.beta::<2>(d);
@@ -150,7 +162,21 @@ impl<'a, T: CoordsFloat> Iterator for Orbit2<'a, T> {
}
OrbitPolicy::Face => {
// THIS CODE IS ONLY VALID IN 2D
- // WE ASSUME THAT THE FACE IS COMPLETE
+ let image1 = self.map_handle.beta::<1>(d);
+ if self.marked.insert(image1) {
+ // if true, we did not see this dart yet
+ // i.e. we need to visit it later
+ self.pending.push_back(image1);
+ }
+ let image2 = self.map_handle.beta::<0>(d);
+ if self.marked.insert(image2) {
+ // if true, we did not see this dart yet
+ // i.e. we need to visit it later
+ self.pending.push_back(image2);
+ }
+ }
+ OrbitPolicy::FaceLinear => {
+ // THIS CODE IS ONLY VALID IN 2D
let image = self.map_handle.beta::<1>(d);
if self.marked.insert(image) {
// if true, we did not see this dart yet
@@ -184,18 +210,34 @@ mod tests {
use super::*;
fn simple_map() -> CMap2 {
- let mut map: CMap2 = CMap2::new(6);
+ let mut map: CMap2 = CMap2::new(11);
+ // tri1
map.force_one_link(1, 2);
map.force_one_link(2, 3);
map.force_one_link(3, 1);
+ // tri2
map.force_one_link(4, 5);
map.force_one_link(5, 6);
map.force_one_link(6, 4);
+ // pent on top
+ map.force_one_link(7, 8);
+ map.force_one_link(8, 9);
+ map.force_one_link(9, 10);
+ map.force_one_link(10, 11);
+ map.force_one_link(11, 7);
+
+ // link all
map.force_two_link(2, 4);
+ map.force_two_link(6, 7);
+
assert!(map.force_write_vertex(1, (0.0, 0.0)).is_none());
assert!(map.force_write_vertex(2, (1.0, 0.0)).is_none());
assert!(map.force_write_vertex(6, (1.0, 1.0)).is_none());
assert!(map.force_write_vertex(3, (0.0, 1.0)).is_none());
+ assert!(map.force_write_vertex(9, (1.5, 1.5)).is_none());
+ assert!(map.force_write_vertex(10, (0.5, 2.0)).is_none());
+ assert!(map.force_write_vertex(11, (-0.5, 1.5)).is_none());
+
map
}
@@ -204,9 +246,30 @@ mod tests {
let map = simple_map();
let orbit = Orbit2::new(&map, OrbitPolicy::Custom(&[1, 2]), 3);
let darts: Vec = orbit.collect();
- assert_eq!(darts.len(), 6);
+ assert_eq!(darts.len(), 11);
// because the algorithm is consistent, we can predict the exact layout
- assert_eq!(&darts, &[3, 1, 2, 4, 5, 6]);
+ assert_eq!(&darts, &[3, 1, 2, 4, 5, 6, 7, 8, 9, 10, 11]);
+ }
+
+ #[test]
+ fn orbit_variants() {
+ let map = simple_map();
+
+ // face is complete, so everything works
+ let face: Vec = Orbit2::new(&map, OrbitPolicy::Face, 7).collect();
+ let face_linear: Vec = Orbit2::new(&map, OrbitPolicy::FaceLinear, 7).collect();
+ let face_custom: Vec =
+ Orbit2::new(&map, OrbitPolicy::Custom(&[0, 1]), 7).collect();
+ assert_eq!(&face, &[7, 8, 11, 9, 10]);
+ assert_eq!(&face_linear, &[7, 8, 9, 10, 11]);
+ assert_eq!(&face_custom, &[7, 11, 8, 10, 9]);
+
+ // vertex is incomplete, so using the linear variant will yield an incomplete orbit
+ let vertex: Vec = Orbit2::new(&map, OrbitPolicy::Vertex, 4).collect();
+ let vertex_linear: Vec =
+ Orbit2::new(&map, OrbitPolicy::VertexLinear, 4).collect();
+ assert_eq!(&vertex, &[4, 3, 7]);
+ assert_eq!(&vertex_linear, &[4, 3]);
}
#[test]
@@ -240,9 +303,8 @@ mod tests {
let map = simple_map();
let orbit = Orbit2::new(&map, OrbitPolicy::Vertex, 4);
let darts: Vec = orbit.collect();
- // note that this one fails if we start at 3, because the vertex is not complete
- assert_eq!(darts.len(), 2);
- assert_eq!(&darts, &[4, 3]);
+ assert_eq!(darts.len(), 3);
+ assert_eq!(&darts, &[4, 3, 7]);
}
#[test]