Source code for ladybug_rhino.togeometry

"""Functions to create Ladybug geometries from Rhino geometries."""
from __future__ import division

try:
    from ladybug_geometry.geometry2d.pointvector import Vector2D, Point2D
    from ladybug_geometry.geometry2d.ray import Ray2D
    from ladybug_geometry.geometry2d.line import LineSegment2D
    from ladybug_geometry.geometry2d.polyline import Polyline2D
    from ladybug_geometry.geometry2d.polygon import Polygon2D
    from ladybug_geometry.geometry2d.mesh import Mesh2D
    from ladybug_geometry.geometry3d.pointvector import Vector3D, Point3D
    from ladybug_geometry.geometry3d.ray import Ray3D
    from ladybug_geometry.geometry3d.line import LineSegment3D
    from ladybug_geometry.geometry3d.polyline import Polyline3D
    from ladybug_geometry.geometry3d.plane import Plane
    from ladybug_geometry.geometry3d.mesh import Mesh3D
    from ladybug_geometry.geometry3d.face import Face3D
    from ladybug_geometry.geometry3d.polyface import Polyface3D
except ImportError as e:
    raise ImportError(
        "Failed to import ladybug_geometry.\n{}".format(e))

try:
    import ladybug.color as lbc
except ImportError as e:
    raise ImportError("Failed to import ladybug.\n{}".format(e))

try:
    import Rhino.Geometry as rg
except ImportError as e:
    raise ImportError("Failed to import Rhino.\n{}".format(e))

import ladybug_rhino.planarize as _planar
from .fromgeometry import from_face3ds_to_joined_brep
from .config import tolerance, angle_tolerance


"""____________2D GEOMETRY TRANSLATORS____________"""


