Source code for honeybee_radiance.properties.model

# coding=utf-8
"""Model Radiance Properties."""
from honeybee.extensionutil import model_extension_dicts
from honeybee.checkdup import check_duplicate_identifiers
from honeybee.boundarycondition import Surface
from honeybee.typing import invalid_dict_error, clean_rad_string, clean_and_id_rad_string
from honeybee.model import Model

from ..sensorgrid import SensorGrid
from ..view import View
from ..dynamic.group import DynamicShadeGroup, DynamicSubFaceGroup
from ..modifierset import ModifierSet
from ..mutil import dict_to_modifier  # imports all modifiers classes
from ..modifier.material import aBSDF, BSDF
from ..lib.modifiers import black, generic_context
from ..lib.modifiersets import generic_modifier_set_visible

try:
    from itertools import izip as zip  # python 2
except ImportError:
    pass   # python 3


[docs] class ModelRadianceProperties(object): """Radiance Properties for Honeybee Model. Args: host: A honeybee_core Model object that hosts these properties. Properties: * host * sensor_grids * views * modifiers * blk_modifiers * room_modifiers * face_modifiers * shade_modifiers * bsdf_modifiers * modifier_sets * global_modifier_set * dynamic_shade_groups * dynamic_subface_groups * shade_group_identifiers * subface_group_identifiers * has_sensor_grids * has_views """ def __init__(self, host, sensor_grids=None, views=None): """Initialize Model radiance properties.""" self._host = host self.sensor_grids = sensor_grids self.views = views @property def host(self): """Get the Model object hosting these properties.""" return self._host @property def sensor_grids(self): """Get or set an array of SensorGrids that are associated with the model.""" return tuple(self._sensor_grids) @sensor_grids.setter def sensor_grids(self, value): if value: try: self._sensor_grids = list(value) for obj in self._sensor_grids: assert isinstance(obj, SensorGrid), 'Expected SensorGrid for Model' \ ' sensor_grids. Got {}.'.format(type(value)) except (ValueError, TypeError): raise TypeError( 'Model sensor_grids must be an array. Got {}.'.format(type(value))) else: self._sensor_grids = [] @property def views(self): """Get or set an array of Views that are associated with the model.""" return tuple(self._views) @views.setter def views(self, value): if value: try: self._views = list(value) for obj in self._views: assert isinstance(obj, View), 'Expected View for Model' \ ' views. Got {}.'.format(type(value)) except (ValueError, TypeError): raise TypeError( 'Model views must be an array. Got {}.'.format(type(value))) else: self._views = [] @property def modifiers(self): """A list of all unique modifiers in the model. This includes modifiers across all Faces, Apertures, Doors, Shades, Room ModifierSets, and modifiers for any dynamic states assigned to these objects. However, it excludes modifiers in the default modifier set. It also excludes blk_modifiers and the modifier_direct of any states, which can be obtained separately from the blk_modifiers property. """ all_mods = self.room_modifiers + self.face_modifiers + self.shade_modifiers return list(set(all_mods)) @property def blk_modifiers(self): """A list of all unique modifier_blk in the model. This includes modifier_blk across all Faces, Apertures, Doors, and Shades. It also includes modifier_direct for any dynamic states assigned to these objects. """ modifiers = [black] for face in self.host.faces: # check all orphaned Face modifiers self._check_and_add_face_modifier_blk(face, modifiers) for ap in self.host.orphaned_apertures: # check all Aperture modifiers self._check_and_add_dynamic_obj_modifier_blk(ap, modifiers) for dr in self.host.orphaned_doors: # check all Door modifiers self._check_and_add_dynamic_obj_modifier_blk(dr, modifiers) for shade in self.host.shades: self._check_and_add_dynamic_obj_modifier_blk(shade, modifiers) for sm in self.host.shade_meshes: # check all ShadeMesh modifiers self._check_and_add_obj_modifier_blk(sm, modifiers) return list(set(modifiers)) @property def room_modifiers(self): """A list of all unique modifiers assigned to Room ModifierSets.""" room_mods = [] for cnstr_set in self.modifier_sets: room_mods.extend(cnstr_set.modified_modifiers_unique) return list(set(room_mods)) @property def face_modifiers(self): """A list of all unique modifiers assigned to Faces, Apertures and Doors. This includes both objects that are a part of Rooms as well as orphaned objects. It does not include the modifiers of any shades assigned to these objects. Nor does it include any blk modifiers. """ modifiers = [] for face in self.host.faces: # check all orphaned Face modifiers self._check_and_add_face_modifier(face, modifiers) for ap in self.host.orphaned_apertures: # check all Aperture modifiers self._check_and_add_dynamic_obj_modifier(ap, modifiers) for dr in self.host.orphaned_doors: # check all Door modifiers self._check_and_add_dynamic_obj_modifier(dr, modifiers) return list(set(modifiers)) @property def shade_modifiers(self): """A list of all unique modifiers assigned to Shade and ShadeMeshes in the model. """ modifiers = [] for room in self.host.rooms: self._check_and_add_room_modifier_shade(room, modifiers) for face in self.host.orphaned_faces: self._check_and_add_face_modifier_shade(face, modifiers) for ap in self.host.orphaned_apertures: self._check_and_add_obj_modifier_shade(ap, modifiers) for dr in self.host.orphaned_doors: self._check_and_add_obj_modifier_shade(dr, modifiers) for shade in self.host.orphaned_shades: self._check_and_add_orphaned_shade_modifier(shade, modifiers) for shade_mesh in self.host.shade_meshes: self._check_and_add_shade_mesh_modifier(shade_mesh, modifiers) return list(set(modifiers)) @property def bsdf_modifiers(self): """A list of all unique BSDF modifiers in the model. This includes any BSDF modifiers in both the Model.modifiers and the Model.blk_modifiers. """ all_mods = self.modifiers + self.blk_modifiers return list(set(mod for mod in all_mods if isinstance(mod, (aBSDF, BSDF)))) @property def modifier_sets(self): """A list of all unique Room-Assigned ModifierSets in the Model.""" modifier_sets = [] for room in self.host.rooms: if room.properties.radiance._modifier_set is not None: if not self._instance_in_array(room.properties.radiance._modifier_set, modifier_sets): modifier_sets.append(room.properties.radiance._modifier_set) return list(set(modifier_sets)) # catch equivalent modifier sets @property def global_modifier_set(self): """The global radiance modifier set. This is what is used whenever no modifier has been assigned to a given Face/Aperture/Door/Shade and there is no modifier_set assigned to the parent Room. """ return generic_modifier_set_visible @property def dynamic_shade_groups(self): """Get a list of DynamicShadeGroups in the model. These can be used to write dynamic shades into radiance files. """ # gather all of the shades with a common identifier into groups group_dict = {} for shade in self.host.shades: if shade.properties.radiance._dynamic_group_identifier: group_id = shade.properties.radiance._dynamic_group_identifier try: group_dict[group_id].append(shade) except KeyError: group_dict[group_id] = [shade] # return DynamicShadeGroup objects return [DynamicShadeGroup(ident, shades)for ident, shades in group_dict.items()] @property def dynamic_subface_groups(self): """Get a list of DynamicSubFaceGroups in the model. These can be used to write dynamic Apertures and Doors into radiance files. """ # gather all of the subfaces with a common identifier into groups group_dict = {} for subface in self.host.apertures + self.host.doors: if subface.properties.radiance._dynamic_group_identifier: group_id = subface.properties.radiance._dynamic_group_identifier try: group_dict[group_id].append(subface) except KeyError: group_dict[group_id] = [subface] # return DynamicSubFaceGroup objects return [DynamicSubFaceGroup(ident, subf)for ident, subf in group_dict.items()] @property def shade_group_identifiers(self): """Get a list of identifiers for all the DynamicShadeGroups in the model.""" group_ids = set() for shade in self.host.shades: if shade.properties.radiance._dynamic_group_identifier: group_ids.add(shade.properties.radiance._dynamic_group_identifier) return list(group_ids) @property def subface_group_identifiers(self): """Get a list of identifiers for all the DynamicSubFaceGroups in the model.""" group_ids = set() for subface in self.host.apertures + self.host.doors: if subface.properties.radiance._dynamic_group_identifier: group_ids.add(subface.properties.radiance._dynamic_group_identifier) return list(group_ids) @property def has_sensor_grids(self): """Get a boolean for whether there are sensor grids assigned to the model.""" return len(self._sensor_grids) != 0 @property def has_views(self): """Get a boolean for whether there are views assigned to the model.""" return len(self._views) != 0
[docs] def remove_sensor_grids(self): """Remove all sensor grids from the model.""" self._sensor_grids = []
[docs] def remove_views(self): """Remove all views from the model.""" self._views = []
[docs] def add_sensor_grid(self, sensor_grid): """Add a SensorGrid to this model. Args: sensor_grid: A SensorGrid to add to this model. """ assert isinstance(sensor_grid, SensorGrid), \ 'Expected SensorGrid. Got {}.'.format(type(sensor_grid)) self._sensor_grids.append(sensor_grid)
[docs] def add_view(self, view): """Add a View to this model. Args: view: A View to add to this model. """ assert isinstance(view, View), 'Expected View. Got {}.'.format(type(view)) self._views.append(view)
[docs] def add_sensor_grids(self, sensor_grids): """Add a list of SensorGrids to this model.""" for grid in sensor_grids: self.add_sensor_grid(grid)
[docs] def add_views(self, views): """Add a list of Views to this model.""" for view in views: self.add_view(view)
[docs] def faces_by_blk(self): """Get all Faces in the model separated by their blk property. This method will also ensure that any faces with Surface boundary condition are not duplicated in the result but are rather offset from the base face to ensure modifiers on opposite sides of interior Faces are accounted for. Furthermore, this method also ensures that any static interior sub-faces (with Surface boundary condition) only have one of such objects in the output lists. Returns: A tuple with 2 lists: - faces: A list of all faces without a unique modifier_blk. - faces_blk: A list of all opaque faces that have a unique modifier_blk. """ faces, faces_blk = [], [] interior_faces, offset = set(), self.host.tolerance * -2 for face in self.host.faces: if isinstance(face.boundary_condition, Surface): if face.identifier in interior_faces: face = face.duplicate() face.move(face.normal * offset) else: interior_faces.add(face.boundary_condition.boundary_condition_object) for subf in face.apertures + face.doors: if subf.properties.radiance.dynamic_group_identifier is None: if subf.properties.radiance._modifier_blk: faces_blk.append(subf) else: faces.append(subf) if face.properties.radiance._modifier_blk: faces_blk.append(face) else: faces.append(face) return faces, faces_blk
[docs] def subfaces_by_blk(self): """Get model exterior sub-faces (Apertures, Doors) grouped by their blk property. Dynamic sub-faces will be excluded from the output lists. Returns: A tuple with 2 lists: - subfaces: A list of all sub-faces without a unique modifier_blk (just using the default black). - subfaces_blk: A list of all sub-faces that have a unique modifier_blk. """ subfaces, subfaces_blk = [], [] for subf in self.host.apertures + self.host.doors: if subf.properties.radiance.dynamic_group_identifier: continue # sub-face will be accounted for in the dynamic objects if isinstance(subf.boundary_condition, Surface): continue # static interior apertures are part of the scene if subf.properties.radiance._modifier_blk: subfaces_blk.append(subf) else: subfaces.append(subf) return subfaces, subfaces_blk
[docs] def shades_by_blk(self): """Get all Shades in the model separated by their blk property. Dynamic shades will be excluded from the output lists. Returns: A tuple with 2 lists: - shades: A list of all opaque shades without a unique modifier_blk (just using the default black or transparent modifier). - shades_blk: A list of all opaque shades that have a unique modifier_blk. """ shades, shades_blk = [], [] for shade in self.host.shades: if shade.properties.radiance.dynamic_group_identifier: continue # shade will be accounted for in the dynamic objects if shade.properties.radiance._modifier_blk: shades_blk.append(shade) else: shades.append(shade) return shades, shades_blk
[docs] def shade_meshes_by_blk(self): """Get all ShadeMeshes in the model separated by their blk property. Returns: A tuple with 2 lists: - shade_meshes: A list of all shade meshes without a unique modifier_blk (just using the default black or transparent modifier). - shade_meshes_blk: A list of all shade meshes that have a unique modifier_blk. """ shade_meshes, shade_meshes_blk = [], [] for shade in self.host.shade_meshes: if shade.properties.radiance._modifier_blk: shade_meshes_blk.append(shade) else: shade_meshes.append(shade) return shade_meshes, shade_meshes_blk
[docs] def move(self, moving_vec): """Move all sensor_grid and view geometry along a vector. Args: moving_vec: A ladybug_geometry Vector3D with the direction and distance to move the objects. """ for grid in self._sensor_grids: grid.move(moving_vec) for view in self._views: view.move(moving_vec)
[docs] def rotate(self, axis, angle, origin): """Rotate all sensor_grid and view geometry. Args: axis: A ladybug_geometry Vector3D axis representing the axis of rotation. angle: An angle for rotation in degrees. origin: A ladybug_geometry Point3D for the origin around which the object will be rotated. """ for grid in self._sensor_grids: grid.rotate(axis, angle, origin) for view in self._views: view.rotate(axis, angle, origin)
[docs] def rotate_xy(self, angle, origin): """Rotate all sensor_grids and views counterclockwise in the world XY plane. Args: angle: An angle in degrees. origin: A ladybug_geometry Point3D for the origin around which the object will be rotated. """ for grid in self._sensor_grids: grid.rotate_xy(angle, origin) for view in self._views: view.rotate_xy(angle, origin)
[docs] def reflect(self, plane): """Reflect all sensor_grid and view geometry across a plane. Args: plane: A ladybug_geometry Plane across which the object will be reflected. """ for grid in self._sensor_grids: grid.reflect(plane) for view in self._views: view.reflect(plane)
[docs] def scale(self, factor, origin=None): """Scale all sensor_grid and view geometry by a factor. Args: factor: A number representing how much the object should be scaled. origin: A ladybug_geometry Point3D representing the origin from which to scale. If None, it will be scaled from the World origin (0, 0, 0). """ for grid in self._sensor_grids: grid.scale(factor, origin) for view in self._views: view.scale(factor, origin)
[docs] def generate_exterior_face_sensor_grid( self, dimension, offset=0.1, face_type='Wall', punched_geometry=False): """Get a radiance SensorGrid generated from all exterior Faces of this Model. The Face geometry without windows punched into it will be used. This will be None if the Model has no exterior Faces. Args: dimension: The dimension of the grid cells as a number. offset: A number for how far to offset the grid from the base face. Positive numbers indicate an offset towards the exterior. (Default is 0.1, which will offset the grid to be 0.1 unit from the faces). face_type: Text to specify the type of face that will be used to generate grids. Note that only Faces with Outdoors boundary conditions will be used, meaning that most Floors will typically be excluded unless they represent the underside of a cantilever. Choose from the following. (Default: Wall). * Wall * Roof * Floor * All punched_geometry: Boolean to note whether the punched_geometry of the faces should be used (True) with the areas of sub-faces removed from the grid or the full geometry should be used (False). (Default:False). Returns: A honeybee_radiance SensorGrid generated from the exterior Faces of the model. Will be None if the Model has no exterior Faces. """ # generate the mesh grid from the exterior Faces face_grid = self.host.generate_exterior_face_grid( dimension, offset, face_type, punched_geometry) if face_grid is None: # no valid mesh could be generated return None # create the sensor grid from the mesh f_nm = 'Faces' if face_type.title() == 'All' else face_type.title() grid_name = '{}_Exterior{}'.format(self.host.display_name, f_nm) sensor_grid = SensorGrid.from_mesh3d(clean_rad_string(grid_name), face_grid) sensor_grid.display_name = grid_name return sensor_grid
[docs] def generate_exterior_aperture_sensor_grid( self, dimension, offset=0.1, aperture_type='All'): """Get a radiance SensorGrid generated from all exterior Apertures of this Model. This will be None if the Model has no exterior Apertures. Args: dimension: The dimension of the grid cells as a number. offset: A number for how far to offset the grid from the base aperture. Positive numbers indicate an offset towards the exterior while negative numbers indicate an offset towards the interior, essentially modeling the value of sun on the building interior. (Default is 0.1, which will offset the grid to be 0.1 unit from the aperture). aperture_type: Text to specify the type of Aperture that will be used to generate grids. Window indicates Apertures in Walls. Choose from the following. (Default: All). * Window * Skylight * All Returns: A honeybee_radiance SensorGrid generated from the exterior Apertures of the model. Will be None if the Model has no exterior Apertures. """ # generate the mesh grid from the exterior Apertures ap_grid = self.host.generate_exterior_aperture_grid( dimension, offset, aperture_type) if ap_grid is None: # no valid mesh could be generated return None # create the sensor grid from the mesh f_nm = 'Apertures' if aperture_type.title() == 'All' else aperture_type.title() grid_name = '{}_Exterior{}'.format(self.host.display_name, f_nm) sensor_grid = SensorGrid.from_mesh3d(clean_rad_string(grid_name), ap_grid) sensor_grid.display_name = grid_name return sensor_grid
[docs] def check_for_extension(self, raise_exception=True, detailed=False): """Check that the Model is valid for Radiance simulation. This process includes all relevant honeybee-core checks as well as checks that apply only for Radiance. Args: raise_exception: Boolean to note whether a ValueError should be raised if any errors are found. If False, this method will simply return a text string with all errors that were found. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A text string with all errors that were found or a list if detailed is True. This string (or list) will be empty if no errors were found. """ # set up defaults to ensure the method runs correctly detailed = False if raise_exception else detailed msgs = [] tol = self.host.tolerance ang_tol = self.host.angle_tolerance # perform checks for duplicate identifiers, which might mess with other checks msgs.append(self.host.check_all_duplicate_identifiers(False, detailed)) # perform several checks for the Honeybee schema geometry rules msgs.append(self.host.check_planar(tol, False, detailed)) msgs.append(self.host.check_self_intersecting(tol, False, detailed)) msgs.append(self.host.check_degenerate_rooms(tol, False, detailed)) # perform geometry checks related to parent-child relationships msgs.append(self.host.check_sub_faces_valid(tol, ang_tol, False, detailed)) msgs.append(self.host.check_sub_faces_overlapping(tol, False, detailed)) msgs.append(self.host.check_rooms_solid(tol, ang_tol, False, detailed)) # perform checks that are specific to Radiance msgs.append(self.check_duplicate_modifier_identifiers(False, detailed)) msgs.append(self.check_duplicate_modifier_set_identifiers(False, detailed)) msgs.append(self.check_duplicate_sensor_grid_identifiers(False, detailed)) msgs.append(self.check_duplicate_view_identifiers(False, detailed)) msgs.append(self.check_sensor_grid_rooms_in_model(False, detailed)) msgs.append(self.check_view_rooms_in_model(False, detailed)) # output a final report of errors or raise an exception full_msgs = [msg for msg in msgs if msg] if detailed: return [m for msg in full_msgs for m in msg] full_msg = '\n'.join(full_msgs) if raise_exception and len(full_msgs) != 0: raise ValueError(full_msg) return full_msg
[docs] def check_all(self, raise_exception=True, detailed=False): """Check all of the aspects of the Model radiance properties. Args: raise_exception: Boolean to note whether a ValueError should be raised if any errors are found. If False, this method will simply return a text string with all errors that were found. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A text string with all errors that were found or a list if detailed is True. This string (or list) will be empty if no errors were found. """ # set up defaults to ensure the method runs correctly detailed = False if raise_exception else detailed msgs = [] # perform checks for key honeybee model schema rules msgs.append(self.check_duplicate_modifier_identifiers(False, detailed)) msgs.append(self.check_duplicate_modifier_set_identifiers(False, detailed)) msgs.append(self.check_duplicate_sensor_grid_identifiers(False, detailed)) msgs.append(self.check_duplicate_view_identifiers(False, detailed)) msgs.append(self.check_sensor_grid_rooms_in_model(False, detailed)) msgs.append(self.check_view_rooms_in_model(False, detailed)) # output a final report of errors or raise an exception full_msgs = [msg for msg in msgs if msg] if detailed: return [m for msg in full_msgs for m in msg] full_msg = '\n'.join(full_msgs) if raise_exception and len(full_msgs) != 0: raise ValueError(full_msg) return full_msg
[docs] def check_duplicate_modifier_identifiers(self, raise_exception=True, detailed=False): """Check that there are no duplicate Modifier identifiers in the model. Args: raise_exception: Boolean to note whether a ValueError should be raised if duplicate identifiers are found. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A string with the message or a list with a dictionary if detailed is True. """ return check_duplicate_identifiers( self.modifiers, raise_exception, 'Radiance Modifier', detailed, '010001', 'Radiance', error_type='Duplicate Modifier Identifier')
[docs] def check_duplicate_modifier_set_identifiers( self, raise_exception=True, detailed=False): """Check that there are no duplicate ModifierSet identifiers in the model. Args: raise_exception: Boolean to note whether a ValueError should be raised if duplicate identifiers are found. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A string with the message or a list with a dictionary if detailed is True. """ return check_duplicate_identifiers( self.modifier_sets, raise_exception, 'ModifierSet', detailed, '010002', 'Radiance', error_type='Duplicate ModifierSet Identifier')
[docs] def check_duplicate_sensor_grid_identifiers( self, raise_exception=True, detailed=False): """Check that there are no duplicate SensorGrid identifiers in the model. Args: raise_exception: Boolean to note whether a ValueError should be raised if duplicate identifiers are found. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A string with the message or a list with a dictionary if detailed is True. """ return check_duplicate_identifiers( self.sensor_grids, raise_exception, 'SensorGrid', detailed, '010003', 'Radiance', error_type='Duplicate SensorGrid Identifier')
[docs] def check_duplicate_view_identifiers(self, raise_exception=True, detailed=False): """Check that there are no duplicate View identifiers in the model. Args: raise_exception: Boolean to note whether a ValueError should be raised if duplicate identifiers are found. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A string with the message or a list with a dictionary if detailed is True. """ return check_duplicate_identifiers( self.views, raise_exception, 'View', detailed, '010004', 'Radiance', error_type='Duplicate View Identifier')
[docs] def check_sensor_grid_rooms_in_model(self, raise_exception=True, detailed=False): """Check that the room_identifiers of SenorGrids are in the model. Args: raise_exception: Boolean to note whether a ValueError should be raised if SensorGrids reference Rooms that are not in the Model. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A string with the message or a list with a dictionary if detailed is True. """ detailed = False if raise_exception else detailed # gather a list of all the missing rooms grid_ids = [(grid, grid.room_identifier) for grid in self.sensor_grids if grid.room_identifier is not None] room_ids = set(room.identifier for room in self.host.rooms) missing_rooms = [] if detailed else set() for grid in grid_ids: if grid[1] not in room_ids: if detailed: missing_rooms.append(grid[0]) else: missing_rooms.add(grid[1]) # if missing rooms were found, then report the issue if len(missing_rooms) != 0: if detailed: all_err = [] for grid in missing_rooms: msg = 'SensorGrid "{}" has a room_identifier that is not in the ' \ 'Model: "{}"'.format(grid.identifier, grid.room_identifier) error_dict = { 'type': 'ValidationError', 'code': '010005', 'error_type': 'SensorGrid Room Not In Model', 'extension_type': 'Radiance', 'element_type': 'SensorGrid', 'element_id': [grid.identifier], 'element_name': [grid.display_name], 'message': msg } all_err.append(error_dict) return all_err else: msg = 'The model has the following missing rooms referenced by sensor ' \ 'grids:\n{}'.format('\n'.join(missing_rooms)) if raise_exception: raise ValueError(msg) return msg return [] if detailed else ''
[docs] def check_view_rooms_in_model(self, raise_exception=True, detailed=False): """Check that the room_identifiers of Views are in the model. Args: raise_exception: Boolean to note whether a ValueError should be raised if Views reference Rooms that are not in the Model. (Default: True). detailed: Boolean for whether the returned object is a detailed list of dicts with error info or a string with a message. (Default: False). Returns: A string with the message or a list with a dictionary if detailed is True. """ detailed = False if raise_exception else detailed # gather a list of all the missing rooms view_ids = [(view, view.room_identifier) for view in self.views if view.room_identifier is not None] room_ids = set(room.identifier for room in self.host.rooms) missing_rooms = [] if detailed else set() for view in view_ids: if view[1] not in room_ids: if detailed: missing_rooms.append(view[0]) else: missing_rooms.add(view[1]) if len(missing_rooms) != 0: if detailed: all_err = [] for view in missing_rooms: msg = 'View "{}" has a room_identifier that is not in the ' \ 'Model: "{}"'.format(view.identifier, view.room_identifier) error_dict = { 'type': 'ValidationError', 'code': '010006', 'error_type': 'View Room Not In Model', 'extension_type': 'Radiance', 'element_type': 'View', 'element_id': [view.identifier], 'element_name': [view.display_name], 'message': msg } all_err.append(error_dict) return all_err else: msg = 'The model has the following missing rooms referenced by ' \ 'views:\n{}'.format('\n'.join(missing_rooms)) if raise_exception: raise ValueError(msg) return msg return [] if detailed else ''
[docs] def merge_duplicate_identifier_grids(self): """Merge SensorGrids in the Model with the same identifier together. This is one automated way of fixing check_duplicate_sensor_grid_identifiers failures. In this case, it will be assumed that the user intended to make the sensor grids with the same identifier apart of the same SensorGrid object. The other possible way to address SensorGrids with duplicate identifiers is to give each grid a unique identifier if the truly are meant to be separate. """ # first group all grids with the same identifier grid_dict = {} for grid in self.sensor_grids: try: grid_dict[grid.identifier].append(grid) except KeyError: grid_dict[grid.identifier] = [grid] # merge girds together if they have the same ID merged_grids = [] for grids in grid_dict.values(): if len(grids) == 1: merged_grids.append(grids[0]) else: merged_grid = SensorGrid.from_merged_grids(grids) merged_grids.append(merged_grid) self.sensor_grids = merged_grids
[docs] def apply_properties_from_dict(self, data): """Apply the radiance properties of a dictionary to the host Model of this object. Args: data: A dictionary representation of an entire honeybee-core Model. Note that this dictionary must have ModelRadianceProperties in order for this method to successfully apply the radiance properties. """ assert 'radiance' in data['properties'], \ 'Dictionary possesses no ModelRadianceProperties.' modifiers, modifier_sets = self.load_properties_from_dict(data) # collect lists of radiance property dictionaries room_e_dicts, face_e_dicts, shd_e_dicts, ap_e_dicts, dr_e_dicts = \ model_extension_dicts(data, 'radiance', [], [], [], [], []) # apply radiance properties to objects using the radiance property dictionaries for room, r_dict in zip(self.host.rooms, room_e_dicts): if r_dict is not None: room.properties.radiance.apply_properties_from_dict( r_dict, modifier_sets) for face, f_dict in zip(self.host.faces, face_e_dicts): if f_dict is not None: face.properties.radiance.apply_properties_from_dict( f_dict, modifiers) for aperture, a_dict in zip(self.host.apertures, ap_e_dicts): if a_dict is not None: aperture.properties.radiance.apply_properties_from_dict( a_dict, modifiers) for door, d_dict in zip(self.host.doors, dr_e_dicts): if d_dict is not None: door.properties.radiance.apply_properties_from_dict( d_dict, modifiers) # apply properties to the Shades with separation of Shades from ShadeMesh all_shades = self.host.shades + self.host._shade_meshes for shade, s_dict in zip(all_shades, shd_e_dicts): if s_dict is not None: shade.properties.radiance.apply_properties_from_dict( s_dict, modifiers) # apply the sensor grids and views if they are in the data. rad_data = data['properties']['radiance'] if 'sensor_grids' in rad_data and rad_data['sensor_grids'] is not None: self.sensor_grids = \ [SensorGrid.from_dict(grid) for grid in rad_data['sensor_grids']] if 'views' in rad_data and rad_data['views'] is not None: self.views = [View.from_dict(view) for view in rad_data['views']]
[docs] def to_dict(self): """Return Model radiance properties as a dictionary.""" base = {'radiance': {'type': 'ModelRadianceProperties'}} # add the global modifier set to the dictionary gs = self.global_modifier_set.to_dict(abridged=True, none_for_defaults=False) gs['type'] = 'GlobalModifierSet' del gs['identifier'] g_mods = self.global_modifier_set.modifiers_unique gs['modifiers'] = [mod.to_dict() for mod in g_mods] gs['context_modifier'] = generic_context.identifier gs['modifiers'].append(generic_context.to_dict()) base['radiance']['global_modifier_set'] = gs # add all ModifierSets to the dictionary base['radiance']['modifier_sets'] = [] modifier_sets = self.modifier_sets for mod_set in modifier_sets: base['radiance']['modifier_sets'].append(mod_set.to_dict(abridged=True)) # add all unique Modifiers to the dictionary room_mods = [] for mod_set in modifier_sets: room_mods.extend(mod_set.modified_modifiers_unique) all_mods = room_mods + self.face_modifiers + self.shade_modifiers modifiers = list(set(all_mods)) base['radiance']['modifiers'] = [] for mod in modifiers: base['radiance']['modifiers'].append(mod.to_dict()) # add the sensor grids and views to the dictionary if len(self._sensor_grids) != 0: base['radiance']['sensor_grids'] = \ [grid.to_dict() for grid in self._sensor_grids] if len(self._views) != 0: base['radiance']['views'] = [view.to_dict() for view in self._views] return base
[docs] def duplicate(self, new_host=None): """Get a copy of this object. new_host: A new Model object that hosts these properties. If None, the properties will be duplicated with the same host. """ _host = new_host or self._host new_grids = [sg.duplicate() for sg in self._sensor_grids] new_views = [vw.duplicate() for vw in self._views] return ModelRadianceProperties(_host, new_grids, new_views)
[docs] @staticmethod def load_properties_from_dict(data): """Load model radiance properties of a dictionary to Python objects. Loaded objects include Modifiers and ModifierSets. The function is called when re-serializing a Model object from a dictionary to load honeybee_radiance objects into their Python object form before applying them to the Model geometry. Args: data: A dictionary representation of an entire honeybee-core Model. Note that this dictionary must have ModelRadianceProperties in order for this method to successfully load the radiance properties. Returns: A tuple with two elements - modifiers: A dictionary with identifiers of modifiers as keys and Python modifier objects as values. - modifier_sets: A dictionary with identifiers of modifier sets as keys and Python modifier set objects as values. """ assert 'radiance' in data['properties'], \ 'Dictionary possesses no ModelRadianceProperties.' # process all modifiers in the ModelRadianceProperties dictionary modifiers = {} if 'modifiers' in data['properties']['radiance'] and \ data['properties']['radiance']['modifiers'] is not None: for mod in data['properties']['radiance']['modifiers']: try: modifiers[mod['identifier']] = dict_to_modifier(mod) except Exception as e: invalid_dict_error(mod, e) # process all modifier sets in the ModelRadianceProperties dictionary modifier_sets = {} if 'modifier_sets' in data['properties']['radiance'] and \ data['properties']['radiance']['modifier_sets'] is not None: if 'modifier_sets' in data['properties']['radiance'] and \ data['properties']['radiance']['modifier_sets'] is not None: for m_set in data['properties']['radiance']['modifier_sets']: try: if m_set['type'] == 'ModifierSet': modifier_sets[m_set['identifier']] = \ ModifierSet.from_dict(m_set) else: modifier_sets[m_set['identifier']] = \ ModifierSet.from_dict_abridged(m_set, modifiers) except Exception as e: invalid_dict_error(m_set, e) return modifiers, modifier_sets
[docs] @staticmethod def dump_properties_to_dict(modifiers=None, modifier_sets=None): """Get a ModelRadianceProperties dictionary from arrays of Python objects. Args: modifiers: A list or tuple of radiance modifier objects. modifier_sets: A list or tuple of modifier set objects. Returns: data: A dictionary representation of ModelRadianceProperties. Note that all objects in this dictionary will follow the abridged schema. """ # process the modifiers and modifier sets all_m = [] if modifiers is None else list(modifiers) all_mod_sets = [] if modifier_sets is None else list(modifier_sets) for mod_set in all_mod_sets: all_m.extend(mod_set.modified_modifiers) # get sets of unique objects all_mods = set(all_m) # add all object dictionaries into one object data = {'type': 'ModelRadianceProperties'} data['modifiers'] = [m.to_dict() for m in all_mods] data['modifier_sets'] = [ms.to_dict(abridged=True) for ms in all_mod_sets] return data
[docs] @staticmethod def reset_resource_ids_in_dict( data, add_uuid=False, reset_modifiers=True, reset_modifier_sets=True): """Reset the identifiers of radiance resource objects in a Model dictionary. This is useful when human-readable names are needed when the model is exported to other formats like Rad and the uniqueness of the identifiers is less of a concern. Args: data: A dictionary representation of an entire honeybee-core Model. Note that this dictionary must have ModelRadianceProperties in order for this method to successfully edit the radiance properties. add_uuid: Boolean to note whether newly-generated resource object IDs should be derived only from a cleaned display_name (False) or whether this new ID should also have a unique set of 8 characters appended to it to guarantee uniqueness. (Default: False). reset_modifiers: Boolean to note whether the IDs of all modifiers in the model should be reset or kept. (Default: True). reset_modifier_sets: Boolean to note whether the IDs of all modifier sets in the model should be reset or kept. (Default: True). Returns: A new Model dictionary with the resource identifiers reset. All references to the reset resources will be correct and valid in the resulting dictionary, assuming that the input is valid. """ model = Model.from_dict(data) modifiers, modifier_sets = \ model.properties.radiance.load_properties_from_dict(data) res_func = clean_and_id_rad_string if add_uuid else clean_rad_string # change the identifiers of the modifiers if reset_modifiers: model_mods = set() for mod in model.properties.radiance.modifiers: mod.unlock() old_id, new_id = mod.identifier, res_func(mod.display_name) mod.identifier = new_id modifiers[old_id].unlock() modifiers[old_id].identifier = new_id model_mods.add(old_id) for old_id, mod in modifiers.items(): if old_id not in model_mods: mod.unlock() mod.identifier = res_func(mod.display_name) # change the identifiers of the modifier_sets if reset_modifier_sets: model_ms = set() for ms in model.properties.radiance.modifier_sets: ms.unlock() old_id, new_id = ms.identifier, res_func(ms.display_name) ms.identifier = new_id modifier_sets[old_id].unlock() modifier_sets[old_id].identifier = new_id model_ms.add(old_id) for old_id, ms in modifier_sets.items(): if old_id not in model_ms: ms.unlock() ms.identifier = res_func(ms.display_name) # create the model dictionary and update any unreferenced resources model_dict = model.to_dict() mr_props = model_dict['properties']['radiance'] mr_props['modifiers'] = [mod.to_dict() for mod in modifiers.values()] mr_props['modifier_sets'] = \ [ms.to_dict(abridged=True) for ms in modifier_sets.values()] return model_dict
def _check_and_add_room_modifier_shade(self, room, modifiers): """Check if a modifier is assigned to a Room's shades and add it to a list.""" self._check_and_add_obj_modifier_shade(room, modifiers) for face in room.faces: # check all Face modifiers self._check_and_add_face_modifier_shade(face, modifiers) def _check_and_add_face_modifier_shade(self, face, modifiers): """Check if a modifier is assigned to a Face's shades and add it to a list.""" self._check_and_add_obj_modifier_shade(face, modifiers) for ap in face.apertures: # check all Aperture modifiers self._check_and_add_obj_modifier_shade(ap, modifiers) for dr in face.doors: # check all Door Shade modifiers self._check_and_add_obj_modifier_shade(dr, modifiers) def _check_and_add_obj_modifier_shade(self, subf, modifiers): """Check if a modifier is assigned to an object's shades and add it to a list.""" for shade in subf.shades: self._check_and_add_dynamic_obj_modifier(shade, modifiers) def _check_and_add_face_modifier(self, face, modifiers): """Check if a modifier is assigned to a face and add it to a list.""" self._check_and_add_obj_modifier(face, modifiers) for ap in face.apertures: # check all Aperture modifiers self._check_and_add_dynamic_obj_modifier(ap, modifiers) for dr in face.doors: # check all Door modifiers self._check_and_add_dynamic_obj_modifier(dr, modifiers) def _check_and_add_face_modifier_blk(self, face, modifiers): """Check if a modifier_blk is assigned to a face and add it to a list.""" self._check_and_add_obj_modifier_blk(face, modifiers) for ap in face.apertures: # check all Aperture modifiers self._check_and_add_dynamic_obj_modifier_blk(ap, modifiers) for dr in face.doors: # check all Door modifiers self._check_and_add_dynamic_obj_modifier_blk(dr, modifiers) def _check_and_add_obj_modifier(self, obj, modifiers): """Check if a modifier is assigned to an object and add it to a list.""" mod = obj.properties.radiance._modifier if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) def _check_and_add_dynamic_obj_modifier(self, obj, modifiers): """Check if a modifier is assigned to a dynamic object and add it to a list.""" mod = obj.properties.radiance._modifier if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) for st in obj.properties.radiance._states: stm = (st._modifier, st._modifier_direct) + \ tuple(s.modifier for s in st._shades) for mod in stm: if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) def _check_and_add_obj_modifier_blk(self, obj, modifiers): """Check if a modifier_blk is assigned to an object and add it to a list.""" mod = obj.properties.radiance._modifier_blk if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) def _check_and_add_dynamic_obj_modifier_blk(self, obj, modifiers): """Check if a modifier_blk is assigned to a dynamic object and add it to a list. """ mod = obj.properties.radiance._modifier_blk if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) for st in obj.properties.radiance._states: for s in st._shades: mod = s.modifier if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) def _check_and_add_orphaned_shade_modifier(self, obj, modifiers): """Check if a modifier is assigned to an object and add it to a list.""" mod = obj.properties.radiance._modifier if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) else: def_mod = generic_context if obj.is_detached else \ generic_modifier_set_visible.shade_set.exterior_modifier if not self._instance_in_array(def_mod, modifiers): modifiers.append(def_mod) for st in obj.properties.radiance._states: stm = (st._modifier, st._modifier_direct) + \ tuple(s.modifier for s in st._shades) for mod in stm: if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) def _check_and_add_shade_mesh_modifier(self, obj, modifiers): """Check if a modifier is assigned to an object and add it to a list.""" mod = obj.properties.radiance._modifier if mod is not None: if not self._instance_in_array(mod, modifiers): modifiers.append(mod) else: def_mod = generic_context if obj.is_detached else \ generic_modifier_set_visible.shade_set.exterior_modifier if not self._instance_in_array(def_mod, modifiers): modifiers.append(def_mod) @staticmethod def _instance_in_array(object_instance, object_array): """Check if a specific object instance is already in an array. This can be much faster than `if object_instance in object_array` when you expect to be testing a lot of the same instance of an object for inclusion in an array since the builtin method uses an == operator to test inclusion. """ for val in object_array: if val is object_instance: return True return False
[docs] def ToString(self): return self.__repr__()
def __repr__(self): return 'Model Radiance Properties: [host: {}]'.format(self.host.display_name)