FWIW, in case anyone finds this thread in the future, I was able to make this work reasonably well (it’s not fast… but fairly consistent at least) using the implementation shown below (GH file attached for reference).
import scriptcontext as sc
from copy import copy
import math
from ladybug_geometry.geometry3d.plane import Plane
from ladybug_geometry.geometry2d.pointvector import Vector2D
from ladybug_geometry.geometry3d.pointvector import Vector3D
from ladybug_geometry.geometry2d.polygon import Polygon2D
from ladybug_geometry.geometry3d.face import Face3D
from ladybug_rhino.fromgeometry import from_point3d
from honeybee import room, face
from honeybee.aperture import Aperture
def cross_product(a, b):
# type: (List[float], List[float]) -> List[float]
return [
a[1]*b[2] - a[2]*b[1],
a[2]*b[0] - a[0]*b[2],
a[0]*b[1] - a[1]*b[0]
]
def dot_product(a, b):
# type: (List[float], List[float]) -> float
return sum([a[i]*b[i] for i in range(len(a))])
def magnitude(a):
# type: (List[float]) -> float
return sum([a[i]**2 for i in range(len(a))])**0.5
def normalize(a):
# type: (List[float) -> List[float]
mag = magnitude(a)
return [a[i]/mag for i in range(len(a))]
def angle_between_planes(plane1, plane2, _tolerance):
# type: (Plane, Plane, float) -> float
# Calculate the x-axes of the planes
plane1_xaxis = cross_product(plane1.n, plane1.x)
plane2_xaxis = cross_product(plane2.n, plane2.x)
# Normalize the x-axes
plane1_xaxis = normalize(plane1_xaxis)
plane2_xaxis = normalize(plane2_xaxis)
# Calculate the dot product between the x-axes
dot_product_value = dot_product(plane1_xaxis, plane2_xaxis)
# Calculate the angle between the x-axes
# Handle parallel or coincident planes
if (1.0 - dot_product_value) < _tolerance:
return 0.0
angle_rad = math.acos(dot_product_value)
# Determine the counterclockwise angle
orientation = cross_product(plane1_xaxis, plane2_xaxis)
if dot_product(plane1.n, orientation) < 0:
angle_rad = 2 * math.pi - angle_rad
return angle_rad
def _get_polygon2d_in_reference_space(_polygon2d, _poly2d_plane, _base_plane, _tolerance):
# type: (Polygon2D, Plane, Plane, float) -> Polygon2D
"""Return a Polygon2D in the reference space of the base polygon."""
# -- Create a Vector2D from each face's origin to the base-geom's origin
base_plane_x_vec = _base_plane.x
face_origin_in_base_plane_space = _base_plane.xyz_to_xy(_poly2d_plane.o)
base_plane_origin = copy(_base_plane.xyz_to_xy(_base_plane.o))
# -- Construct a Move vector from the face's origin to the base-plane's origin
mv_x = face_origin_in_base_plane_space.x - base_plane_origin.x
mv_y = face_origin_in_base_plane_space.y - base_plane_origin.y
move_vec = Vector2D(mv_x, mv_y) # type: ignore
# ------------------------------------------------------------------------
# -- Move the face's Polygon2D into the base polygon's space
moved_polygon = _polygon2d.move(move_vec)
# ------------------------------------------------------------------------
# -- Rotate the Polygon to align with the base-plane
angle = angle_between_planes(_base_plane, _poly2d_plane, _tolerance)
rotated_polygon = moved_polygon.rotate(angle=angle, origin=face_origin_in_base_plane_space)
return rotated_polygon
def _create_new_HB_Face(_face3D, _ref_face):
# type: (Face3D, Face) -> Face
"""Create a new HB-Face using a Face3D and a reference HB-Face."""
new_face = face.Face(
identifier=_ref_face.identifier,
geometry=_face3D,
type=_ref_face.type,
boundary_condition=_ref_face.boundary_condition,
)
new_face.display_name = _ref_face.display_name
new_face._user_data = (
None if _ref_face.user_data is None else _ref_face.user_data.copy()
)
new_face._properties._duplicate_extension_attr(_ref_face._properties)
return new_face
def _create_new_Face3D(_poly2D, _base_plane, _ref_face):
# type: (Polygon2D, Plane, Face) -> Face3D
return Face3D(
boundary=tuple(_base_plane.xy_to_xyz(v) for v in _poly2D.vertices),
plane=_ref_face.geometry.plane,
)
def _add_sub_face(_face, _aperture):
# type: (Face, Aperture) -> Face
"""Add an HB-sub-face (either HB-Aperture or HB-Door) to a parent Face.
NOTE: this method is copied from honeybee's Grasshopper component "HB Add Subface"
"""
if isinstance(_aperture, Aperture): # the sub-face is an Aperture
_face.add_aperture(_aperture)
else: # the sub-face is a Door
_face.add_door(_aperture)
return _face
def _check_and_add_sub_face(_face, _apertures, _tolerance, _angle_tolerance,):
# type: (Face, List[Aperture], float, float) -> None
"""Check whether a HB-sub-face is valid for an HB-face and, if so, add it.
NOTE: this method is copied from honeybee's Grasshopper component "HB Add Subface"
"""
for aperture in _apertures:
if _face.geometry.is_sub_face(aperture.geometry, _tolerance, _angle_tolerance):
_add_sub_face(_face, aperture)
def find_parent_and_child_polygons(_polygons):
# type: (List[Polygon2D]): -> Tuple[List[Polygon2D], List[Polyon2D]]
# Initialize empty lists for parent and child surfaces
parent_polygon = []
child_polygons = []
# Compare each surface with all other surfaces
for i, _polygon in enumerate(_polygons):
is_inside_any = False
for j, other_polygon in enumerate(_polygons):
if i == j:
continue
if _polygon.is_polygon_inside(other_polygon):
is_inside_any = True
break
# If the surface is inside any other surface, add it to child_surfaces
# Otherwise, it is a parent surface
if is_inside_any:
parent_polygon.append(_polygon)
else:
child_polygons.append(_polygon)
return parent_polygon, child_polygons
def _merge_hb_face_group(_faces, _tolerance, _angle_tolerance):
# type: (List[Face], float, float) -> List[Face]
"""Merge a group of HB-Faces into the fewest number of faces possible."""
if not _faces:
return []
if len(_faces) == 1:
return _faces
# -------------------------------------------------------------------------
# -- Before anything else, preserve all the Apertures for adding back in later
apertures = []
for f in _faces:
apertures.extend([ap.duplicate() for ap in f.apertures])
# -------------------------------------------------------------------------
# -- This will be the reference face for everything else to match
reference_face = _faces.pop(0).duplicate() # type: face.Face
reference_plane = copy(reference_face.geometry.plane)
polygons_in_ref_space = [
reference_face.geometry.polygon2d,
]
# -------------------------------------------------------------------------
# -- Get all the Polygon2Ds in the same reference space
poly2ds = (f.geometry.polygon2d for f in _faces)
planes = (f.geometry.plane for f in _faces)
for poly2d_plane, poly2d in zip(planes, poly2ds):
polygons_in_ref_space.append(
_get_polygon2d_in_reference_space(poly2d, poly2d_plane, reference_plane, _tolerance)
)
# -------------------------------------------------------------------------
# -- Merge all the new Polygon2Ds together.
merged_polygons = Polygon2D.boolean_union_all(polygons_in_ref_space, _tolerance)
# -------------------------------------------------------------------------
# -- Create new Face3D and HB-Faces from the Polygon2Ds
faces = []
if len(merged_polygons) == 1:
# -- Create new faces for the merged Polygon2Ds
face3ds = [_create_new_Face3D(p, reference_plane, reference_face) for p in merged_polygons]
faces = [_create_new_HB_Face(f3d, reference_face) for f3d in face3ds]
elif len(merged_polygons) > 1:
# -- It may mean that there are 'holes' in a surface? So try and find
# -- the parent and any child surfaces.
parent_polygon, child_polygons = find_parent_and_child_polygons(merged_polygons)
# -- Check the results
if len(parent_polygon) != 1:
# -- Something went wrong, give up.
_faces.append(reference_face)
return _faces
# -- If only 1 parent, lets make some Face3Ds and Faces
parent_face_3d = [_create_new_Face3D(p, reference_plane, reference_face) for p in parent_polygon]
child_face_3ds = [_create_new_Face3D(p, reference_plane, reference_face) for p in child_polygons]
face_3ds = [Face3D.from_punched_geometry(parent_face_3d[0], child_face_3ds)]
faces = [_create_new_HB_Face(f3d, reference_face) for f3d in face_3ds]
# -------------------------------------------------------------------------
# -- Add the apertures back in
faces_with_apertures_ = []
for _face in faces:
_check_and_add_sub_face(_face, apertures, _tolerance, _angle_tolerance)
faces_with_apertures_.append(_face)
return faces_with_apertures_
# -- Create new merged faces
new_faces_ = []
for face_group in _face_groups.Branches:
new_faces_.extend(
_merge_hb_face_group(
list(face_group),
sc.doc.ModelAbsoluteTolerance,
sc.doc.ModelAngleToleranceDegrees
)
)
I’m sure it could use a lot of cleaning up, but seems good enough for now.
best
@edpmay
merge_example.gh (421.2 KB)