@@ -2188,7 +2188,8 @@ def gap_crossing_boundary(polygons, min_separation, tolerance):
2188
2188
return closed_polys
2189
2189
2190
2190
@staticmethod
2191
- def common_axes (polygons , direction , min_distance , merge_distance , angle_tolerance ):
2191
+ def common_axes (polygons , direction , min_distance , merge_distance , angle_tolerance ,
2192
+ filter_tolerance = 0 ):
2192
2193
"""Get LineSegment2Ds for the most common axes across a set of Polygon2Ds.
2193
2194
2194
2195
This is often useful as a step before aligning a set of polygons to these
@@ -2212,6 +2213,11 @@ def common_axes(polygons, direction, min_distance, merge_distance, angle_toleran
2212
2213
angle_tolerance: The max angle difference in radians that the polygon
2213
2214
segments direction can differ from the input direction before the
2214
2215
segments are not factored into this calculation of common axes.
2216
+ filter_tolerance: A number that can be used to filter out axes in the
2217
+ result, which are already perfectly aligned with the input polygon
2218
+ segments. Setting this to zero wil guarantee that no axes are
2219
+ filtered out no matter how close they are to the existing polygon
2220
+ segments. (Default: 0).
2215
2221
2216
2222
Returns:
2217
2223
A tuple with two elements.
@@ -2281,27 +2287,49 @@ def common_axes(polygons, direction, min_distance, merge_distance, angle_toleran
2281
2287
return [], [] # none of the generated axes are relevant
2282
2288
2283
2289
# group the axes by proximity
2284
- last_ax = rel_axes [0 ]
2290
+ grp_base_axis = last_ax = rel_axes [0 ]
2285
2291
axes_groups = [[last_ax ]]
2286
2292
group_values = [[axes_value [0 ]]]
2287
2293
for axis , val in zip (rel_axes [1 :], axes_value [1 :]):
2288
- if axis .p .distance_to_point (last_ax .p ) <= merge_distance :
2294
+ if axis .p .distance_to_point (last_ax .p ) <= merge_distance and \
2295
+ axis .p .distance_to_point (grp_base_axis .p ) <= 2 * merge_distance :
2289
2296
axes_groups [- 1 ].append (axis )
2290
2297
group_values [- 1 ].append (val )
2291
2298
else : # start a new group
2292
2299
axes_groups .append ([axis ])
2293
2300
group_values .append ([val ])
2301
+ grp_base_axis = axis
2294
2302
last_ax = axis
2295
2303
2296
2304
# average the line segments that are within the merge_distance of one another
2297
- axis_values = [max (val ) for val in group_values ]
2298
- common_axes = []
2305
+ average_values = [max (val ) for val in group_values ]
2306
+ average_axes = []
2299
2307
for ax_group , grp_vals in zip (axes_groups , group_values ):
2300
2308
if len (ax_group ) == 1 :
2301
- common_axes .append (ax_group [0 ])
2309
+ average_axes .append (ax_group [0 ])
2302
2310
else :
2303
2311
index_max = max (range (len (grp_vals )), key = grp_vals .__getitem__ )
2304
- common_axes .append (ax_group [index_max ])
2312
+ average_axes .append (ax_group [index_max ])
2313
+
2314
+ # snap common axes to the nearest polygon segment
2315
+ common_axes , axis_values = [], []
2316
+ for com_ax , ax_val in zip (average_axes , average_values ):
2317
+ dists , m_vecs = [], []
2318
+ for pt in mid_pts :
2319
+ close_pt = closest_point2d_on_line2d_infinite (pt , com_ax )
2320
+ dist = close_pt .distance_to_point (pt )
2321
+ if dist <= merge_distance :
2322
+ dists .append (dist )
2323
+ m_vecs .append (pt - close_pt )
2324
+ if filter_tolerance > 0 : # evaluate whether axis is already aligned
2325
+ if max (dists ) - min (dists ) <= filter_tolerance :
2326
+ continue # axis is already aligned
2327
+ sort_vecs = [v for _ , v in sorted (zip (dists , m_vecs ),
2328
+ key = lambda pair : pair [0 ])]
2329
+ m_vec = sort_vecs [0 ]
2330
+ common_axes .append (com_ax .move (m_vec ))
2331
+ axis_values .append (ax_val )
2332
+
2305
2333
return common_axes , axis_values
2306
2334
2307
2335
@staticmethod
0 commit comments