Source code for ladybug_rhino.preview

"""Classes to preview things in the Rhino Display Pipeline."""
from __future__ import division

from ladybug_geometry.geometry2d import Vector2D, Point2D, Ray2D, LineSegment2D, \
    Polyline2D, Arc2D, Polygon2D, Mesh2D
from ladybug_geometry.geometry3d import Vector3D, Point3D, Ray3D, Plane, LineSegment3D, \
    Polyline3D, Arc3D, Face3D, Mesh3D, Polyface3D, Sphere, Cone, Cylinder

from ladybug.graphic import GraphicContainer
from ladybug_display.altnumber import Default
from ladybug_display.geometry3d import DisplayText3D
from ladybug_display.visualization import AnalysisGeometry

from .config import units_system, rhino_version
from .color import color_to_color, argb_color_to_color, black
from .fromgeometry import from_point2d, from_vector2d, from_ray2d, \
    from_arc2d, from_polygon2d, from_polyline2d, from_mesh2d, \
    from_point3d, from_vector3d, from_ray3d, from_plane, \
    from_arc3d, from_polyline3d, from_mesh3d, from_face3d, from_polyface3d, \
    from_sphere, from_cone, from_cylinder

try:
    import System
except ImportError as e:
    raise ImportError("Failed to import Windows/.NET libraries\n{}".format(e))

try:
    import Rhino.Geometry as rg
    import Rhino.Display as rd
    import Rhino.DocObjects as ro
except ImportError as e:
    raise ImportError('Failed to import Rhino.\n{}'.format(e))

try:
    import scriptcontext as sc
except ImportError:
    pass  # previewing 2D legends will not be available


