Skip to content

Commit 09488b5

Browse files
committed
fix(face): Add a method to separate Face3D boundary and holes
This will be used in the importers from IDF, OSM, and gbXML, which all represent Faces with a single flat list of vertices.
1 parent 9bc3a40 commit 09488b5

File tree

2 files changed

+99
-0
lines changed

2 files changed

+99
-0
lines changed

ladybug_geometry/geometry3d/face.py

+72
Original file line numberDiff line numberDiff line change
@@ -1002,6 +1002,78 @@ def remove_colinear_vertices(self, tolerance):
10021002
_new_face = Face3D(_boundary, self.plane, _holes, enforce_right_hand=False)
10031003
return _new_face
10041004

1005+
def separate_boundary_and_holes(self, tolerance):
1006+
"""Get a version of this face with boundaries and holes separated.
1007+
1008+
This method is intended for the case that a Face3D has been imported from
1009+
a format where everything was collapsed into a single list of vertices.
1010+
As such, the Face3D.boundary includes both the real shape boundary and the
1011+
holes by winding inward to cut them out.
1012+
1013+
Args:
1014+
tolerance: The minimum distance between vertices at which point they are
1015+
considered equivalent.
1016+
"""
1017+
# first check the holes are not already separated
1018+
if self.has_holes:
1019+
return self
1020+
# loop through the vertices and identify pairs of duplicate vertices
1021+
boundary, all_holes = self.boundary, []
1022+
iter_count, max_holes = 0, int(len(boundary) / 3)
1023+
while iter_count < max_holes:
1024+
boundary, hole = self._separate_inner_most_hole(boundary, tolerance)
1025+
if hole is None:
1026+
break
1027+
else:
1028+
all_holes.append(hole)
1029+
iter_count += 1
1030+
return Face3D(boundary, self.plane, all_holes)
1031+
1032+
@staticmethod
1033+
def _separate_inner_most_hole(vertices, tolerance):
1034+
"""Separate the inner-most hole from a list of flat vertices:
1035+
1036+
Args:
1037+
vertices: A flat list of Point3D.
1038+
tolerance: The minimum distance between vertices at which point they are
1039+
considered equivalent.
1040+
1041+
Returns:
1042+
A tuple with two elements
1043+
1044+
- remain_vertices: A list of Point3D with the inner-most hole
1045+
separated from it.
1046+
1047+
- hole: A list of Point3D for the inner-most hole. Will be None
1048+
if the input vertices had no hole.
1049+
"""
1050+
# loop through the vertices and identify pairs of duplicate vertices
1051+
dup_pairs, all_dups = [], set()
1052+
for i, pt in enumerate(vertices):
1053+
if i in all_dups:
1054+
continue
1055+
if i + 2 >= len(vertices):
1056+
break
1057+
for j in range(i + 2, len(vertices)):
1058+
if pt.is_equivalent(vertices[j], tolerance):
1059+
dup_pairs.append((i, j))
1060+
all_dups.add(i)
1061+
all_dups.add(j)
1062+
break
1063+
if len(dup_pairs) == 0:
1064+
return vertices, None # no holes were detected
1065+
# find the duplicate pair with no other duplicates between them (inner most)
1066+
for pair_low, pair_high in dup_pairs:
1067+
for btw_pt_i in range(pair_low + 1, pair_high):
1068+
if btw_pt_i in all_dups:
1069+
break # not an inner-most hole
1070+
else:
1071+
break # the current pair is an inner-most one
1072+
# separate the inner-most hole from the boundary
1073+
hole = vertices[pair_low:pair_high]
1074+
remain_vertices = vertices[:pair_low] + vertices[pair_high + 2:]
1075+
return remain_vertices, hole
1076+
10051077
def flip(self):
10061078
"""Get a face with a flipped direction from this one."""
10071079
_new_face = Face3D(reversed(self.vertices), self.plane.flip(),

tests/face3d_test.py

+27
Original file line numberDiff line numberDiff line change
@@ -791,6 +791,33 @@ def test_remove_colinear_vertices_custom():
791791
assert len(face_geo.remove_colinear_vertices(0.0001).vertices) == 16
792792

793793

794+
def test_separate_boundary_and_holes():
795+
"""Test the separate_boundary_and_holes."""
796+
bound_pts = [Point3D(0, 0), Point3D(4, 0), Point3D(4, 4), Point3D(0, 4)]
797+
hole_pts_1 = [Point3D(1, 1), Point3D(1.5, 1), Point3D(1.5, 1.5), Point3D(1, 1.5)]
798+
hole_pts_2 = [Point3D(2, 2), Point3D(3, 2), Point3D(3, 3), Point3D(2, 3)]
799+
face_1 = Face3D(bound_pts)
800+
face_2 = Face3D(Face3D(bound_pts, None, [hole_pts_2]).vertices)
801+
face_3 = Face3D(Face3D(bound_pts, None, [hole_pts_1, hole_pts_2]).vertices)
802+
803+
clean_face_1 = face_1.separate_boundary_and_holes(0.01)
804+
assert len(clean_face_1.boundary) == 4
805+
assert not clean_face_1.has_holes
806+
807+
clean_face_2 = face_2.separate_boundary_and_holes(0.01)
808+
assert len(clean_face_2.boundary) == 4
809+
assert len(clean_face_2.holes) == 1
810+
assert len(clean_face_2.holes[0]) == 4
811+
assert clean_face_2.area == face_2.area
812+
813+
clean_face_3 = face_3.separate_boundary_and_holes(0.01)
814+
assert len(clean_face_3.boundary) == 4
815+
assert len(clean_face_3.holes) == 2
816+
assert len(clean_face_3.holes[0]) == 4
817+
assert len(clean_face_3.holes[1]) == 4
818+
assert clean_face_3.area == face_3.area
819+
820+
794821
def test_triangulated_mesh_and_centroid():
795822
"""Test the triangulation properties of Face3D."""
796823
pts_1 = (Point3D(0, 0, 2), Point3D(2, 0, 2), Point3D(2, 2, 2), Point3D(0, 2, 2))

0 commit comments

Comments
 (0)