[docs] def to_vector2d(vector): """Ladybug Vector2D from Rhino Vector3d.""" return Vector2D(vector.X, vector.Y)
[docs] def to_point2d(point): """Ladybug Point2D from Rhino Point3d.""" return Point2D(point.X, point.Y)
[docs] def to_ray2d(ray): """Ladybug Ray2D from Rhino Ray3d.""" return Ray2D(to_point2d(ray.Position), to_vector2d(ray.Direction))
[docs] def to_linesegment2d(line): """Ladybug LineSegment2D from Rhino LineCurve.""" try: # assume the likely scenario it is a LineCurve return LineSegment2D.from_end_points( to_point2d(line.PointAtStart), to_point2d(line.PointAtEnd)) except AttributeError: # likely the Line struct and not a LineCurve return LineSegment2D.from_end_points(to_point2d(line.To), to_point2d(line.From))
[docs] def to_polyline2d(polyline): """Ladybug Polyline2D from a Rhino PolyLineCurve. A LineSegment2D will be returned if the input polyline has only two points. """ if isinstance(polyline, rg.PolylineCurve): pts = [to_point2d(polyline.Point(i)) for i in range(polyline.PointCount)] elif isinstance(polyline, rg.Polyline): pts = [to_point2d(polyline[i]) for i in range(polyline.Count)] elif isinstance(polyline, rg.PolyCurve): # convert poly curve to a polyline if polyline.IsPolyline(): segs = polyline.Explode() pts = [] for seg in segs: if isinstance(seg, rg.LineCurve): pts.append(to_point2d(seg.PointAtStart)) elif isinstance(seg, rg.PolylineCurve): for i in range(seg.PointCount - 1): pts.append(to_point2d(seg.Point(i))) pts.append(to_point2d(segs[-1].PointAtEnd)) else: polyline = polyline.ToPolyline(tolerance, angle_tolerance, 0, 1e12) pts = [to_point2d(polyline.Point(i)) for i in range(polyline.PointCount)] elif isinstance(polyline, rg.LineCurve): # extract end points pts = [to_point2d(polyline.PointAtStart), to_point2d(polyline.PointAtEnd)] elif isinstance(polyline, rg.Line): # extract end points pts = [to_point2d(polyline.To), to_point2d(polyline.From)] else: msg = 'Object type {} cannot be translated to Polyline2D.'.format(type(polyline)) raise AttributeError(msg) return Polyline2D(pts) if len(pts) != 2 else LineSegment2D.from_end_points(*pts)
[docs] def to_polygon2d(polygon): """Ladybug Polygon2D from Rhino closed PolyLineCurve.""" if isinstance(polygon, rg.PolylineCurve): pts = [to_point2d(polygon.Point(i)) for i in range(polygon.PointCount)] elif isinstance(polygon, rg.Polyline): pts = [to_point2d(polygon[i]) for i in range(polygon.Count)] elif isinstance(polygon, rg.PolyCurve): # convert poly curve to a polyline if polygon.IsPolyline(): segs = polygon.Explode() pts = [] for seg in segs: if isinstance(seg, rg.LineCurve): pts.append(to_point2d(seg.PointAtStart)) elif isinstance(seg, rg.PolylineCurve): for i in range(seg.PointCount - 1): pts.append(to_point2d(seg.Point(i))) pts.append(to_point2d(segs[-1].PointAtEnd)) else: polygon = polygon.ToPolyline(tolerance, angle_tolerance, 0, 1e12) pts = [to_point2d(polygon.Point(i)) for i in range(polygon.PointCount)] else: msg = 'Object type {} cannot be translated to Polygon2D.'.format(type(polygon)) raise AttributeError(msg) if pts[0].is_equivalent(pts[-1], tolerance): # duplicated last vertex pts.pop(-1) assert polygon.IsClosed, \ 'Rhino PolyLineCurve must be closed to make a Ladybug Polygon2D.' return Polygon2D(pts)
[docs] def to_mesh2d(mesh, color_by_face=True): """Ladybug Mesh2D from Rhino Mesh.""" lb_verts = tuple(to_point2d(pt) for pt in mesh.Vertices) lb_faces, colors = _extract_mesh_faces_colors(mesh, color_by_face) return Mesh2D(lb_verts, lb_faces, colors)
"""____________3D GEOMETRY TRANSLATORS____________"""
[docs] def to_vector3d(vector): """Ladybug Vector3D from Rhino Vector3d.""" return Vector3D(vector.X, vector.Y, vector.Z)
[docs] def to_point3d(point): """Ladybug Point3D from Rhino Point3d.""" return Point3D(point.X, point.Y, point.Z)
[docs] def to_ray3d(ray): """Ladybug Ray3D from Rhino Ray3d.""" return Ray3D(to_point3d(ray.Position), to_vector3d(ray.Direction))
[docs] def to_linesegment3d(line): """Ladybug LineSegment3D from Rhino LineCurve.""" try: # assume the likely scenario it is a LineCurve return LineSegment3D.from_end_points( to_point3d(line.PointAtStart), to_point3d(line.PointAtEnd)) except AttributeError: # likely the Line struct and not a LineCurve return LineSegment3D.from_end_points(to_point3d(line.To), to_point3d(line.From))
[docs] def to_polyline3d(polyline): """Ladybug Polyline3D from a Rhino PolyLineCurve. A LineSegment3D will be returned if the input polyline has only two points. """ if isinstance(polyline, rg.PolylineCurve): pts = [to_point3d(polyline.Point(i)) for i in range(polyline.PointCount)] elif isinstance(polyline, rg.Polyline): pts = [to_point3d(polyline[i]) for i in range(polyline.Count)] elif isinstance(polyline, rg.PolyCurve): # convert poly curve to a polyline if polyline.IsPolyline(): segs = polyline.Explode() pts = [] for seg in segs: if isinstance(seg, rg.LineCurve): pts.append(to_point3d(seg.PointAtStart)) elif isinstance(seg, rg.PolylineCurve): for i in range(seg.PointCount - 1): pts.append(to_point3d(seg.Point(i))) pts.append(to_point3d(segs[-1].PointAtEnd)) else: polyline = polyline.ToPolyline(tolerance, angle_tolerance, 0, 1e12) pts = [to_point3d(polyline.Point(i)) for i in range(polyline.PointCount)] elif isinstance(polyline, rg.LineCurve): # extract end points pts = [to_point3d(polyline.PointAtStart), to_point3d(polyline.PointAtEnd)] elif isinstance(polyline, rg.Line): # extract end points pts = [to_point3d(polyline.To), to_point3d(polyline.From)] else: msg = 'Object type {} cannot be translated to Polyline3D.'.format(type(polyline)) raise AttributeError(msg) return Polyline3D(pts) if len(pts) != 2 else LineSegment3D.from_end_points(*pts)
[docs] def to_plane(pl): """Ladybug Plane from Rhino Plane.""" return Plane( to_vector3d(pl.ZAxis), to_point3d(pl.Origin), to_vector3d(pl.XAxis))
[docs] def to_face3d(geo, meshing_parameters=None): """List of Ladybug Face3D objects from a Rhino Brep, Surface or Mesh. Args: geo: A Rhino Brep, Surface, Extrusion or Mesh that will be converted into a list of Ladybug Face3D. meshing_parameters: Optional Rhino Meshing Parameters to describe how curved faces should be converted into planar elements. If None, Rhino's Default Meshing Parameters will be used. """ faces = [] # list of Face3Ds to be populated and returned if isinstance(geo, rg.Mesh): # convert each Mesh face to a Face3D pts = tuple(to_point3d(pt) for pt in geo.Vertices) for face in geo.Faces: if face.IsQuad: all_verts = (pts[face[0]], pts[face[1]], pts[face[2]], pts[face[3]]) lb_face = Face3D(all_verts) if lb_face.area != 0: for _v in lb_face.vertices: if lb_face.plane.distance_to_point(_v) >= tolerance: # non-planar quad split the quad into two planar triangles verts1 = (pts[face[0]], pts[face[1]], pts[face[2]]) verts2 = (pts[face[3]], pts[face[0]], pts[face[1]]) faces.append(Face3D(verts1)) faces.append(Face3D(verts2)) break else: faces.append(lb_face) else: all_verts = (pts[face[0]], pts[face[1]], pts[face[2]]) lb_face = Face3D(all_verts) if lb_face.area != 0: faces.append(lb_face) else: # convert each Brep Face to a Face3D meshing_parameters = meshing_parameters or rg.MeshingParameters.Default if not isinstance(geo, rg.Brep): # it's likely an extrusion object geo = geo.ToBrep() # extrusion objects must be cast to Brep in Rhino 8 for b_face in geo.Faces: if b_face.IsPlanar(tolerance): try: bf_plane = to_plane(b_face.FrameAt(0, 0)[-1]) except Exception: # failed to extract the plane from the geometry bf_plane = None # auto-calculate the plane from the vertices all_verts = [] for count in range(b_face.Loops.Count): # Each loop is a boundary/hole success, loop_pline = \ b_face.Loops[count].To3dCurve().TryGetPolyline() if not success: # Failed to get a polyline; there's a curved edge loop_verts = _planar.planar_face_curved_edge_vertices( b_face, count, meshing_parameters) else: # we have a polyline representing the loop loop_verts = tuple(to_point3d(loop_pline[i]) for i in range(loop_pline.Count - 1)) all_verts.append(_remove_dup_verts(loop_verts)) if len(all_verts[0]) >= 3: if len(all_verts) == 1: # No holes in the shape faces.append(Face3D(all_verts[0], plane=bf_plane)) else: # There's at least one hole in the shape hls = [hl for hl in all_verts[1:] if len(hl) >= 3] faces.append(Face3D( boundary=all_verts[0], holes=hls, plane=bf_plane)) else: # curved face must be meshed into planar Face3D objects faces.extend(_planar.curved_surface_faces(b_face, meshing_parameters)) return faces
[docs] def to_polyface3d(geo, meshing_parameters=None): """A Ladybug Polyface3D object from a Rhino Brep. Args: geo: A Rhino Brep, Surface or Mesh that will be converted into a single Ladybug Polyface3D. meshing_parameters: Optional Rhino Meshing Parameters to describe how curved faces should be converted into planar elements. If None, Rhino's Default Meshing Parameters will be used. """ mesh_par = meshing_parameters or rg.MeshingParameters.Default # default if not isinstance(geo, rg.Mesh): if not isinstance(geo, rg.Brep): # it's likely an extrusion object geo = geo.ToBrep() # extrusion objects must be cast to Brep in Rhino 8 if _planar.has_curved_face(geo): # keep solidity new_brep = from_face3ds_to_joined_brep( _planar.curved_solid_faces(geo, mesh_par)) return Polyface3D.from_faces(to_face3d(new_brep[0], mesh_par), tolerance) return Polyface3D.from_faces(to_face3d(geo, mesh_par), tolerance) return Polyface3D.from_faces(to_face3d(geo, mesh_par), tolerance)
[docs] def to_mesh3d(mesh, color_by_face=True): """Ladybug Mesh3D from Rhino Mesh.""" lb_verts = tuple(to_point3d(pt) for pt in mesh.Vertices) lb_faces, colors = _extract_mesh_faces_colors(mesh, color_by_face) return Mesh3D(lb_verts, lb_faces, colors)
"""________ADDITIONAL 3D GEOMETRY TRANSLATORS________"""
[docs] def to_gridded_mesh3d(brep, grid_size, offset_distance=0): """Create a gridded Ladybug Mesh3D from a Rhino Brep. This is useful since Rhino's grid meshing is often more beautiful than what ladybug_geometry can produce. However, the ladybug_geometry Face3D.get_mesh_grid method provides a workable alternative to this if it is needed. Args: brep: A Rhino Brep that will be converted into a gridded Ladybug Mesh3D. grid_size: A number for the grid size dimension with which to make the mesh. offset_distance: A number for the distance at which to offset the mesh from the underlying brep. The default is 0. """ if not isinstance(brep, rg.Brep): # it's likely an extrusion object brep = brep.ToBrep() # extrusion objects must be cast to Brep in Rhino 8 meshing_param = rg.MeshingParameters.Default meshing_param.MaximumEdgeLength = grid_size meshing_param.MinimumEdgeLength = grid_size meshing_param.GridAspectRatio = 1 mesh_grids = rg.Mesh.CreateFromBrep(brep, meshing_param) if len(mesh_grids) == 1: # only one mesh was generated mesh_grid = mesh_grids[0] else: # join the meshes into one mesh_grid = rg.Mesh() for m_grid in mesh_grids: mesh_grid.Append(m_grid) if offset_distance != 0: temp_mesh = rg.Mesh() mesh_grid.Normals.UnitizeNormals() for pt, vec in zip(mesh_grid.Vertices, mesh_grid.Normals): temp_mesh.Vertices.Add(pt + (rg.Vector3f.Multiply(vec, offset_distance))) for face in mesh_grid.Faces: temp_mesh.Faces.AddFace(face) mesh_grid = temp_mesh return to_mesh3d(mesh_grid)
[docs] def to_joined_gridded_mesh3d(geometry, grid_size, offset_distance=0): """Create a single gridded Ladybug Mesh3D from an array of Rhino geometry. Args: breps: An array of Rhino Breps and/or Rhino meshes that will be converted into a single, joined gridded Ladybug Mesh3D. grid_size: A number for the grid size dimension with which to make the mesh. offset_distance: A number for the distance at which to offset the mesh from the underlying brep. The default is 0. """ lb_meshes = [] for geo in geometry: if isinstance(geo, rg.Mesh): lb_meshes.append(to_mesh3d(geo)) else: # assume that it's a Brep lb_meshes.append(to_gridded_mesh3d(geo, grid_size, offset_distance)) if len(lb_meshes) == 1: return lb_meshes[0] else: return Mesh3D.join_meshes(lb_meshes)
"""________________EXTRA HELPER FUNCTIONS________________""" def _extract_mesh_faces_colors(mesh, color_by_face): """Extract face indices and colors from a Rhino mesh.""" colors = None lb_faces = [] for face in mesh.Faces: if face.IsQuad: lb_faces.append((face[0], face[1], face[2], face[3])) else: lb_faces.append((face[0], face[1], face[2])) if len(mesh.VertexColors) != 0: colors = [] if color_by_face is True: for face in mesh.Faces: col = mesh.VertexColors[face[0]] colors.append(lbc.Color(col.R, col.G, col.B)) else: for col in mesh.VertexColors: colors.append(lbc.Color(col.R, col.G, col.B)) return lb_faces, colors def _remove_dup_verts(vertices): """Remove vertices from an array of Point3Ds that are equal within the tolerance.""" return [pt for i, pt in enumerate(vertices) if not pt.is_equivalent(vertices[i - 1], tolerance)]