[docs] class VisualizationSetConduit(rd.DisplayConduit): """Class to preview VisualizationSet in the Rhino Display pipeline. Args: visualization_set: A Ladybug Display VisualizationSet object to be translated into arguments for the Rhino display pipeline. render_3d_legend: A Boolean to note whether the VisualizationSet should be rendered with 3D legends for any AnalysisGeometries it includes. (Default: False). render_2d_legend: A Boolean to note whether the VisualizationSet should be rendered with 2D screen-oriented legends for any AnalysisGeometries it includes. (Default: False). """ def __init__(self, visualization_set, render_3d_legend=False, render_2d_legend=False): """Initialize VisualizationSetConduit.""" # set the primary properties self.vis_con = VisualizationSetConverter( visualization_set, render_3d_legend, render_2d_legend)
[docs] def CalculateBoundingBox(self, calculateBoundingBoxEventArgs): """Overwrite the method that passes the bounding box to the display.""" calculateBoundingBoxEventArgs.IncludeBoundingBox(self.vis_con.bbox)
[docs] def DrawForeground(self, drawEventArgs): """Overwrite the method that draws the objects in the display.""" # get the DisplayPipeline from the event arguments display = drawEventArgs.Display # for each object to be rendered, pass the drawing arguments for draw_args in self.vis_con.draw_2d_text: display.Draw2dText(*draw_args) for draw_args in self.vis_con.draw_sprite: display.DrawSprite(*draw_args)
[docs] def PreDrawObjects(self, drawEventArgs): """Overwrite the method that draws the objects in the display.""" # get the DisplayPipeline from the event arguments display = drawEventArgs.Display # for each object to be rendered, pass the drawing arguments for draw_args in self.vis_con.draw_mesh_false_colors: display.DrawMeshFalseColors(draw_args) for draw_args in self.vis_con.draw_mesh_shaded: display.DrawMeshFalseColors(draw_args[0]) for draw_args in self.vis_con.draw_brep_shaded: display.DrawBrepShaded(*draw_args)
[docs] def PostDrawObjects(self, drawEventArgs): """Overwrite the method that draws the objects in the display.""" # get the DisplayPipeline from the event arguments display = drawEventArgs.Display # for each object to be rendered, pass the drawing arguments for draw_args in self.vis_con.draw_3d_text: display.Draw3dText(*draw_args) for draw_args in self.vis_con.draw_mesh_wires: display.DrawMeshWires(*draw_args) for draw_args in self.vis_con.draw_mesh_vertices: display.DrawMeshVertices(*draw_args) for draw_args in self.vis_con.draw_point: display.DrawPoint(*draw_args) for draw_args in self.vis_con.draw_arrow: display.DrawArrow(*draw_args) for draw_args in self.vis_con.draw_brep_wires: display.DrawBrepWires(*draw_args) for draw_args in self.vis_con.draw_line: display.DrawLine(*draw_args) for draw_args in self.vis_con.draw_patterned_line: display.DrawPatternedLine(*draw_args) for draw_args in self.vis_con.draw_patterned_polyline: display.DrawPatternedPolyline(*draw_args) for draw_args in self.vis_con.draw_curve: display.DrawCurve(*draw_args) for draw_args in self.vis_con.draw_circle: display.DrawCircle(*draw_args) for draw_args in self.vis_con.draw_arc: display.DrawArc(*draw_args) for draw_args in self.vis_con.draw_sphere: display.DrawSphere(*draw_args) for draw_args in self.vis_con.draw_cone: display.DrawCone(*draw_args) for draw_args in self.vis_con.draw_cylinder: display.DrawCylinder(*draw_args)
[docs] class VisualizationSetConverter(object): """Class to translate VisualizationSets to arguments for the Rhino display pipeline. Args: visualization_set: A Ladybug Display VisualizationSet object to be translated into arguments for the Rhino display pipeline. render_3d_legend: A Boolean to note whether the VisualizationSet should be rendered with 3D legends for any AnalysisGeometries it includes. (Default: False). render_2d_legend: A Boolean to note whether the VisualizationSet should be rendered with 2D screen-oriented legends for any AnalysisGeometries it includes. (Default: False). Properties: * vis_set * render_3d_legend * render_2d_legend * min_pt * max_pt * bbox * draw_sprite * draw_2d_text * draw_3d_text * draw_mesh_false_colors * draw_mesh_shaded * draw_mesh_wires * draw_mesh_vertices * draw_brep_shaded * draw_brep_wires * draw_point * draw_arrow * draw_line * draw_patterned_line * draw_patterned_polyline * draw_curve * draw_circle * draw_arc * draw_sphere * draw_cone * draw_cylinder """ TEXT_HORIZ = { 'Left': ro.TextHorizontalAlignment.Left, 'Center': ro.TextHorizontalAlignment.Center, 'Right': ro.TextHorizontalAlignment.Right } TEXT_VERT = { 'Top': ro.TextVerticalAlignment.Top, 'Middle': ro.TextVerticalAlignment.Middle, 'Bottom': ro.TextVerticalAlignment.Bottom } LINE_TYPE = { 'Dashed': int('0x1C7', base=16), 'Dotted': int('0x00001111', base=16), 'DashDot': int('0x1F11F1', base=16) } def __init__( self, visualization_set, render_3d_legend=False, render_2d_legend=False): """Initialize VisualizationSetConverter.""" # set the primary properties self.vis_set = visualization_set self.render_3d_legend = bool(render_3d_legend) self.render_2d_legend = bool(render_2d_legend) # get the active viewport width and height if 2D legends are requested if self.render_2d_legend: active_view = sc.doc.Views.ActiveView.ActiveViewport v_size = active_view.Size self.view_width = v_size.Width self.view_height = v_size.Height else: # just use some defaults as it's probably not critical self.view_width, self.view_height = 800, 600 # ensure the visualization set is in Rhino model units units_sys = units_system() if self.vis_set.units is not None and units_sys is not None \ and self.vis_set.units != units_sys: self.vis_set.convert_to_units(units_sys) # set up the bounding box and min/max point self.min_pt = self.vis_set.min_point self.max_pt = self.vis_set.max_point if self.render_3d_legend: # leave extra room for legend center = Point3D( (self.min_pt.x + self.max_pt.x) / 2, (self.min_pt.y + self.max_pt.y) / 2, (self.min_pt.z + self.max_pt.z) / 2) self.bbox = rg.BoundingBox( from_point3d(self.min_pt.scale(2, center)), from_point3d(self.max_pt.scale(2, center))) else: self.bbox = rg.BoundingBox( from_point3d(self.min_pt), from_point3d(self.max_pt)) # translate all of the rhino objects to be rendered in the scene self.translate_objects() self.sort_shaded_objects()
[docs] def translate_objects(self): """Translate all of the objects in the VisualizationSet into a Rhino equivalent. This is a necessary pre-step before frames of the visualization set can be rendered in the display pipeline. """ # initialize all of the lists to hold the drawing arguments self.draw_sprite = [] self.draw_2d_text = [] self.draw_3d_text = [] self.draw_mesh_false_colors = [] self.draw_mesh_shaded = [] self.draw_mesh_wires = [] self.draw_mesh_vertices = [] self.draw_brep_shaded = [] self.draw_brep_wires = [] self.draw_point = [] self.draw_arrow = [] self.draw_line = [] self.draw_patterned_line = [] self.draw_patterned_polyline = [] self.draw_curve = [] self.draw_circle = [] self.draw_arc = [] self.draw_sphere = [] self.draw_cone = [] self.draw_cylinder = [] # loop through visualization geometry objects and draw them default_leg_x3d, default_leg_x2d, default_leg_y2d = 0, 10, 50 for geo in self.vis_set.geometry: if geo.hidden: continue # translate it as AnalysisGeometry if specified if isinstance(geo, AnalysisGeometry): # generate the colors that correspond to the values data = geo.data_sets[geo.active_data] graphic = GraphicContainer( data.values, self.min_pt, self.max_pt, data.legend_parameters, data.data_type, data.unit) colors = graphic.value_colors # translate the analysis geometry using the matching method if geo.matching_method == 'faces': c_count = 0 for mesh in geo.geometry: mesh.colors = colors[c_count:c_count + len(mesh.faces)] c_count += len(mesh.faces) self.translate_analysis_mesh(mesh, geo.display_mode) elif geo.matching_method == 'vertices': c_count = 0 for mesh in geo.geometry: mesh.colors = colors[c_count:c_count + len(mesh.vertices)] c_count += len(mesh.vertices) self.translate_analysis_mesh(mesh, geo.display_mode) else: # one color per geometry object for geo_obj, col in zip(geo.geometry, colors): self.translate_analysis_geometry( geo_obj, col, geo.display_mode) # if the object is set to translate 3D legends, then display it if self.render_3d_legend: # ensure multiple legends are not on top of each other if graphic.legend_parameters.is_base_plane_default: l_par = graphic.legend_parameters m_vec = Vector3D(default_leg_x3d, 0, 0) l_par.base_plane = l_par.base_plane.move(m_vec) l_par.properties_3d._is_base_plane_default = True leg_width = l_par.segment_width + 6 * l_par.text_height \ if l_par.vertical else \ l_par.segment_width * (l_par.segment_count + 2) default_leg_x3d += leg_width self.translate_legend3d(graphic.legend) # if the object is set to translate 2D legends, then display it if self.render_2d_legend: # ensure multiple legends are not on top of each other l_par = graphic.legend_parameters if l_par.vertical and l_par.is_origin_x_default: l_par.origin_x = '{}px'.format(default_leg_x2d) l_par.properties_2d._is_origin_x_default = True sw = graphic.legend.parse_dim_2d( l_par.segment_width_2d, self.view_width) th = graphic.legend.parse_dim_2d( l_par.text_height_2d, self.view_height) leg_width = sw + 6 * th default_leg_x2d += leg_width elif not l_par.vertical and l_par.is_origin_y_default: l_par.origin_y = '{}px'.format(default_leg_y2d) l_par.properties_2d._is_origin_y_default = True sh = graphic.legend.parse_dim_2d( l_par.segment_height_2d, self.view_height) th = graphic.legend.parse_dim_2d( l_par.text_height_2d, self.view_height) leg_height = sh + 4 * th default_leg_y2d += leg_height self.translate_legend2d(graphic.legend) else: # it's a ContextGeometry object for display_obj in geo.geometry: if isinstance(display_obj, DisplayText3D): self.translate_display_text3d(display_obj) else: self.translate_context_geometry(display_obj)
[docs] def translate_display_text3d(self, text_obj): """Translate ladybug DisplayText3D into arguments for drawing in the scene. Args: text_obj: The ladybug DisplayText3D to be translated. """ self.draw_3d_text.append(( text_obj.text, color_to_color(text_obj.color), from_plane(text_obj.plane), text_obj.height, text_obj.font, False, False, self.TEXT_HORIZ[text_obj.horizontal_alignment], self.TEXT_VERT[text_obj.vertical_alignment]))
[docs] def translate_legend3d(self, legend): """Translate a ladybug Legend into arguments for drawing in the scene. Args: legend: A Ladybug Legend object to be converted to Rhino geometry. """ # translate the legend mesh rh_mesh = from_mesh3d(legend.segment_mesh) self.draw_mesh_false_colors.append(rh_mesh) self.draw_mesh_wires.append((rh_mesh, black())) # translate the legend text _height = legend.legend_parameters.text_height _font = legend.legend_parameters.font if legend.legend_parameters.continuous_legend is False: legend_text = [ DisplayText3D(txt, loc, _height, None, _font, 'Left', 'Bottom') for txt, loc in zip(legend.segment_text, legend.segment_text_location)] elif legend.legend_parameters.vertical is True: legend_text = [ DisplayText3D(txt, loc, _height, None, _font, 'Left', 'Middle') for txt, loc in zip(legend.segment_text, legend.segment_text_location)] else: legend_text = [ DisplayText3D(txt, loc, _height, None, _font, 'Center', 'Bottom') for txt, loc in zip(legend.segment_text, legend.segment_text_location)] legend_title = DisplayText3D( legend.title, legend.title_location, _height, None, _font) legend_text.insert(0, legend_title) for txt_obj in legend_text: self.translate_display_text3d(txt_obj)
[docs] def translate_legend2d(self, legend): """Translate a ladybug Legend into arguments for drawing in the scene. Args: legend: A Ladybug Legend object to be converted to 2D screen-oriented geometry. """ d_sprite, d_2d_text = self.convert_legend2d( legend, self.view_width, self.view_height) self.draw_sprite.append(d_sprite) self.draw_2d_text.extend(d_2d_text)
[docs] def translate_analysis_mesh(self, mesh, display_mode): """Translate an analysis mesh into arguments for drawing in the scene. Args: mesh: The Ladybug Mesh3D or Mesh2D object to be translated. display_mode: Text for the display mode in which to translate the mesh. """ if display_mode in ('Surface', 'SurfaceWithEdges'): rh_mesh = from_mesh3d(mesh) if isinstance(mesh, Mesh3D) \ else from_mesh2d(mesh) self.draw_mesh_false_colors.append(rh_mesh) if display_mode == 'SurfaceWithEdges': self.draw_mesh_wires.append((rh_mesh, black())) elif display_mode == 'Wireframe': rh_mesh = from_mesh3d(mesh) if isinstance(mesh, Mesh3D) \ else from_mesh2d(mesh) self.draw_mesh_wires.append((rh_mesh, black())) elif display_mode == 'Points': if isinstance(mesh, Mesh3D): points = [from_point3d(pt) for pt in mesh.face_centroids] \ if mesh.is_color_by_face else \ [from_point3d(pt) for pt in mesh.vertices] else: points = [from_point2d(pt) for pt in mesh.face_centroids] \ if mesh.is_color_by_face else \ [from_point2d(pt) for pt in mesh.vertices] colors = [color_to_color(col) for col in mesh.colors] for col, pt in zip(colors, points): self.draw_point.append((pt, rd.PointStyle.RoundSimple, 3, col))
[docs] def translate_analysis_geometry(self, geo_obj, color, display_mode): """Translate analysis geometry objects into arguments for drawing in the scene. Args: geo_obj: The Ladybug Geometry object to be translated. color: The Ladybug Color object with which the object will be translated. display_mode: Text for the display mode in which to translate the mesh. """ # translate the color col = color_to_color(color) if isinstance(geo_obj, (Point3D, Point2D)): # translate analysis point pt = from_point3d(geo_obj) if isinstance(geo_obj, Point3D) \ else from_point2d(geo_obj) self.draw_point.append((pt, rd.PointStyle.RoundSimple, 5, col)) elif isinstance(geo_obj, (LineSegment3D, LineSegment2D)): # translate analysis line segment if isinstance(geo_obj, LineSegment3D): pt1, pt2 = from_point3d(geo_obj.p1), from_point3d(geo_obj.p2) else: pt1, pt2 = from_point2d(geo_obj.p1), from_point2d(geo_obj.p2) self.draw_line.append((pt1, pt2, col, 3)) elif isinstance(geo_obj, (Polyline3D, Polyline2D)): # translate analysis polyline crv = from_polyline3d(geo_obj) if isinstance(geo_obj, Polyline3D) \ else from_polyline2d(geo_obj) self.draw_curve.append((crv, col, 3)) elif isinstance(geo_obj, (Face3D, Polyface3D)): # translate analysis Face3D and Polyface3D if display_mode in ('Surface', 'SurfaceWithEdges'): mat = rd.DisplayMaterial(col) rh_obj = from_face3d(geo_obj) if isinstance(geo_obj, Face3D) \ else from_polyface3d(geo_obj) self.draw_brep_shaded.append((rh_obj, mat)) if display_mode == 'SurfaceWithEdges': self.draw_brep_wires.append((rh_obj, black(), -1)) elif display_mode == 'Wireframe': rh_obj = from_face3d(geo_obj) if isinstance(geo_obj, Face3D) \ else from_polyface3d(geo_obj) self.draw_brep_wires.append((rh_obj, col, 1)) elif display_mode == 'Points': for pt in geo_obj.vertices: self.draw_point.append( (from_point3d(pt), rd.PointStyle.RoundSimple, 3, col)) elif isinstance(geo_obj, (Mesh3D, Mesh2D)): # translate analysis mesh rh_obj = from_mesh3d(geo_obj) if isinstance(geo_obj, Mesh3D) \ else from_mesh2d(geo_obj) if display_mode in ('Surface', 'SurfaceWithEdges'): mat = rd.DisplayMaterial(col) rh_obj.VertexColors.CreateMonotoneMesh(col) self.draw_mesh_shaded.append((rh_obj, mat)) if display_mode == 'SurfaceWithEdges': self.draw_mesh_wires.append((rh_obj, black(), 1)) elif display_mode == 'Wireframe': self.draw_mesh_wires.append((rh_obj, col, 3)) elif display_mode == 'Points': self.draw_mesh_vertices.append((rh_obj, col)) elif isinstance(geo_obj, (Ray2D, Ray3D)): # translate analysis rays rh_ray = from_ray3d(geo_obj) if isinstance(geo_obj, Ray3D) \ else from_ray2d(geo_obj) rh_obj = rg.Line(rh_ray.Position, rh_ray.Direction) self.draw_arrow.append((rh_obj, col)) elif isinstance(geo_obj, Sphere): # translate analysis sphere if display_mode in ('Surface', 'SurfaceWithEdges'): rh_obj, mat = from_sphere(geo_obj), rd.DisplayMaterial(col) self.draw_brep_shaded.append((rh_obj.ToBrep(), mat)) if display_mode == 'SurfaceWithEdges': self.draw_sphere.append((rh_obj, black())) elif display_mode == 'Wireframe': self.draw_sphere.append((from_sphere(geo_obj), col)) elif display_mode == 'Points': pt_arg = (from_point3d(geo_obj.center), rd.PointStyle.RoundSimple, 3, col) self.draw_point.append(pt_arg) elif isinstance(geo_obj, (Vector2D, Vector3D)): # translate analysis vectors rh_vec = from_vector3d(geo_obj) if isinstance(geo_obj, Vector3D) \ else from_vector2d(geo_obj) rh_obj = rg.Line(rg.Point3d(0, 0, 0), rh_vec) self.draw_arrow.append((rh_obj, col)) elif isinstance(geo_obj, Polygon2D): # translate analysis polygon self.draw_curve.append((from_polygon2d(geo_obj), col, 3)) elif isinstance(geo_obj, (Arc3D, Arc2D)): # translate analysis arcs crv = from_arc3d(geo_obj) if isinstance(geo_obj, Arc3D) \ else from_arc2d(geo_obj) if geo_obj.is_circle: self.draw_circle.append((crv, col, 3)) else: self.draw_arc.append((crv, col, 3)) elif isinstance(geo_obj, Cone): # translate analysis cone if display_mode in ('Surface', 'SurfaceWithEdges'): rh_obj, mat = from_cone(geo_obj), rd.DisplayMaterial(col) self.draw_brep_shaded.append((rh_obj.ToBrep(True), mat)) if display_mode == 'SurfaceWithEdges': self.draw_cone.append((rh_obj, black())) elif display_mode == 'Wireframe': self.draw_cone.append((from_cone(geo_obj), col)) elif display_mode == 'Points': pt_arg = (from_point3d(geo_obj.vertex), rd.PointStyle.RoundSimple, 3, col) self.draw_point.append(pt_arg) elif isinstance(geo_obj, Cylinder): # translate analysis cylinder if display_mode in ('Surface', 'SurfaceWithEdges'): rh_obj, mat = from_cylinder(geo_obj), rd.DisplayMaterial(col) self.draw_brep_shaded.append((rh_obj.ToBrep(True, True), mat)) if display_mode == 'SurfaceWithEdges': self.draw_cylinder.append((rh_obj, black())) elif display_mode == 'Wireframe': self.draw_cylinder.append((from_cylinder(geo_obj), col)) elif display_mode == 'Points': pt_arg = (from_point3d(geo_obj.center), rd.PointStyle.RoundSimple, 3, col) self.draw_point.append(pt_arg) elif isinstance(geo_obj, Plane): # translate analysis planes pln = from_plane(geo_obj) r = 10 # default radius for a plane object in rhino model units # grid lines for j in range(-5, 6): i = j / 5 p0 = pln.PointAt(i * r, -r) p1 = pln.PointAt(i * r, r) self.draw_line.append((p0, p1, col, 1)) p0 = pln.PointAt(-r, i * r) p1 = pln.PointAt(r, i * r) self.draw_line.append((p0, p1, col, 1)) # axes self.draw_line.append((pln.Origin, pln.Origin + pln.XAxis * r, col, 3)) self.draw_line.append((pln.Origin, pln.Origin + pln.YAxis * r, col, 3))
[docs] def translate_context_geometry(self, dis_obj): """Translate a display geometry object into arguments for drawing in the scene. Args: dis_obj: The Ladybug Display geometry object to be translated. """ # first translate the color and get the geometry object col = color_to_color(dis_obj.color) geo_obj = dis_obj.geometry if isinstance(geo_obj, (Point3D, Point2D)): # translate display point pt = from_point3d(geo_obj) if isinstance(geo_obj, Point3D) \ else from_point2d(geo_obj) radius = 3 if isinstance(dis_obj.radius, Default) else int(dis_obj.radius) self.draw_point.append((pt, rd.PointStyle.RoundSimple, radius, col)) elif isinstance(geo_obj, (LineSegment3D, LineSegment2D)): # translate display line segment if isinstance(geo_obj, LineSegment3D): pt1, pt2 = from_point3d(geo_obj.p1), from_point3d(geo_obj.p2) else: pt1, pt2 = from_point2d(geo_obj.p1), from_point2d(geo_obj.p2) lw = 1 if isinstance(dis_obj.line_width, Default) \ else int(dis_obj.line_width) if dis_obj.line_type == 'Continuous': self.draw_line.append((pt1, pt2, col, lw)) else: self.draw_patterned_line.append( (pt1, pt2, col, self.LINE_TYPE[dis_obj.line_type], lw)) elif isinstance(geo_obj, (Polyline3D, Polyline2D)): # translate display polyline lw = 1 if isinstance(dis_obj.line_width, Default) \ else int(dis_obj.line_width) if dis_obj.line_type == 'Continuous': crv = from_polyline3d(geo_obj) if isinstance(geo_obj, Polyline3D) \ else from_polyline2d(geo_obj) self.draw_curve.append((crv, col, lw)) else: # ensure the line pattern is respected verts = [from_point3d(pt) for pt in geo_obj.vertices] \ if isinstance(geo_obj, Polyline3D) else \ [from_point2d(pt) for pt in geo_obj.vertices] self.draw_patterned_polyline.append( (verts, col, self.LINE_TYPE[dis_obj.line_type], lw, False)) elif isinstance(geo_obj, (Ray2D, Ray3D)): # translate display rays rh_ray = from_ray3d(geo_obj) if isinstance(geo_obj, Ray3D) \ else from_ray2d(geo_obj) rh_obj = rg.Line(rh_ray.Position, rh_ray.Direction) self.draw_arrow.append((rh_obj, col)) elif isinstance(geo_obj, (Face3D, Polyface3D)): # translate display Face3D and Polyface3D if dis_obj.display_mode in ('Surface', 'SurfaceWithEdges'): mat = rd.DisplayMaterial(col, 1 - (dis_obj.color.a / 255)) rh_obj = from_face3d(geo_obj) if isinstance(geo_obj, Face3D) \ else from_polyface3d(geo_obj) self.draw_brep_shaded.append((rh_obj, mat)) if dis_obj.display_mode == 'SurfaceWithEdges': self.draw_brep_wires.append((rh_obj, black(), -1)) elif dis_obj.display_mode == 'Wireframe': rh_obj = from_face3d(geo_obj) if isinstance(geo_obj, Face3D) \ else from_polyface3d(geo_obj) self.draw_brep_wires.append((rh_obj, col, 1)) elif dis_obj.display_mode == 'Points': for pt in geo_obj.vertices: self.draw_point.append( (from_point3d(pt), rd.PointStyle.RoundSimple, 3, col)) elif isinstance(geo_obj, (Mesh3D, Mesh2D)): # translate display mesh rh_obj = from_mesh3d(geo_obj) if isinstance(geo_obj, Mesh3D) \ else from_mesh2d(geo_obj) if dis_obj.display_mode in ('Surface', 'SurfaceWithEdges'): mat = rd.DisplayMaterial(col, 1 - (dis_obj.color.a / 255)) t_color = argb_color_to_color(dis_obj.color) rh_obj.VertexColors.CreateMonotoneMesh(t_color) self.draw_mesh_shaded.append((rh_obj, mat)) if dis_obj.display_mode == 'SurfaceWithEdges': self.draw_mesh_wires.append((rh_obj, black(), 1)) elif dis_obj.display_mode == 'Wireframe': self.draw_mesh_wires.append((rh_obj, col, 1)) elif dis_obj.display_mode == 'Points': self.draw_mesh_vertices.append((rh_obj, col)) elif isinstance(geo_obj, Sphere): # translate analysis sphere if dis_obj.display_mode in ('Surface', 'SurfaceWithEdges'): rh_obj = from_sphere(geo_obj) mat = rd.DisplayMaterial(col, 1 - (dis_obj.color.a / 255)) self.draw_brep_shaded.append((rh_obj.ToBrep(), mat)) if dis_obj.display_mode == 'SurfaceWithEdges': self.draw_sphere.append((rh_obj, black())) elif dis_obj.display_mode == 'Wireframe': self.draw_sphere.append((from_sphere(geo_obj), col)) elif dis_obj.display_mode == 'Points': pt_arg = (from_point3d(geo_obj.center), rd.PointStyle.RoundSimple, 3, col) self.draw_point.append(pt_arg) elif isinstance(geo_obj, Polygon2D): # translate display polygon lw = 1 if isinstance(dis_obj.line_width, Default) \ else int(dis_obj.line_width) if dis_obj.line_type == 'Continuous': self.draw_curve.append((from_polygon2d(geo_obj), col, lw)) else: # ensure the line pattern is respected verts = [from_point2d(pt) for pt in geo_obj.vertices] self.draw_patterned_polyline.append( (verts, col, self.LINE_TYPE[dis_obj.line_type], lw, True)) elif isinstance(geo_obj, (Arc3D, Arc2D)): # translate display arc lw = 1 if isinstance(dis_obj.line_width, Default) \ else int(dis_obj.line_width) if dis_obj.line_type == 'Continuous': crv = from_arc3d(geo_obj) if isinstance(geo_obj, Arc3D) else \ from_arc2d(geo_obj) if geo_obj.is_circle: self.draw_circle.append((crv, col, lw)) else: self.draw_arc.append((crv, col, lw)) else: # ensure the line pattern is respected p_line = geo_obj.to_polyline( int(abs(geo_obj.a2 - geo_obj.a1) / 0.0523599)) if geo_obj.is_circle: arc_verts, closed = p_line.vertices[:-1], True else: arc_verts, closed = p_line.vertices, False verts = [from_point3d(pt) for pt in arc_verts] \ if isinstance(p_line, Polyline3D) else \ [from_point2d(pt) for pt in arc_verts] self.draw_patterned_polyline.append( (verts, col, self.LINE_TYPE[dis_obj.line_type], lw, closed)) elif isinstance(geo_obj, Cone): # translate analysis cone if dis_obj.display_mode in ('Surface', 'SurfaceWithEdges'): rh_obj = from_cone(geo_obj) mat = rd.DisplayMaterial(col, 1 - (dis_obj.color.a / 255)) self.draw_brep_shaded.append((rh_obj.ToBrep(True), mat)) if dis_obj.display_mode == 'SurfaceWithEdges': self.draw_cone.append((rh_obj, black())) elif dis_obj.display_mode == 'Wireframe': self.draw_cone.append((from_cone(geo_obj), col)) elif dis_obj.display_mode == 'Points': pt_arg = (from_point3d(geo_obj.vertex), rd.PointStyle.RoundSimple, 3, col) self.draw_point.append(pt_arg) elif isinstance(geo_obj, Cylinder): # translate analysis cylinder if dis_obj.display_mode in ('Surface', 'SurfaceWithEdges'): rh_obj = from_cylinder(geo_obj) mat = rd.DisplayMaterial(col, 1 - (dis_obj.color.a / 255)) self.draw_brep_shaded.append((rh_obj.ToBrep(True, True), mat)) if dis_obj.display_mode == 'SurfaceWithEdges': self.draw_cylinder.append((rh_obj, black())) elif dis_obj.display_mode == 'Wireframe': self.draw_cylinder.append((from_cylinder(geo_obj), col)) elif dis_obj.display_mode == 'Points': pt_arg = (from_point3d(geo_obj.center), rd.PointStyle.RoundSimple, 3, col) self.draw_point.append(pt_arg) elif isinstance(geo_obj, Plane): # translate analysis planes pln = from_plane(geo_obj) r = 10 # default radius for a plane object in rhino model units # grid lines if dis_obj.show_grid: for j in range(-5, 6): i = j / 5 p0 = pln.PointAt(i * r, -r) p1 = pln.PointAt(i * r, r) self.draw_line.append((p0, p1, col, 1)) p0 = pln.PointAt(-r, i * r) p1 = pln.PointAt(r, i * r) self.draw_line.append((p0, p1, col, 1)) # axes if dis_obj.show_axes: self.draw_line.append((pln.Origin, pln.Origin + pln.XAxis * r, col, 3)) self.draw_line.append((pln.Origin, pln.Origin + pln.YAxis * r, col, 3))
[docs] def sort_shaded_objects(self): """Sort shaded objects according to their transparency. This ensures that the resulting display has visible objects behind any transparent objects. """ if len(self.draw_brep_shaded) != 0: trans = (args[1].Transparency for args in self.draw_brep_shaded) self.draw_brep_shaded = \ [a for _, a in sorted(zip(trans, self.draw_brep_shaded))] if len(self.draw_mesh_shaded) != 0: trans = (args[1].Transparency for args in self.draw_mesh_shaded) self.draw_mesh_shaded = \ [a for _, a in sorted(zip(trans, self.draw_mesh_shaded))]
[docs] @staticmethod def convert_legend2d(legend, vw=None, vh=None): """Convert a ladybug Legend into arguments for drawing in 2D on the screen. Args: legend: A Ladybug Legend object to be converted to 2D screen-oriented geometry. vw: An integer for the view width in pixels. If None, it will be obtained from the active viewport. (Default: None). vh: An integer for the view height in pixels. If None, it will be obtained from the active viewport. (Default: None). Returns: A tuple with two values needed to render the legend in the 2D scene: - legend_mesh -- A colored mesh for the legend. - legend_title -- A text object for the legend title. """ # figure out the dimensions of the active viewport if vw is None or vh is None: active_view = sc.doc.Views.ActiveView.ActiveViewport v_size = active_view.Size vw = v_size.Width if vw is None else vw vh = v_size.Height if vh is None else vh # translate the color matrix to a bitmap l_par = legend.legend_parameters color_mtx = legend.color_map_2d(vw, vh) color_mtx = [[color_to_color(c) for c in row] for row in color_mtx] if (8, 14) <= rhino_version < (8, 15, 25007): color_mtx = list(reversed(color_mtx)) net_bm = System.Drawing.Bitmap(len(color_mtx[0]), len(color_mtx)) for y, row in enumerate(color_mtx): for x, col in enumerate(row): net_bm.SetPixel(x, y, col) rh_bm = rd.DisplayBitmap(net_bm) or_x, or_y, sh, sw, th = legend._pixel_dims_2d(vw, vh) s_count = l_par.segment_count s_count = s_count - 1 if l_par.continuous_legend else s_count leg_width = sw if l_par.vertical else sw * s_count leg_height = sh if not l_par.vertical else sh * s_count cent_pt = rg.Point2d(or_x + (leg_width / 2), or_y + (leg_height / 2)) draw_sprite = (rh_bm, cent_pt, leg_width, leg_height) # translate the legend text _height = legend.parse_dim_2d(l_par.text_height_2d, vh) _font = l_par.font txt_pts = legend.segment_text_location_2d(vw, vh) cent_txt = False if l_par.vertical else True draw_2d_text = [ (txt, black(), rg.Point2d(loc.x, loc.y), cent_txt, _height, _font) for txt, loc in zip(legend.segment_text, txt_pts)] t_pt = legend.title_location_2d(vw, vh) legend_title = (legend.title, black(), rg.Point2d(t_pt.x, t_pt.y), False, _height, _font) draw_2d_text.insert(0, legend_title) return draw_sprite, draw_2d_text
[docs] def ToString(self): """Overwrite .NET ToString method.""" return 'VisualizationSet Converter'
[docs] class RhinoPointConduit(rd.DisplayConduit): """An optimized class to preview Rhino Point3d in the Rhino Display pipeline. Args: points: A list of Rhino Point3d to be displayed in the pipeline. """ def __init__(self, points): """Initialize RhinoPointConduit.""" self.draw_point_args = [] col = black() for pt in points: self.draw_point_args.append((pt, rd.PointStyle.RoundSimple, 5, col)) self.bbox = rg.BoundingBox(points)
[docs] def CalculateBoundingBox(self, calculateBoundingBoxEventArgs): """Overwrite the method that passes the bounding box to the display.""" calculateBoundingBoxEventArgs.IncludeBoundingBox(self.bbox)
[docs] def PostDrawObjects(self, drawEventArgs): """Overwrite the method that draws the objects in the display.""" # get the DisplayPipeline from the event arguments display = drawEventArgs.Display # for each object to be rendered, pass the drawing arguments for draw_args in self.draw_point_args: display.DrawPoint(*draw_args)