Skip to content

Commit d9eae33

Browse files
committed
fix(polygon): Add an extra step to common axes to snap axes to polygons
1 parent 8d83ddb commit d9eae33

File tree

2 files changed

+36
-8
lines changed

2 files changed

+36
-8
lines changed

ladybug_geometry/geometry2d/polygon.py

+35-7
Original file line numberDiff line numberDiff line change
@@ -2188,7 +2188,8 @@ def gap_crossing_boundary(polygons, min_separation, tolerance):
21882188
return closed_polys
21892189

21902190
@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):
21922193
"""Get LineSegment2Ds for the most common axes across a set of Polygon2Ds.
21932194
21942195
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
22122213
angle_tolerance: The max angle difference in radians that the polygon
22132214
segments direction can differ from the input direction before the
22142215
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).
22152221
22162222
Returns:
22172223
A tuple with two elements.
@@ -2281,27 +2287,49 @@ def common_axes(polygons, direction, min_distance, merge_distance, angle_toleran
22812287
return [], [] # none of the generated axes are relevant
22822288

22832289
# group the axes by proximity
2284-
last_ax = rel_axes[0]
2290+
grp_base_axis = last_ax = rel_axes[0]
22852291
axes_groups = [[last_ax]]
22862292
group_values = [[axes_value[0]]]
22872293
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:
22892296
axes_groups[-1].append(axis)
22902297
group_values[-1].append(val)
22912298
else: # start a new group
22922299
axes_groups.append([axis])
22932300
group_values.append([val])
2301+
grp_base_axis = axis
22942302
last_ax = axis
22952303

22962304
# 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 = []
22992307
for ax_group, grp_vals in zip(axes_groups, group_values):
23002308
if len(ax_group) == 1:
2301-
common_axes.append(ax_group[0])
2309+
average_axes.append(ax_group[0])
23022310
else:
23032311
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+
23052333
return common_axes, axis_values
23062334

23072335
@staticmethod

tests/polygon2d_test.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1217,7 +1217,7 @@ def test_common_axes():
12171217
polygons, Vector2D(1, 0), min_distance=0.1, merge_distance=0.3,
12181218
angle_tolerance=math.pi / 180)
12191219

1220-
assert len(axes) == 31
1220+
assert len(axes) == 52
12211221
for item in axes:
12221222
assert isinstance(item, LineSegment2D)
12231223

0 commit comments

Comments
 (0)