@@ -1002,6 +1002,78 @@ def remove_colinear_vertices(self, tolerance):
1002
1002
_new_face = Face3D (_boundary , self .plane , _holes , enforce_right_hand = False )
1003
1003
return _new_face
1004
1004
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
+
1005
1077
def flip (self ):
1006
1078
"""Get a face with a flipped direction from this one."""
1007
1079
_new_face = Face3D (reversed (self .vertices ), self .plane .flip (),
0 commit comments