Source code for dragonfly.building

# coding: utf-8
"""Dragonfly Building."""
from __future__ import division

import math
try:
    from itertools import izip as zip  # python 2
except ImportError:
    xrange = range  # python 3

from ladybug_geometry.geometry2d import Vector2D, Point2D, LineSegment2D, Polygon2D
from ladybug_geometry.geometry3d import Vector3D, Point3D, Ray3D
from ladybug_geometry_polyskel.polysplit import perimeter_core_subfaces

from honeybee.model import Model
from honeybee.room import Room
from honeybee.shade import Shade
from honeybee.boundarycondition import Outdoors
from honeybee.boundarycondition import boundary_conditions as bcs
from honeybee.typing import clean_string, invalid_dict_error
from honeybee.units import parse_distance_string

from ._base import _BaseGeometry
from .properties import BuildingProperties
from .story import Story
from .roof import RoofSpecification
from .room2d import Room2D
from .windowparameter import _AsymmetricBase
import dragonfly.writer.building as writer


[docs] class Building(_BaseGeometry): """A complete Building defined by Stories (and optional extra 3D rooms). Buildings must have at least one dragonfly Story or one Honeybee Room under the room_3ds property. Args: identifier: Text string for a unique Building ID. Must be < 100 characters and not contain any spaces or special characters. unique_stories: An array of unique Dragonfly Story objects that together form the entire building. Stories input here can be in any order but they will be automatically sorted from lowest floor to highest floor when they are assigned to the Building. Note that, if a given Story is repeated several times over the height of the Building, the unique Story included in this list should be the first (lowest) Story of the repeated floors. (Default: None). room_3ds: An optional array of 3D Honeybee Room objects for additional Rooms that are a part of the Building but are not represented within the unique_stories. This is useful when there are parts of the Building geometry that cannot easily be represented with the extruded floor plate and sloped roof assumptions that underlie Dragonfly Room2Ds and RoofSpecification. Cases where this input is most useful include sloped walls and certain types of domed roofs that become tedious to implement with RoofSpecification. Matching the Honeybee Room.story property to the Dragonfly Story.display_name of an object within the unique_stories will effectively place the Honeybee Room on that Story for the purposes of floor_area, exterior_wall_area, etc. However, note that the Honeybee Room.multiplier property takes precedence over whatever multiplier is assigned to the Dragonfly Story that the Room.story may reference. (Default: None). sort_stories: A boolean to note whether the unique_stories should be sorted from lowest to highest story upon initialization (True) or whether the input order of unique_stories should be left as-is. (Default: True). Properties: * identifier * display_name * full_id * unique_stories * unique_room_2ds * room_3ds * room_3d_faces * room_3d_apertures * room_3d_doors * room_3d_shades * has_room_2ds * has_room_3ds * room_2d_story_names * room_3d_story_names * story_count * story_count_above_ground * unique_stories_above_ground * height * height_above_ground * height_from_first_floor * footprint_area * floor_area * exterior_wall_area * exterior_aperture_area * volume * min * max * user_data """ __slots__ = ('_unique_stories', '_room_3ds', '_roofs') def __init__(self, identifier, unique_stories=None, room_3ds=None, sort_stories=True): """A complete Building defined by Stories.""" # initialize and perform a basic check that there's some geometry _BaseGeometry.__init__(self, identifier) # process the identifier if (unique_stories is None or len(unique_stories) == 0) and \ (room_3ds is None or len(room_3ds) == 0): raise ValueError( 'Building must have at least one Story or one Room under room_3ds.') # process the story geometry if unique_stories is not None: for story in unique_stories: assert isinstance(story, Story), \ 'Expected dragonfly Story. Got {}'.format(type(story)) story._parent = self if sort_stories: unique_stories = \ tuple(sorted(unique_stories, key=lambda x: x.floor_height)) else: unique_stories = tuple(unique_stories) self._unique_stories = unique_stories else: self._unique_stories = () # process the room_3d geometry if room_3ds is not None: for room in room_3ds: assert isinstance(room, Room), \ 'Expected honeybee Room. Got {}'.format(type(room)) room._parent = self # assign stories to any Rooms that lack them if not all([r.story is not None for r in room_3ds]): Room.stories_by_floor_height(room_3ds) self._room_3ds = tuple(room_3ds) else: self._room_3ds = () # initialize properties self._roofs = None # used under the hood to correctly assign roofs self._properties = BuildingProperties(self) # properties for extensions
[docs] @classmethod def from_footprint(cls, identifier, footprint, floor_to_floor_heights, perimeter_offset=0, tolerance=0): """Initialize a Building from an array of Face3Ds representing a footprint. All of the resulting Room2Ds will have a floor-to-ceiling height equal to the Story floor-to-floor height. Also, none of the Room2Ds will have contact with the ground or top exposure but the separate_top_bottom_floors method can be used to automatically break these floors out from the multiplier representation and assign these properties. Args: identifier: Text string for a unique Building ID. Must be < 100 characters and not contain any spaces or special characters. footprint: An array of horizontal ladybug-geometry Face3Ds that together represent the the footprint of the Building. floor_to_floor_heights: An array of float values with a length equal to the number of stories in the Building. Each value in the list represents the floor_to_floor height of the Story starting from the first floor and then moving to the top floor. Note that numbers should be in the units system of the footprint geometry. perimeter_offset: An optional positive number that will be used to offset the perimeter of the footprint to create core/perimeter zones. If this value is 0, no offset will occur and each story will be represented with a single Room2D per polygon (Default: 0). tolerance: The maximum difference between z values at which point vertices are considered to be in the same horizontal plane. This is used to check that all vertices of the input floor_geometry lie in the same horizontal floor plane. Default is 0, which will not perform any check. """ # generate the unique Room2Ds from the footprint room_2ds = cls._generate_room_2ds( footprint, floor_to_floor_heights[0], perimeter_offset, identifier, 1, tolerance) # generate the unique stories from the floor_to_floor_heights stories = [] total_height = 0 prev_flr_to_flr = None for i, flr_hgt in enumerate(floor_to_floor_heights): if flr_hgt != prev_flr_to_flr: if i != 0: rooms = [room.duplicate() for room in room_2ds] move_vec = Vector3D(0, 0, total_height) for j, room in enumerate(rooms): room.move(move_vec) room.floor_to_ceiling_height = flr_hgt room.identifier = \ '{}_Floor{}_Room{}'.format(identifier, i + 1, j + 1) if perimeter_offset != 0: # reset all boundary conditions for room in rooms: room.boundary_conditions = [bcs.outdoors] * len(room) Room2D.solve_adjacency(rooms, tolerance) else: rooms = room_2ds stories.append(Story( '{}_Floor{}'.format(identifier, i + 1), rooms, flr_hgt)) else: stories[-1].multiplier += 1 total_height += flr_hgt prev_flr_to_flr = flr_hgt return cls(identifier, stories)
[docs] @classmethod def from_all_story_geometry(cls, identifier, all_story_geometry, floor_to_floor_heights, perimeter_offset=0, tolerance=0.01): """Initialize a Building from an array of Face3Ds arrays representing all floors. This method will test to see which of the stories are geometrically unique (accoutring for both the floor plate geometry and the floor_to_floor_heights). It will only include the unique floor geometries in the resulting Building. All of the resulting Room2Ds will have a floor-to-ceiling height equal to the Story floor-to-floor height. Args: identifier: Text string for a unique Building ID. Must be < 100 characters and not contain any spaces or special characters. all_story_geometry: An array of arrays with each sub-array possessing horizontal ladybug-geometry Face3Ds that representing the floor plates of the building. Together, these Face3Ds should represent all Stories of a building and each array of Face3Ds should together represent one Story. floor_to_floor_heights: An array of float values with a length equal to the number of stories in the Building. Each value in the list represents the floor_to_floor height of the Story starting from the first floor and then moving to the top floor. Note that numbers should be in the units system of the footprint geometry. perimeter_offset: An optional positive number that will be used to offset the perimeter of the all_story_geometry to create core/perimeter zones. If this value is 0, no offset will occur and each story will be represented with a single Room2D per polygon (Default: 0). tolerance: The maximum difference between x, y, and z values at which point vertices are considered to be the same. This is also needed as a means to determine which floor geometries are equivalent to one another and should be a part the same Story. Default: 0.01, suitable for objects in meters. """ # generate the first story of the building room_2ds = cls._generate_room_2ds( all_story_geometry[0], floor_to_floor_heights[0], perimeter_offset, identifier, 1, tolerance) stories = [Story('{}_Floor1'.format(identifier), room_2ds, floor_to_floor_heights[0])] # generate the remaining unique stories from the floor_to_floor_heights remaining_geo = all_story_geometry[1:] remaining_flr_hgts = floor_to_floor_heights[1:] prev_geo = all_story_geometry[0] prev_flr_to_flr = floor_to_floor_heights[0] for i, (room_geo, flr_hgt) in enumerate(zip(remaining_geo, remaining_flr_hgts)): # test is anything is geometrically different if flr_hgt != prev_flr_to_flr or len(room_geo) != len(prev_geo) or \ not all(cls._is_story_equivalent(rm1, rm2, tolerance) for rm1, rm2 in zip(room_geo, prev_geo)): room_2ds = cls._generate_room_2ds( room_geo, flr_hgt, perimeter_offset, identifier, i + 2, tolerance) stories.append(Story( '{}_Floor{}'.format(identifier, i + 2), room_2ds, flr_hgt)) else: # geometry is the same as the floor below stories[-1].multiplier += 1 prev_geo = room_geo prev_flr_to_flr = flr_hgt return cls(identifier, stories)
[docs] @classmethod def from_dict(cls, data, tolerance=0, angle_tolerance=0, sort_stories=True): """Initialize an Building from a dictionary. Args: data: A dictionary representation of a Building object. tolerance: The maximum difference between z values at which point vertices are considered to be in the same horizontal plane. This is used to check that all vertices of the input floor_geometry lie in the same horizontal floor plane. Default is 0, which will not perform any check. angle_tolerance: The max angle difference in degrees that vertices are allowed to differ from one another in order to consider them colinear. Default is 0, which makes no attempt to evaluate whether the Room volume is closed. sort_stories: A boolean to note whether the unique_stories should be sorted from lowest to highest story upon initialization (True) or whether the input order of unique_stories should be left as-is. (Default: True). """ # check the type of dictionary assert data['type'] == 'Building', 'Expected Building dictionary. ' \ 'Got {}.'.format(data['type']) # extract the 2D Stories stories = [] if 'unique_stories' in data and data['unique_stories'] is not None: for s_dict in data['unique_stories']: try: stories.append(Story.from_dict(s_dict, tolerance)) except Exception as e: invalid_dict_error(s_dict, e) # extract any additional 3D Rooms room_3ds = [] if 'room_3ds' in data and data['room_3ds'] is not None: for r_dict in data['room_3ds']: try: room_3ds.append(Room.from_dict(r_dict, tolerance, angle_tolerance)) except Exception as e: invalid_dict_error(r_dict, e) # create the Building object building = cls(data['identifier'], stories, room_3ds, sort_stories=sort_stories) # assign all other properties that are not a part of initializer if 'roof' in data and data['roof'] is not None and 'geometry' in data['roof'] \ and len(data['roof']['geometry']) > 0: roof = RoofSpecification.from_dict(data['roof']) building.add_roof_geometry(roof.geometry, tolerance) if '_roofs' in data and data['_roofs'] is not None: # secret for filtered roofs bldg_roofs = [] for st_id, r_spec in data['_roofs']: if r_spec is not None: roof = RoofSpecification.from_dict(r_spec) rf_height = (roof.max_height + roof.min_height) / 2 bldg_roofs.append((st_id, rf_height, roof)) else: bldg_roofs.append((st_id, None, None)) building._roofs = bldg_roofs if 'display_name' in data and data['display_name'] is not None: building.display_name = data['display_name'] if 'user_data' in data and data['user_data'] is not None: building.user_data = data['user_data'] if data['properties']['type'] == 'BuildingProperties': building.properties._load_extension_attr_from_dict(data['properties']) return building
[docs] @classmethod def from_honeybee(cls, model, conversion_method='AllRoom2D'): """Initialize a Building from a Honeybee Model. If each Room has a story, these will be used to determine the separation into Dragonfly stories. Otherwise, stories will be auto-generated based on the floor heights of rooms. Args: model: A Honeybee Model to be converted to a Dragonfly Building. conversion_method: Text to indicate how the Honeybee Rooms should be converted to Dragonfly. Note that the AllRoom2D option may result in some loss or simplification of the 3D Honeybee geometry but ensures that all of Dragonfly's features for editing the rooms can be used. The ExtrudedOnly method will convert only the 3D Rooms that would have no loss or simplification of geometry when converted to Room2D. AllRoom3D keeps all detailed 3D geometry on the Building.room_3ds property, enabling you to convert the 3D Rooms to Room2D using the Building.convert_room_3ds_to_2d() method as you see fit. (Default: AllRoom2D). Choose from the following options. * AllRoom2D - All Honeybee Rooms converted to Dragonfly Room2D * ExtrudedOnly - Only pure extrusions converted to Dragonfly Room2D * AllRoom3D - All Honeybee Rooms left as-is on Building.room_3ds """ # if the rooms are being left as they are, just create the Building method = conversion_method.lower() if method in ('allroom3d', 'extrudedonly'): dup_rooms = [r.duplicate() for r in model.rooms] bldg = cls(model.identifier, room_3ds=dup_rooms) bldg._display_name = model._display_name if method == 'extrudedonly': bldg.convert_all_room_3ds_to_2d( extrusion_rooms_only=True, tolerance=model.tolerance, angle_tolerance=model.angle_tolerance) for story in bldg.unique_stories: story._reset_adjacencies_from_honeybee( story.room_2ds, model.tolerance) return bldg elif method != 'allroom2d': msg = 'Building.from_honeybee conversion_method "{}" is not recognized\n' \ 'Choose from: AllRoom2D, ExtrudedOnly, AllRoom3D.'.format( conversion_method) raise ValueError(msg) # proceed to convert all to Room2D; assign stories if they don't already exist min_diff = parse_distance_string('2m', model.units) remove_stories = False if not all([room.story is not None for room in model.rooms]): model.assign_stories_by_floor_height(min_diff) remove_stories = True # group the rooms by story and create dragonfly Stories story_dict = {} for room in model.rooms: try: story_dict[room.story].append(room) except KeyError: story_dict[room.story] = [room] # evaluate floor heights to see if floors should be split removed_flrs, new_flrs = [], {} for s_id, rms in story_dict.items(): if not cls._room_story_geometry_valid(rms): rm_grps, flr_hts = Room.group_by_floor_height(rms, min_diff) for grp, ht in zip(rm_grps, flr_hts): new_flrs['{}_{}'.format(s_id, ht)] = grp removed_flrs.append(s_id) for r_flr in removed_flrs: story_dict.pop(r_flr) story_dict.update(new_flrs) # create the Story and Building objects stories = [] for s_id, rms in story_dict.items(): story_id = clean_string(str(s_id)) valid_rooms = [r for r in rms if not r.exclude_floor_area] if len(valid_rooms) != 0: story = Story.from_honeybee(story_id, rms, model.tolerance) stories.append(story) bldg = cls(model.identifier, stories) bldg._display_name = model._display_name # if stories were auto-generated, remove them to avoid editing the input if remove_stories: for rm in model.rooms: rm.story = None return bldg
@staticmethod def _room_story_geometry_valid(rooms): """Check that a set of Honeybee Rooms have geometry that makes a valid Story. Args: rooms: An array of Honeybee Rooms that will be checked to ensure their geometry makes a valid Story. Returns: True if the Room geometries make a valid Story. False if they do not. """ if len(rooms) == 1: return True flr_hts = sorted([rm.geometry.min.z for rm in rooms]) min_flr_to_ceil = min([rm.geometry.max.z - rm.geometry.min.z for rm in rooms]) return True if flr_hts[-1] - flr_hts[0] < min_flr_to_ceil else False @property def unique_stories(self): """Get a tuple of only unique Story objects that form the Building. Repeated stories are represented only once but will have a non-unity multiplier. """ return self._unique_stories @property def unique_room_2ds(self): """Get a list of the unique Room2D objects that form the Building.""" rooms = [] for story in self._unique_stories: rooms.extend(story.room_2ds) return rooms @property def room_3ds(self): """Get a tuple of additional 3D Honeybee Rooms assigned to the Building. These rooms are a part of the Building but are not represented within the unique_stories or unique_room_2ds. Matching the Honeybee Room.story property to the Dragonfly Story.display_name of an object within the unique_stories will effectively place the Honeybee Room on that Story for the purposes of floor_area, exterior_wall_area, etc. However, note that the Honeybee Room.multiplier property takes precedence over whatever multiplier is assigned to the Dragonfly Story that the Room.story may reference. """ return self._room_3ds @property def room_3d_faces(self): """Get a list of all Face objects for the 3D Honeybee Rooms in the Building.""" return [face for room in self._room_3ds for face in room._faces] @property def room_3d_apertures(self): """Get a list of all Aperture objects for the 3D Honeybee Rooms in the Building. """ child_apertures = [] for room in self._room_3ds: for face in room._faces: child_apertures.extend(face._apertures) return child_apertures @property def room_3d_doors(self): """Get a list of all Door objects for the 3D Honeybee Rooms in the Building.""" child_doors = [] for room in self._room_3ds: for face in room._faces: child_doors.extend(face._doors) return child_doors @property def room_3d_shades(self): """Get a list of all Shade objects for the 3D Honeybee Rooms in the Building.""" child_shades = [] for room in self._room_3ds: child_shades.extend(room.shades) for face in room.faces: child_shades.extend(face.shades) for ap in face._apertures: child_shades.extend(ap.shades) for dr in face._doors: child_shades.extend(dr.shades) return child_shades @property def has_room_2ds(self): """Get a boolean noting whether this Building has Room2Ds assigned under stories. """ return len(self._unique_stories) != 0 @property def has_room_3ds(self): """Get a boolean noting whether this Building has 3D Honeybee Rooms. """ return len(self._room_3ds) != 0 @property def room_2d_story_names(self): """Get a tuple of all Story display_names that have Room2Ds on them.""" return tuple(story.display_name for story in self._unique_stories) @property def room_3d_story_names(self): """Get a tuple of all story display_names that have 3D Honeybee Rooms on them.""" return tuple(set(rm.story for rm in self._room_3ds)) @property def story_count(self): """Get an integer for the number of stories in the building. This includes both the Room2Ds within unique_stories (including the Story.multiplier) as well as all stories defined by the room_3ds. """ r3d_stories = 0 if self.has_room_3ds: story_2ds = self.room_2d_story_names for st in self.room_3d_story_names: if st not in story_2ds: r3d_stories += 1 return sum((story.multiplier for story in self._unique_stories)) + r3d_stories @property def story_count_above_ground(self): """Get an integer for the number of stories above the ground. All stories defined by 3D Rooms are assumed to be above ground. """ r3d_stories = 0 if self.has_room_3ds: story_2ds = self.room_2d_story_names for st in self.room_3d_story_names: if st not in story_2ds: r3d_stories += 1 return sum((story.multiplier for story in self.unique_stories_above_ground)) + \ r3d_stories @property def unique_stories_above_ground(self): """Get a tuple of unique Story objects that are above the ground. A story is considered above the ground if at least one of its Room2Ds has an outdoor boundary condition. """ return [story for story in self._unique_stories if story.is_above_ground] @property def height(self): """Get a number for the roof height of the Building as an absolute Z-coordinate. This property will account for the fact that the tallest Room may be a 3D Honeybee Room. """ r2_h, r3_h = None, None if self.has_room_3ds: r3_h = max(r.max.z for r in self.room_3ds) if self.has_room_2ds: last_flr = self._unique_stories[-1] r2_h = last_flr.floor_height + \ (last_flr.floor_to_floor_height * last_flr.multiplier) if r2_h is not None and r3_h is not None: return max(r2_h, r3_h) elif r2_h is not None: return r2_h return r3_h @property def height_above_ground(self): """Get a the height difference between the roof and first floor above the ground. This property will account for any 3D Room if they exist. """ r2_h, r3_h, bldg_h = None, None, self.height try: r2_h = bldg_h - self.unique_stories_above_ground[0].floor_height except IndexError: # building completely below ground or no Room2Ds r2_h = 0 if self.has_room_3ds: r3_h = bldg_h - min(r.min.z for r in self.room_3ds) if r2_h is not None and r3_h is not None: return max(r2_h, r3_h) elif r2_h is not None: return r2_h return r3_h @property def height_from_first_floor(self): """Get a the height difference between the roof and the bottom-most floor. This property will account for any 3D Room if they exist. """ r2_h, r3_h, bldg_h = None, None, self.height try: r2_h = bldg_h - self.unique_stories[0].floor_height except IndexError: # building completely below ground or no Room2Ds r2_h = 0 if self.has_room_3ds: r3_h = bldg_h - min(r.min.z for r in self.room_3ds) if r2_h is not None and r3_h is not None: return max(r2_h, r3_h) elif r2_h is not None: return r2_h return r3_h @property def footprint_area(self): """Get a number for the total footprint area of the Building. The footprint is derived from the lowest dragonfly Story of the building unless the Building is composed entirely of 3D Rooms, in which case it is the combined floor area of the Rooms belonging to the lowest story. """ try: return self._unique_stories[0].floor_area except IndexError: # no Room2Ds return sum(r.floor_area for r in self._lowest_story_room_3ds() if not r.exclude_floor_area) @property def floor_area(self): """Get a number for the total floor area in the Building. This property uses both the 2D Story multipliers and the 3D Room multipliers to determine the total floor area. """ fa_r2 = sum([story.floor_area * story.multiplier for story in self._unique_stories]) fa_r3 = sum([room.floor_area * room.multiplier for room in self._room_3ds if not room.exclude_floor_area]) return fa_r2 + fa_r3 @property def exterior_wall_area(self): """Get a number for the total exterior wall area in the Building. This property uses both the 2D Story multipliers and the 3D Room multipliers to determine the total exterior wall area. """ ewa_r2 = sum([story.exterior_wall_area * story.multiplier for story in self._unique_stories]) ewa_r3 = sum([r.exterior_wall_area * r.multiplier for r in self._room_3ds]) return ewa_r2 + ewa_r3 @property def exterior_aperture_area(self): """Get a number for the total exterior wall aperture area in the Building. This property uses both the 2D Story multipliers and the 3D Room multipliers to determine the total exterior wall aperture area. All skylights apertures are excluded. """ eaa_r2 = sum([story.exterior_aperture_area * story.multiplier for story in self._unique_stories]) eaa_r3 = sum([room.exterior_wall_aperture_area * room.multiplier for room in self._room_3ds]) return eaa_r2 + eaa_r3 @property def volume(self): """Get a number for the volume of all the Rooms in the Building. This property uses both the 2D Story multipliers and the 3D Room multipliers to determine the total Building volume. """ v_2r = sum([story.volume * story.multiplier for story in self._unique_stories]) v_3r = sum([room.volume * room.multiplier for room in self._room_3ds]) return v_2r + v_3r @property def min(self): """Get a Point2D for the min bounding rectangle vertex in the XY plane. This is useful in calculations to determine if this Building is in proximity to other objects. """ r2_min_pt, r3_min_pt = None, None if self.has_room_2ds: r2_min_pt = self._calculate_min(self._unique_stories) if self.has_room_3ds: r3_min_pt = Model._calculate_min(self._room_3ds) if r2_min_pt is not None and r3_min_pt is not None: return Point2D(min(r2_min_pt.x, r3_min_pt.x), min(r2_min_pt.y, r3_min_pt.y)) elif r2_min_pt is not None: return r2_min_pt return Point2D(r3_min_pt.x, r3_min_pt.y) @property def max(self): """Get a Point2D for the max bounding rectangle vertex in the XY plane. This is useful in calculations to determine if this Building is in proximity to other objects. """ r2_max_pt, r3_max_pt = None, None if self.has_room_2ds: r2_max_pt = self._calculate_max(self._unique_stories) if self.has_room_3ds: r3_max_pt = Model._calculate_max(self._room_3ds) if r2_max_pt is not None and r3_max_pt is not None: return Point2D(max(r2_max_pt.x, r3_max_pt.x), max(r2_max_pt.y, r3_max_pt.y)) elif r2_max_pt is not None: return r2_max_pt return Point2D(r3_max_pt.x, r3_max_pt.y)
[docs] def all_stories(self): """Get a list of all Story objects that form the Building. The Story objects returned here each have a multiplier of 1 and repeated stories are represented will their own Story object. 3D Rooms are not included in this output. """ all_stories = [] for story in self._unique_stories: new_story = story.duplicate() new_story.multiplier = 1 all_stories.append(new_story) if story.multiplier != 1: for i in range(story.multiplier - 1): new_story = story.duplicate() new_story.add_prefix('Flr{}'.format(i + 1)) new_story.multiplier = 1 m_vec = Vector3D(0, 0, story.floor_to_floor_height * (i + 1)) new_story.move(m_vec) all_stories.append(new_story) return all_stories
[docs] def all_room_2ds(self): """Get a list of all Room2D objects that form the Building.""" rooms = [] for story in self.all_stories(): rooms.extend(story.room_2ds) return rooms
[docs] def room_2ds_by_display_name(self, room_name): """Get all of the Room2Ds with a given display_name in the Building.""" rooms = [] for room in self.unique_room_2ds: if room.display_name == room_name: rooms.append(room) return rooms
[docs] def room_3ds_by_display_name(self, room_name): """Get all of the 3D Rooms with a given display_name in the Building.""" rooms = [] for room in self.room_3ds: if room.display_name == room_name: rooms.append(room) return rooms
[docs] def room_3ds_by_story(self, story_name): """Get all of the 3D Honeybee Room objects assigned to a particular story. Args: story_name: Text for the display_name of the Story for which Honeybee Room objects will be returned. """ rooms = [] for room in self.room_3ds: if room.story == story_name: rooms.append(room) return rooms
[docs] def footprint(self, tolerance=0.01): """A list of Face3D objects representing the footprint of the building. The footprint is derived from the lowest story of the building and, if all Room2Ds of this story can be joined into a single continuous polyface, then only one Face3D will be contained in the list output from this method. Otherwise, several Face3Ds may be output. Args: tolerance: The minimum distance between points at which they are not considered touching. Default: 0.01, suitable for objects in meters. """ if self.has_room_2ds: ground_story = self._unique_stories[0] if len(ground_story.room_2ds) == 1: # no need to create any new geometry return [ground_story.room_2ds[0].floor_geometry] else: # need a single list of Face3Ds for the whole footprint return ground_story.footprint(tolerance) foot_rooms = self._lowest_story_room_3ds() return Room.grouped_horizontal_boundary(foot_rooms, tolerance=tolerance)
[docs] def shade_representation( self, exclude_index=None, cap=False, include_room3ds=False, tolerance=0.01): """A list of honeybee Shade objects representing the building geometry. These can be used to account for this Building's shade in the simulation of another nearby Building. Args: exclude_index: An optional index for a unique_story to be excluded from the shade representation. If None, all stories will be included in the result. (Default: None). cap: Boolean to note whether the shade representation should be capped with a top face. Usually, this is not necessary to account for blocked sun and is only needed when it's important to account for reflected sun off of roofs. (Default: False). include_room3ds: Boolean to note whether the 3D Rooms assigned to this Building should be included in the shade representation. Only exterior geometries are included. (Default: False). tolerance: The minimum distance between points at which they are not considered touching. Default: 0.01, suitable for objects in meters. """ context_shades = [] if exclude_index is None: for story in self.unique_stories: context_shades.extend(story.shade_representation(cap, tolerance)) else: for i, story in enumerate(self.unique_stories): if i != exclude_index: context_shades.extend(story.shade_representation(cap, tolerance)) else: mult_shd = story.shade_representation_multiplier( cap=cap, tolerance=tolerance) context_shades.extend(mult_shd) if include_room3ds and self.has_room_3ds: for room in self.room_3ds: for face in room.faces: if isinstance(face.boundary_condition, Outdoors): context_shades.append(Shade(face.identifier, face.geometry)) return context_shades
[docs] def suggested_alignment_axes( self, distance, direction=Vector2D(0, 1), angle_tolerance=1.0): """Get suggested LineSegment2Ds to be used for this Building. This method will return the most common axes across the Building's Room2D geometry along with the number of Room2D segments that correspond to each axis. The latter can be used to filter the suggested alignment axes to get only the most common ones across the Building. Args: distance: A number for the distance that will be used in the alignment operation. This will be used to determine the resolution at which alignment axes are generated and evaluated. Smaller alignment distances will result in the generation of more common_axes since a finer resolution can differentiate common that would typically be grouped together. For typical building geometry, an alignment distance of 0.3 meters or 1 foot is typically suitable for eliminating unwanted details while not changing the geometry too much from its original location. direction: A Vector2D object to represent the direction in which the common axes will be evaluated and generated. angle_tolerance: The max angle difference in radians that the Room2D segment direction can differ from the input direction before the segments are not factored into this calculation of common axes. Returns: A tuple with two elements. - common_axes: A list of LineSegment2D objects for the common axes across the input Room2Ds. - axis_values: A list of integers that aligns with the common_axes and denotes how many segments of the input Room2D each axis relates to. Higher numbers indicate that that the axis is more commonly aligned across the Room2Ds. """ return Room2D.generate_alignment_axes( self.unique_room_2ds, distance, direction, angle_tolerance)
[docs] def find_adjacency_gaps(self, gap_distance=0.1, tolerance=0.01): """Identify gaps smaller than a gap_distance between this Building's Room2Ds. All cases where gaps can create failed adjacency solving or failed intersections between adjacent stories will be checked. Args: gap_distance: The maximum distance between two Room2Ds that is considered an adjacency gap. Differences between Room2Ds that are higher than this distance are considered meaningful gaps to be preserved. This value should be higher than the tolerance to be meaningful. (Default: 0.1, suitable for objects in meters). tolerance: The minimum difference between the coordinate values at which point they are considered equivalent. (Default: 0.01, suitable for objects in meters). Returns: A list of Point2Ds that note the location of any gaps between this Building's Room2Ds, which are larger than the tolerance but less than the gap_distance. """ gap_points = [] prev_mult, story_count = 0, len(self._unique_stories) for i, story in enumerate(self._unique_stories): if prev_mult == 1: # test this story together with the one below room_group = story.room_2ds + self._unique_stories[i - 1].room_2ds pts = Room2D.find_adjacency_gaps(room_group, gap_distance, tolerance) gap_points.extend(pts) elif story.multiplier != 1: # lone bottom/middle story to test pts = Room2D.find_adjacency_gaps(story.room_2ds, gap_distance, tolerance) gap_points.extend(pts) elif i + 1 == story_count: # lone top story to test pts = Room2D.find_adjacency_gaps(story.room_2ds, gap_distance, tolerance) gap_points.extend(pts) prev_mult = story.multiplier return list(set(gap_points)) # remove duplicates in the result
[docs] def convert_multipliers_to_stories(self): """Convert this Building's stories with non-unity multipliers to geometry.""" exist_story_ids = set(story.identifier for story in self.unique_stories) stories_to_add = [] for story in self.all_stories(): if story.identifier not in exist_story_ids: stories_to_add.append(story) self.add_stories(stories_to_add) for story in self.unique_stories: story.multiplier = 1
[docs] def add_stories(self, stories, add_duplicate_ids=False): """Add additional Story objects to this Building. Using this method will ensure that Stories are ordered according to their floor height as they are added. Also, in the case that Story identifiers match an existing one in this Building, these Stories will be merged together. If add_duplicate_ids is False, Room2Ds that have matching identifiers within a merged Story will not be ignored in order to avoid ID conflicts. Args: stories: A list or tuple of Story objects to be added to this Building. add_duplicate_ids: A boolean to note whether added Room2Ds that have matching identifiers within each Story should be ignored (False) or they should be added to the Story creating an ID collision that can be resolved later (True). (Default: False). """ # check to be sure all of the input is correct for story in stories: assert isinstance(story, Story), \ 'Expected dragonfly Story. Got {}'.format(type(story)) # create the list of new stories, merging stories that have the same identifier new_stories = list(self._unique_stories) for o_story in stories: for e_story in new_stories: if o_story.identifier == e_story.identifier: e_story.add_room_2ds(o_story.room_2ds, add_duplicate_ids) break else: o_story._parent = self new_stories.append(o_story) # sort the stories by floor level and assign them to this Building unique_stories = tuple(sorted(new_stories, key=lambda x: x.floor_height)) self._unique_stories = unique_stories
[docs] def add_room_3ds(self, rooms, add_duplicate_ids=False): """Add additional 3D Honeybee Room objects to this Building. Args: stories: A list or tuple of Honeybee Room objects to be added to this building. add_duplicate_ids: A boolean to note whether added Rooms that have matching identifiers within the current Building should be ignored (False) or they should be added to the Building creating an ID collision that can be resolved later (True). (Default: False). """ # check to be sure that the input is composed of Rooms for room in rooms: assert isinstance(room, Room), \ 'Expected honeybee Room. Got {}'.format(type(room)) # add the rooms and deal with duplicated IDs appropriately new_room_3ds = list(self._room_3ds) if add_duplicate_ids: for room in rooms: room._parent = self if room.story is None: room.story = 'Unknown' new_room_3ds.append(room) else: exist_set = {rm.identifier for rm in self._room_3ds} for room in rooms: if room.identifier not in exist_set: room._parent = self if room.story is None: room.story = 'Unknown' new_room_3ds.append(room) # assign the new Rooms to this Building self._room_3ds = tuple(new_room_3ds)
[docs] def add_roof_geometry(self, roof_geometry, tolerance=0.01, overlap_threshold=0): """Add roof geometry to the stories of this Building. This method will attempt add each roof geometry to the best Story in the Building by checking for overlaps between the Story's Room2Ds and the Roof geometry in plan. When a given roof geometry overlaps with several Stories more than the specified overlap_threshold, the top-most Story will get the roof geometry assigned to it unless this top Story has a floor_height above the roof geometry, in which case the next highest story will be checked until a compatible one is found. If a given roof geometry does not overlap with any story geometry or lies below all of the stories, it will not be assigned to the Building. Args: roof_geometry: An array of Face3D objects representing the geometry of the Roof. tolerance: The maximum difference between values at which point vertices are considered to be the same. (Default: 0.01, suitable for objects in Meters). overlap_threshold: A number between 0 and 1 for the fraction of a room's area that must be covered by a given roof geometry for it to be considered overlapping with that room. This is intended to prevent incorrect roof assignment in cases where roofs extend slightly past the room they are intended for. (Default: 0.05). """ # convert all roof geometries to clean 2D polygons roof_polygons, clean_roofs = [], [] for r_geo in roof_geometry: try: clean_geo = r_geo.remove_colinear_vertices(tolerance) except AssertionError: # degenerate roof geometry to ignore continue clean_poly = Polygon2D(tuple(Point2D(pt.x, pt.y) for pt in r_geo.boundary)) clean_roofs.append(clean_geo) roof_polygons.append(clean_poly) roof_geometry = clean_roofs if len(roof_geometry) == 0: return # prepare the stories for checking the roofs rev_stories = list(reversed(self.unique_stories)) story_polygons, story_heights, room_heights = [], [], [] for story in rev_stories: room_polygons = tuple(rm.floor_geometry.polygon2d for rm in story.room_2ds) rm_heights = tuple(rm.floor_height for rm in story.room_2ds) story_polygons.append(room_polygons) story_heights.append(story.floor_height) room_heights.append(rm_heights) # loop through the roof_geometry and find a compatible story proj_dir = Vector3D(0, 0, 1) ot = overlap_threshold story_roofs = [[] for _ in rev_stories] # holds geo assigned to each story for rf_geo, rf_poly in zip(roof_geometry, roof_polygons): zip_obj = zip(story_heights, story_polygons, room_heights) for i, (st_ht, story_poly, rm_hts) in enumerate(zip_obj): if rf_geo.max.z < st_ht: continue # roof completely below story; valid assignment impossible overlaps_story = False for rm_poly, rm_ht in zip(story_poly, rm_hts): poly_rel = rf_poly.polygon_relationship(rm_poly, tolerance) if poly_rel >= 0: try: rm_poly = rm_poly.remove_colinear_vertices(tolerance) except AssertionError: # degenerate room to ignore continue try: overlap_polys = rf_poly.boolean_intersect(rm_poly, tolerance) \ if poly_rel == 0 else [rm_poly] except Exception: continue # not considered a significant overlap if sum(ply.area for ply in overlap_polys) < rm_poly.area * ot: continue # not considered a significant overlap plane_ints = [] for ov_poly in overlap_polys: for pt in ov_poly: r_ray = Ray3D(Point3D(pt.x, pt.y, rm_ht), proj_dir) plane_ints.append(rf_geo.plane.intersect_line_ray(r_ray)) if all(pi is not None for pi in plane_ints): overlaps_story = True else: # roof extends below room; valid assignment impossible overlaps_story = False break if overlaps_story: # we have found the story to assign the roof geometry story_roofs[i].append(rf_geo) break # create the RoofSpecification objects and assign them to the stories for story, roof_geos in zip(rev_stories, story_roofs): if len(roof_geos) != 0: if story.roof is not None: # combine the existing roof with the new one new_roof = RoofSpecification(story.roof.geometry + tuple(roof_geos)) else: new_roof = RoofSpecification(roof_geos) story.roof = new_roof
[docs] def remove_duplicate_roofs(self, tolerance=0.01): """Remove any roof geometries in the Building that appear more than once. This includes duplicated roof geometries assigned to different stories. Args: tolerance: The maximum difference between values at which point vertices are considered to be the same. (Default: 0.01, suitable for objects in Meters). """ # collect all roof geometries across all stories roof_geos = [] for story in self.unique_stories: if story.roof is not None: roof_geos.extend(story.roof.geometry) story.roof = None # remove duplicate geometries from the list clean_roof_geo = [] for r_geo in roof_geos: for exist_geo in clean_roof_geo: if r_geo.is_geometrically_equivalent(exist_geo, tolerance): break # duplicate geometry found else: # the geometry is not yet in the clean list clean_roof_geo.append(r_geo) # re-assign the roof geometry to the stories self.add_roof_geometry(clean_roof_geo, tolerance)
[docs] def convert_room_3d_to_2d(self, room_3d_identifier, tolerance=0.01): """Convert a single 3D Honeybee Room to a Dragonfly Room2D on this Building. This process will add the Room2D to an existing Dragonfly Story on the Building if the Honeybee Room.story matches a Story.display_name on this object. If not, a new Story on this Building will be initialized. Args: room_3d_identifier: The identifier of the 3D honeybee Room on this Building that will be converted to a dragonfly Room2D. tolerance: The maximum difference between values at which point vertices are considered to be the same. (Default: 0.01, suitable for objects in Meters). Returns: The newly-created Room2D object from the converted Room. Will be None if the Honeybee Room is not a closed solid and cannot be converted to a valid Room2D. """ # get the Honeybee Room object to be converted hb_room_i = [i for i, r in enumerate(self.room_3ds) if r.identifier == room_3d_identifier] if len(hb_room_i) == 0: raise ValueError( 'No 3D Honeybee Room with an identifier of "{}" was found on ' 'Building "{}"'.format(room_3d_identifier, self.display_name)) elif len(hb_room_i) != 1: raise ValueError( 'Multiple 3D Honeybee Rooms with an identifier of "{}" were found on ' 'Building "{}"'.format(room_3d_identifier, self.display_name)) new_room_3ds = list(self._room_3ds) hb_room = new_room_3ds.pop(hb_room_i[0]) # create a Dragonfly Room2D from the Honeybee Room try: df_room = Room2D.from_honeybee(hb_room, tolerance) except Exception: # room is not a closed solid return None self._room_3ds = tuple(new_room_3ds) # assign the Room2D to an existing Story or create a new one for story in self._unique_stories: if story.display_name == hb_room.story: story.add_room_2d(df_room) break else: # a new Story object has to be initialized new_story = Story(clean_string(hb_room.story), (df_room,)) new_story.display_name = hb_room.story self.add_stories([new_story]) return df_room
[docs] def convert_room_3ds_to_2d(self, room_3d_identifiers, tolerance=0.01): """Convert several 3D Honeybee Rooms on this Building to a Dragonfly Room2Ds. This process will add the Room2Ds to an existing Dragonfly Story on the Building if the Honeybee Room.story matches a Story.display_name on this object. If not, a new Story on this Building will be initialized. Args: room_3d_identifiers: A list of the identifiers for the 3D honeybee Rooms on this Building that will be converted to dragonfly Room2Ds. tolerance: The maximum difference between values at which point vertices are considered to be the same. (Default: 0.01, suitable for objects in Meters). Returns: A list of the newly-created Room2D objects from the converted Rooms. If a given 3D Room is not valid and cannot be converted to a Room2D, it will not be included in this output. """ df_rooms = [] for r3_id in room_3d_identifiers: new_r2 = self.convert_room_3d_to_2d(r3_id, tolerance) if new_r2 is not None: df_rooms.append(new_r2) return df_rooms
[docs] def convert_all_room_3ds_to_2d( self, extrusion_rooms_only=True, tolerance=0.01, angle_tolerance=1): """Convert all 3D Honeybee Rooms on this Building to a Dragonfly Room2Ds. This process will add the Room2Ds to an existing Dragonfly Story on the Building if the Honeybee Room.story matches a Story.display_name on this object. If not, a new Story on this Building will be initialized. Args: extrusion_rooms_only: A boolean to note whether only the 3D Rooms that can be represented as a Room2D without loss of geometry should be converted to Room2Ds. When True, all 3D Rooms that are not pure extrusions will be left as they are. If False, all 3D Rooms in the model will be translated to Room2D regardless of whether they are extrusions or not, meaning that there may be some loss of geometry or simplification of it. tolerance: The maximum difference between values at which point vertices are considered to be the same. (Default: 0.01, suitable for objects in Meters). angle_tolerance: The max angle difference in degrees that Face3D normals are allowed to differ from the vertical or horizontal before they are no longer considered as such. (Default: 1 degree). Returns: A list of the newly-created Room2D objects from the converted Rooms. """ # collect the relevant 3D Rooms if extrusion_rooms_only is selected new_room_3ds = [] if extrusion_rooms_only: hb_rooms = [] for hb_room in self.room_3ds: if self._is_room_3d_extruded(hb_room, tolerance, angle_tolerance): hb_rooms.append(hb_room) else: new_room_3ds.append(hb_room) else: hb_rooms = self.room_3ds # convert the relevant 3D Rooms to Room2D df_rooms = [] for hb_room in hb_rooms: # create a Dragonfly Room2D from the Honeybee Room try: df_room = Room2D.from_honeybee(hb_room, tolerance) except Exception: # invalid Honeybee Room that is not a closed solid new_room_3ds.append(hb_room) continue # assign the Room2D to an existing Story or create a new one for story in self._unique_stories: if story.display_name == hb_room.story: story.add_room_2d(df_room) break else: # a new Story object has to be initialized new_story = Story(clean_string(hb_room.story), (df_room,)) new_story.display_name = hb_room.story self.add_stories([new_story]) df_rooms.append(df_room) # reset the 3D Rooms on this object self._room_3ds = tuple(new_room_3ds) return df_rooms
[docs] def add_prefix(self, prefix): """Change the object identifier and all child objects by inserting a prefix. This is particularly useful in workflows where you duplicate and edit a starting object and then want to combine it with the original object into one Model (like making a model of repeating buildings) since all objects within a Model must have unique identifiers. Args: prefix: Text that will be inserted at the start of this object's (and child objects') identifier and display_name. It is recommended that this prefix be short to avoid maxing out the 100 allowable characters for dragonfly identifiers. """ self.identifier = clean_string('{}_{}'.format(prefix, self.identifier)) if self._display_name is not None: self.display_name = '{}_{}'.format(prefix, self.display_name) self.properties.add_prefix(prefix) for story in self.unique_stories: story.add_prefix(prefix) for room in self.room_3ds: room.add_prefix(prefix)
[docs] def sort_stories(self): """Sort the stories assigned to this Building by their floor heights""" self._unique_stories = \ tuple(sorted(self._unique_stories, key=lambda x: x.floor_height))
[docs] def separate_top_bottom_floors(self): """Separate top/bottom Stories with non-unity multipliers into their own Stories. The resulting first and last Stories will each have a multiplier of 1 and duplicated middle Stories will be added as needed. This method also automatically assigns the first story Room2Ds to have a ground contact floor and the top story Room2Ds to have an outdoor-exposed roof. This is particularly helpful when using to_honeybee workflows with multipliers but one wants to account for the heat exchange of the top or bottom floors with the ground or outdoors. """ # do not do anything if the Building has no 2D Stories if not self.has_room_2ds: return # empty tuples in case no floors are added new_ground_floor, new_top_floor = (), () # ensure that the bottom floor is unique if self._unique_stories[0].multiplier != 1: story = self._unique_stories[0] new_ground_floor = (self._separated_ground_floor(story),) story.multiplier = story.multiplier - 1 story.move(Vector3D(0, 0, story.floor_to_floor_height)) # 2nd floor # ensure that the top floor is unique if self._unique_stories[-1].multiplier != 1: story = self._unique_stories[-1] new_top_floor = (self._separated_top_floor(story),) story.multiplier = story.multiplier - 1 # set the unique stories to include any new top and bottom floors self._unique_stories = new_ground_floor + self._unique_stories + new_top_floor # assign the is_ground_contact and is_top_exposed properties self._unique_stories[0].set_ground_contact() self._unique_stories[-1].set_top_exposed()
[docs] def separate_mid_floors(self, tolerance=0.01): """Separate all Stories with non-unity multipliers into two or three Stories. This method automatically assigns the first story Room2Ds to have a ground contact floor and will separate the top story of each unique story to have outdoor-exposed roofs when no Room2Ds are sensed above a given room. This is particularly helpful when using to_honeybee workflows with multipliers but one wants to account for the heat exchange of the top or bottom floors with the ground or outdoors. Args: tolerance: The tolerance that will be used to compute the point within the floor boundary that is used to check whether there is geometry above each Room2D. It is recommended that this number not be less than 1 centimeter to avoid long computation times. Default: 0.01, suitable for objects in meters. """ # do not do anything if the Building has no 2D Stories if not self.has_room_2ds: return # ensure that the bottom floor is unique if self._unique_stories[0].multiplier != 1: story = self._unique_stories[0] new_ground_floor = self._separated_ground_floor(story) story.multiplier = story.multiplier - 1 story.move(Vector3D(0, 0, story.floor_to_floor_height)) # 2nd floor else: new_ground_floor = self._unique_stories[0] if len(self._unique_stories) > 1: new_ground_floor.set_top_exposed_by_story_above( self._unique_stories[1], tolerance) self._unique_stories = self._unique_stories[1:] # ensure that the top floor is unique new_top_floors = [] for i, story in enumerate(self._unique_stories): if story.multiplier != 1: new_top_floor = self._separated_top_floor(story) story.multiplier = story.multiplier - 1 try: new_top_floor.set_top_exposed_by_story_above( self._unique_stories[i + 1], tolerance) except IndexError: # this is the last story new_top_floor.set_top_exposed() new_top_floors.extend((story, new_top_floor)) else: if i == len(self._unique_stories) - 1: story.set_top_exposed() else: story.set_top_exposed_by_story_above( self._unique_stories[i + 1], tolerance) new_top_floors.append(story) # set the unique stories to include any new top and bottom floors self._unique_stories = (new_ground_floor,) + tuple(new_top_floors) # assign the is_ground_contact and is_top_exposed properties self._unique_stories[0].set_ground_contact()
[docs] def split_room_2d_vertically(self, room_id, tolerance=0.01): """Split a Room2D in this Building vertically if it crosses multiple stories. Args: room_id: The identifier of a Room2D within this Building which will be split vertically with the Stories above it. tolerance: The tolerance to be used for determining whether the Room2D should be split. Default: 0.01, suitable for objects in meters. Returns: A list of all the new rooms created by running the method. This can be used to post-process the rooms for attributes like adjacency within the Story they are placed. """ # loop through the stories of the model and find the Room2D found_room, split_heights, split_stories = None, [], [] for story in self._unique_stories: if found_room is not None: flr_hgt = story.median_room2d_floor_height if found_room.ceiling_height - tolerance > flr_hgt: split_heights.append(flr_hgt) split_stories.append(story) else: for rm in story.room_2ds: if rm.identifier == room_id: found_room = rm break # check if the room was found and whether it should be split if found_room is None: msg = 'No Room2D with the identifier "{}" was found in the ' \ 'Building.'.format(room_id) raise ValueError(msg) if len(split_heights) == 0: return [] # no splitting to be done # split the room across the stories new_rooms = [] for i, (split_hgt, add_story) in enumerate(zip(split_heights, split_stories)): new_room = found_room.duplicate() new_room.identifier = '{}_split{}'.format(new_room.identifier, i) shift_dist = split_hgt - found_room.floor_height move_vec = Vector3D(0, 0, shift_dist) new_room.move(move_vec) # move the room to the correct floor height try: new_ceil_hgt = split_heights[i + 1] new_room.is_top_exposed = False new_room.has_ceiling = False except IndexError: # last story of the split list new_ceil_hgt = found_room.ceiling_height new_room.floor_to_ceiling_height = new_ceil_hgt - new_room.floor_height new_room.is_ground_contact = False new_room.has_floor = False new_w_par = [] # shift all of the window parameters for the room for wp, seg in zip(found_room.window_parameters, found_room.floor_segments): if isinstance(wp, _AsymmetricBase): wp = wp.shift_vertically(-shift_dist) wp.adjust_for_segment(seg, new_ceil_hgt, tolerance) new_w_par.append(wp) new_room.window_parameters = new_w_par add_story.add_room_2d(new_room) new_rooms.append(new_room) # change the height of the original Room2D so that it doesn't overlap new rooms found_room.floor_to_ceiling_height = split_heights[0] - found_room.floor_height found_room.is_top_exposed = False found_room.has_ceiling = False new_w_par = [] # shift all of the window parameters for the room for wp, seg in zip(found_room.window_parameters, found_room.floor_segments): if isinstance(wp, _AsymmetricBase): wp.adjust_for_segment(seg, found_room.floor_to_ceiling_height, tolerance) new_w_par.append(wp) found_room.window_parameters = new_w_par # move any roofs if need be if found_room.parent.roof is not None: kept_roofs, moved_roofs = [], [] roof = found_room.parent.roof for r_geo, r_poly in zip(roof.geometry, roof.boundary_geometry_2d): room_poly = found_room.floor_geometry.boundary_polygon2d if room_poly.polygon_relationship(r_poly, tolerance) >= 0: moved_roofs.append(r_geo) else: kept_roofs.append(r_geo) if len(moved_roofs) != 0: if len(kept_roofs) != 0: found_room.parent.roof = RoofSpecification(kept_roofs) else: found_room.parent.roof = None if new_rooms[-1].parent.roof is None: new_rooms[-1].parent.roof = RoofSpecification(moved_roofs) else: new_geo = new_rooms[-1].parent.roof.geometry + tuple(moved_roofs) new_rooms[-1].parent.roof = RoofSpecification(new_geo) return new_rooms
[docs] def set_outdoor_window_parameters(self, window_parameter): """Set all of the outdoor walls to have the same window parameters.""" for story in self._unique_stories: story.set_outdoor_window_parameters(window_parameter)
[docs] def set_outdoor_shading_parameters(self, shading_parameter): """Set all of the outdoor walls to have the same shading parameters.""" for story in self._unique_stories: story.set_outdoor_shading_parameters(shading_parameter)
[docs] def to_rectangular_windows(self): """Convert all of the windows of the Story to the RectangularWindows format.""" for story in self._unique_stories: story.to_rectangular_windows()
[docs] def move(self, moving_vec): """Move this Building along a vector. Args: moving_vec: A ladybug_geometry Vector3D with the direction and distance to move the object. """ for story in self._unique_stories: story.move(moving_vec) for room in self._room_3ds: room.move(moving_vec) self.properties.move(moving_vec)
[docs] def rotate_xy(self, angle, origin): """Rotate this Building counterclockwise in the XY plane by a certain angle. Args: angle: An angle in degrees. origin: A ladybug_geometry Point3D for the origin around which the object will be rotated. """ for story in self._unique_stories: story.rotate_xy(angle, origin) for room in self._room_3ds: room.rotate_xy(angle, origin) self.properties.rotate_xy(angle, origin)
[docs] def reflect(self, plane): """Reflect this Building across a plane. Args: plane: A ladybug_geometry Plane across which the object will be reflected. """ for story in self._unique_stories: story.reflect(plane) for room in self._room_3ds: room.reflect(plane) self.properties.reflect(plane)
[docs] def scale(self, factor, origin=None): """Scale this Building by a factor from an origin point. 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 story in self._unique_stories: story.scale(factor, origin) for room in self._room_3ds: room.scale(factor, origin) self.properties.scale(factor, origin)
[docs] def has_floors_ceilings(self, use_multiplier=True): """Get a list of tuples for each Room noting whether it has a floor/ceiling. Args: use_multiplier: Boolean to note whether the returned list should be assume that the Rooms use multipliers or not. The list will typically be longer when use_multiplier is False. (Default: True). """ has_flr_ceil = [] for story in self._unique_stories: story_list = [] for room in story.room_2ds: story_list.append((room.has_floor, room.has_ceiling)) if use_multiplier: has_flr_ceil.extend(story_list) else: for _ in range(story.multiplier): has_flr_ceil.extend(story_list) return has_flr_ceil
[docs] def to_honeybee(self, use_multiplier=True, add_plenum=False, tolerance=0.01, enforce_adj=True, enforce_solid=True): """Convert Dragonfly Building to a Honeybee Model. Args: use_multiplier: If True, the multipliers on this Building's Stories will be passed along to the generated Honeybee Room objects, indicating the simulation will be run once for each unique room and then results will be multiplied. If False, full geometry objects will be written for each and every floor in the building that are represented through multipliers and all resulting multipliers will be 1. (Default: True). add_plenum: Boolean to indicate whether ceiling/floor plenums should be auto-generated for the Rooms. (Default: False). tolerance: The minimum distance in z values of floor_height and floor_to_ceiling_height at which adjacent Faces will be split. Default: 0.01, suitable for objects in meters. enforce_adj: Boolean to note whether an exception should be raised if an adjacency between two Room2Ds is invalid (True) or if the invalid Surface boundary condition should be replaced with an Outdoor boundary condition (False). If False, any Walls containing WindowParameters and an illegal boundary condition will also be replaced with an Outdoor boundary condition. (Default: True). enforce_solid: Boolean to note whether rooms should be translated as solid extrusions whenever translating them with custom roof geometry produces a non-solid result (True) or the non-solid room geometry should be allowed to remain in the result (False). The latter is useful for understanding why a particular roof geometry has produced a non-solid result. (Default: True). Returns: A honeybee Model that represent the Building. """ # compute the story heights once so they're not constantly recomputed reset_roofs = False if self._roofs is None: self._roofs = self._compute_roof_heights() reset_roofs = True # generate all of the Honeybee Rooms hb_rooms = [] if use_multiplier: for story in self._unique_stories: hb_rooms.extend(story.to_honeybee( True, add_plenum=add_plenum, tolerance=tolerance, enforce_adj=enforce_adj, enforce_solid=enforce_solid)) else: for story in self.all_stories(): hb_rooms.extend(story.to_honeybee( False, add_plenum=add_plenum, tolerance=tolerance, enforce_adj=enforce_adj, enforce_solid=enforce_solid)) for room in self.room_3ds: hb_rooms.append(room) hb_mod = Model(self.identifier, hb_rooms) hb_mod._display_name = self._display_name hb_mod._user_data = self._user_data # put back the old roofs if they were not set originally if reset_roofs: self._roofs = None return hb_mod
[docs] def to_dict(self, abridged=False, included_prop=None): """Return Building as a dictionary. Args: abridged: Boolean to note whether the extension properties of the object (ie. construction sets) should be included in detail (False) or just referenced by identifier (True). Default: False. included_prop: List of properties to filter keys that must be included in output dictionary. For example ['energy'] will include 'energy' key if available in properties to_dict. By default all the keys will be included. To exclude all the keys from extensions use an empty list. """ base = {'type': 'Building'} base['identifier'] = self.identifier base['display_name'] = self.display_name if len(self._unique_stories) != 0: base['unique_stories'] = [s.to_dict(abridged, included_prop) for s in self._unique_stories] if len(self._room_3ds) != 0: base['room_3ds'] = [r.to_dict(abridged, included_prop) for r in self._room_3ds] base['properties'] = self.properties.to_dict(abridged, included_prop) if self.user_data is not None: base['user_data'] = self.user_data if self._roofs is not None: # secret key used for filtered dictionaries rf_dicts = [] for st_id, _, roof in self._roofs: rf_dicts.append((st_id, roof.to_dict())) base['_roofs'] = rf_dicts return base
@property def to(self): """Building writer object. Use this method to access Writer class to write the building in other formats. """ return writer
[docs] @staticmethod def process_alleys(buildings, distance=1.0, adiabatic=False, tolerance=0.01): """Remove windows from any walls that within a distance of other buildings. This method can also optionally set the boundary conditions of these walls to adiabatic. This is helpful when attempting to account for alleys or parti walls that may exist between buildings of a denser urban district. Note that this staticmethod will edit the buildings in place so it may be appropriate to duplicate the Buildings before running this method. Args: buildings: Dragonfly Building objects which will have their windows removed if their walls lie within the distance of another building. distance: A number for the maximum distance of an alleyway in model units. If a wall is closer to another Building than this distance, the windows will be removed. (Default: 1.0; suitable for objects in meters). adiabatic: A boolean to note whether the walls that have their windows removed should also receive an Adiabatic boundary condition. This is useful when the alleyways are more like parti walls than distinct pathways that someone could traverse. """ # get the adiabatic boundary condition in case we need it try: ad_bc = bcs.adiabatic except AttributeError: # honeybee_energy is not loaded ad_bc = bcs.outdoors if not adiabatic else bcs.ground # get the footprints, heights and bounding points of all of the buildings story_heights, story_polys = [], [] for bldg in buildings: bldg_polys, bldg_s_hgts = [], [] for story in bldg.unique_stories: flr_hgt = story.floor_height bldg_s_hgts.append((flr_hgt, flr_hgt + story.floor_to_floor_height)) story_foot = story.footprint(tolerance) st_poly = [Polygon2D((Point2D(p.x, p.y) for p in face.vertices)) for face in story_foot] bldg_polys.append(st_poly) story_heights.append(bldg_s_hgts) story_polys.append(bldg_polys) bldg_heights = [b.height for b in buildings] bldg_pts = [] for bldg in buildings: b_min, b_max = bldg.min, bldg.max center = Point2D((b_min.x + b_max.x) / 2, (b_min.y + b_max.y) / 2) bldg_pts.append((b_min, center, b_max)) # loop through the buildings and set the properties of the relevant walls for i, bldg in enumerate(buildings): # first determine the relevant buildings and building heights rel_st_polys, rel_st_heights, rel_b_heights = [], [], [] other_indices = list(range(i)) + list(range(i + 1, len(buildings))) for j in other_indices: if Building._bound_rect_in_dist(bldg_pts[i], bldg_pts[j], distance): rel_st_polys.append(story_polys[j]) rel_st_heights.append(story_heights[j]) rel_b_heights.append(bldg_heights[j]) # then, loop through the story Room2Ds and set properties of relevant walls for story in bldg.unique_stories: st_hgt, st_f2f = story.floor_height, story.floor_to_floor_height st_c_hgt = st_hgt + st_f2f for rm in story.room_2ds: zip_r_objs = zip(rm.boundary_conditions, rm.floor_segments_2d, rm.segment_normals) new_bcs = list(rm.boundary_conditions) new_win_pars = list(rm.window_parameters) for k, (bc, seg, normal) in enumerate(zip_r_objs): if not isinstance(bc, Outdoors): # nothing to change continue seg_mid = seg.midpoint.move(normal * -tolerance) seg_ray = LineSegment2D.from_sdl(seg_mid, normal, distance) zip_b_objs = zip(rel_b_heights, rel_st_polys, rel_st_heights) for bh, rel_poly, rel_hgt in zip_b_objs: if st_hgt >= bh - tolerance: continue # story above other bldg; we can ignore it for o_story, o_h in zip(rel_poly, rel_hgt): overlap = min((o_h[1], st_c_hgt)) - max((o_h[0], st_hgt)) if overlap >= st_f2f * 0.33: # more than 1/3 overlap for o_poly in o_story: if len(o_poly.intersect_line_ray(seg_ray)) > 0: # we have found an alleyway! new_win_pars[k] = None if adiabatic: new_bcs[k] = ad_bc break # assign the new window parameters and boundary conditions rm.window_parameters = new_win_pars rm.boundary_conditions = new_bcs
[docs] @staticmethod def district_to_honeybee( buildings, use_multiplier=True, add_plenum=False, tolerance=0.01, enforce_adj=True, enforce_solid=True): """Convert an array of Building objects into a single district honeybee Model. Args: buildings: An array of Building objects to be converted into a honeybee Model. use_multiplier: If True, the multipliers on this Building's Stories will be passed along to the generated Honeybee Room objects, indicating the simulation will be run once for each unique room and then results will be multiplied. If False, full geometry objects will be written for each and every floor in the building that are represented through multipliers and all resulting multipliers will be 1. (Default: True). add_plenum: Boolean to indicate whether ceiling/floor plenums should be auto-generated for the Rooms. (Default: False). tolerance: The minimum distance in z values of floor_height and floor_to_ceiling_height at which adjacent Faces will be split. Default: 0.01, suitable for objects in meters. enforce_adj: Boolean to note whether an exception should be raised if an adjacency between two Room2Ds is invalid (True) or if the invalid Surface boundary condition should be replaced with an Outdoor boundary condition (False). If False, any Walls containing WindowParameters and an illegal boundary condition will also be replaced with an Outdoor boundary condition. (Default: True). enforce_solid: Boolean to note whether rooms should be translated as solid extrusions whenever translating them with custom roof geometry produces a non-solid result (True) or the non-solid room geometry should be allowed to remain in the result (False). The latter is useful for understanding why a particular roof geometry has produced a non-solid result. (Default: True). Returns: A honeybee Model that represent the district. """ # create a base model to which everything will be added base_model = buildings[0].to_honeybee( use_multiplier, add_plenum=add_plenum, tolerance=tolerance, enforce_adj=enforce_adj, enforce_solid=enforce_solid) # loop through each Building, create a model, and add it to the base one for bldg in buildings[1:]: base_model.add_model(bldg.to_honeybee( use_multiplier, add_plenum=add_plenum, tolerance=tolerance, enforce_adj=enforce_adj, enforce_solid=enforce_solid)) return base_model
[docs] @staticmethod def buildings_to_honeybee( buildings, context_shades=None, shade_distance=None, use_multiplier=True, add_plenum=False, cap=False, tolerance=0.01, enforce_adj=True, enforce_solid=True): """Convert an array of Buildings into several honeybee Models with self-shading. Each input Building will be exported into its own Model. For each Model, the other input Buildings will appear as context shade geometry. Thus, each Model is its own simulate-able unit accounting for the total self-shading of the input Buildings. Args: buildings: An array of Building objects to be converted into honeybee Models that account for their own shading of one another. context_shades: An optional array of ContextShade objects that will be added to the honeybee Models if their bounding box overlaps with a given building within the shade_distance. shade_distance: An optional number to note the distance beyond which other objects' shade should not be exported into a given Model. This is helpful for reducing the simulation run time of each Model when other connected buildings are too far away to have a meaningful impact on the results. If None, all other buildings will be included as context shade in each and every Model. Set to 0 to exclude all neighboring buildings from the resulting models. Default: None. use_multiplier: If True, the multipliers on this Building's Stories will be passed along to the generated Honeybee Room objects, indicating the simulation will be run once for each unique room and then results will be multiplied. If False, full geometry objects will be written for each and every floor in the building that are represented through multipliers and all room multipliers will be 1. (Default: True). add_plenum: Boolean to indicate whether ceiling/floor plenums should be auto-generated for the Rooms. (Default: False). cap: Boolean to note whether building shade representations should be capped with a top face. Usually, this is not necessary to account for blocked sun and is only needed when it's important to account for reflected sun off of roofs. (Default: False). tolerance: The minimum distance in z values of floor_height and floor_to_ceiling_height at which adjacent Faces will be split. Default: 0.01, suitable for objects in meters. enforce_adj: Boolean to note whether an exception should be raised if an adjacency between two Room2Ds is invalid (True) or if the invalid Surface boundary condition should be replaced with an Outdoor boundary condition (False). If False, any Walls containing WindowParameters and an illegal boundary condition will also be replaced with an Outdoor boundary condition. (Default: True). enforce_solid: Boolean to note whether rooms should be translated as solid extrusions whenever translating them with custom roof geometry produces a non-solid result (True) or the non-solid room geometry should be allowed to remain in the result (False). The latter is useful for understanding why a particular roof geometry has produced a non-solid result. (Default: True). Returns: A list of honeybee Models that represent the Building. """ # create lists with all context representations of the buildings + shade bldg_shades, bldg_pts, con_shades, con_pts = Building._honeybee_shades( buildings, context_shades, shade_distance, cap, tolerance) # loop through each Building and create a model models = [] # list to be filled with Honeybee Models num_bldg = len(buildings) for i, bldg in enumerate(buildings): model = bldg.to_honeybee( use_multiplier, add_plenum=add_plenum, tolerance=tolerance, enforce_adj=enforce_adj, enforce_solid=enforce_solid) Building._add_context_to_honeybee(model, bldg_shades, bldg_pts, con_shades, con_pts, shade_distance, num_bldg, i) models.append(model) # append to the final list of Models return models
[docs] @staticmethod def stories_to_honeybee( buildings, context_shades=None, shade_distance=None, use_multiplier=True, add_plenum=False, cap=False, tolerance=0.01, enforce_adj=True, enforce_solid=True): """Convert an array of Buildings into one honeybee Model per story. Each Story of each input Building will be exported into its own Model. For each Honeybee Model, the other input Buildings will appear as context shade geometry as will all of the other stories of the same building. Thus, each Model is its own simulate-able unit accounting for the total self-shading of the input Buildings. Args: buildings: An array of Building objects to be converted into an array of honeybee Models with one story per model. context_shades: An optional array of ContextShade objects that will be added to the honeybee Models if their bounding box overlaps with a given building within the shade_distance. shade_distance: An optional number to note the distance beyond which other objects' shade should not be exported into a given Model. This is helpful for reducing the simulation run time of each Model when other connected buildings are too far away to have a meaningful impact on the results. If None, all other buildings will be included as context shade in each and every Model. Set to 0 to exclude all neighboring buildings from the resulting models. Default: None. use_multiplier: If True, the multipliers on this Building's Stories will be passed along to the generated Honeybee Room objects, indicating the simulation will be run once for each unique room and then results will be multiplied. If False, full geometry objects will be written for each and every floor in the building that are represented through multipliers and all room multipliers will be 1. (Default: True). add_plenum: Boolean to indicate whether ceiling/floor plenums should be auto-generated for the Rooms. (Default: False). cap: Boolean to note whether building shade representations should be capped with a top face. Usually, this is not necessary to account for blocked sun and is only needed when it's important to account for reflected sun off of roofs. (Default: False). tolerance: The minimum distance in z values of floor_height and floor_to_ceiling_height at which adjacent Faces will be split. Default: 0.01, suitable for objects in meters. enforce_adj: Boolean to note whether an exception should be raised if an adjacency between two Room2Ds is invalid (True) or if the invalid Surface boundary condition should be replaced with an Outdoor boundary condition (False). If False, any Walls containing WindowParameters and an illegal boundary condition will also be replaced with an Outdoor boundary condition. (Default: True). enforce_solid: Boolean to note whether rooms should be translated as solid extrusions whenever translating them with custom roof geometry produces a non-solid result (True) or the non-solid room geometry should be allowed to remain in the result (False). The latter is useful for understanding why a particular roof geometry has produced a non-solid result. (Default: True). Returns: A list of honeybee Models that represent the Stories. """ # create lists with all context representations of the buildings + shade bldg_shades, bldg_pts, con_shades, con_pts = Building._honeybee_shades( buildings, context_shades, shade_distance, cap, tolerance) # loop through each Building and create a model models = [] # list to be filled with Honeybee Models num_bldg = len(buildings) for i, bldg in enumerate(buildings): dummy_model = Model(bldg.identifier) # blank model to hold context shade Building._add_context_to_honeybee( dummy_model, bldg_shades, bldg_pts, con_shades, con_pts, shade_distance, num_bldg, i) bldg_con = list(dummy_model.orphaned_shades) if use_multiplier: for j, story in enumerate(bldg.unique_stories): hb_rooms = story.to_honeybee( True, add_plenum, tolerance=tolerance, enforce_adj=enforce_adj, enforce_solid=enforce_solid) if bldg.has_room_3ds: hb_rooms.extend(bldg.room_3ds_by_story(story.display_name)) shds = bldg_con + bldg.shade_representation(j, cap, False, tolerance) model = Model(story.identifier, hb_rooms, orphaned_shades=shds) model.display_name = story.display_name models.append(model) # append to the final list of Models else: self_shds = [story.shade_representation(cap, tolerance) for story in bldg.unique_stories] full_shades = [] for j, story in enumerate(bldg.unique_stories): for k in range(story.multiplier): mult_shd = story.shade_representation_multiplier( k, cap=cap, tolerance=tolerance) mult_shd.extend([s for s_ar in self_shds[:j] for s in s_ar]) mult_shd.extend([s for s_ar in self_shds[j + 1:] for s in s_ar]) full_shades.append(mult_shd) for story, shades in zip(bldg.all_stories(), full_shades): hb_rooms = story.to_honeybee( True, add_plenum, tolerance=tolerance, enforce_adj=enforce_adj, enforce_solid=enforce_solid) if bldg.has_room_3ds: hb_rooms.extend(bldg.room_3ds_by_story(story.display_name)) shds = bldg_con + shades model = Model(story.identifier, hb_rooms, orphaned_shades=shds) model.display_name = story.display_name models.append(model) # append to the final list of Models if bldg.has_room_3ds: # organize them by story and add them accounted_for = bldg.room_2d_story_names r3_story_dict = bldg._story_dict_room_3d() shds = bldg_con + bldg.shade_representation( None, cap, False, tolerance) for story_id, hb_rooms in r3_story_dict.items(): if story_id not in accounted_for: model = Model(story_id, hb_rooms, orphaned_shades=shds) models.append(model) # append to the final list of Models return models
def _compute_roof_heights(self): """Get a list with the center height of each RoofSpecification in the Building. This method is used internally during Honeybee serialization. """ roof_specs = [] for story in self._unique_stories: if story.roof is not None: rf_height = (story.roof.max_height + story.roof.min_height) / 2 roof_specs.append((story.identifier, rf_height, story.roof)) else: roof_specs.append((story.identifier, None, None)) return roof_specs def _story_roofs(self, story): """Get a list of RoofSpecifications that are relevant for a given Story. The returned list will contain tuples where the first item is the center height of the roof and the second item is the RoofSpecification object. This is used under the hood to determine whether roofs of other Stories should influence a given Room2D. Args: story: A Story object within the Building. """ # compute the center height of each roof specification bldg_roofs = self._compute_roof_heights() if self._roofs is None else self._roofs # filter out the roofs for the relevant story rel_roofs, story_found = [], False for st_id, hgt, roof in bldg_roofs: if story_found and roof is not None: rel_roofs.append((hgt, roof)) if story.identifier == st_id: story_found = True return rel_roofs def _story_dict_room_3d(self): """Get a dictionary of 3D Honeybee Rooms organized by story.""" r3_story_dict = {} for room in self._room_3ds: try: r3_story_dict[room.story].append(room) except KeyError: r3_story_dict[room.story] = [room] return r3_story_dict def _lowest_story_room_3ds(self): """Get a list of Honeybee Rooms for the lowest story of the Building. Note that this method should typically only be used when the Building is composed entirely of 3D Honeybee Rooms. """ r3_story_dict = self._story_dict_room_3d() floor_hgts, floor_rooms = [], [] for rooms in r3_story_dict.values(): flr_hgt = sum(r.average_floor_height for r in rooms) / len(rooms) floor_hgts.append(flr_hgt) floor_rooms.append(rooms) sort_rooms = [rs for _, rs in sorted(zip(floor_hgts, floor_rooms), key=lambda pair: pair[0])] return sort_rooms[0] @staticmethod def _is_room_3d_extruded(hb_room, tolerance, angle_tolerance): """Test if a 3D Room is a pure extrusion. Pure extrusions can be converted into Room2Ds without any loss or simplification of geometry. Args: hb_room: The 3D Honeybee Room to be tested. tolerance: The absolute tolerance with which the Room geometry will be evaluated. angle_tolerance: The angle tolerance at which the geometry will be evaluated in degrees. Returns: True if the 3D Room is a pure extrusion. False if not. """ # set up the parameters for evaluating vertical or horizontal vert_vec = Vector3D(0, 0, 1) min_v_ang = math.radians(angle_tolerance) max_v_ang = math.pi - min_v_ang min_h_ang = (math.pi / 2) - min_v_ang max_h_ang = (math.pi / 2) + min_v_ang # loop through the 3D Room faces and test them for face in hb_room._faces: try: # first make sure that the geometry is not degenerate clean_geo = face.geometry.remove_colinear_vertices(tolerance) v_ang = clean_geo.normal.angle(vert_vec) if v_ang <= min_v_ang or v_ang >= max_v_ang: continue elif min_h_ang <= v_ang <= max_h_ang: continue return False except AssertionError: # degenerate face to ignore pass return True @staticmethod def _honeybee_shades(buildings, context_shades, shade_distance, cap, tolerance): """Get lists of Honeybee shades from Building and ContextShade objects.""" bldg_shades, bldg_pts = [], [] con_shades, con_pts = [], [] if shade_distance is None or shade_distance > 0: for bldg in buildings: b_shades = bldg.shade_representation( cap=cap, include_room3ds=True, tolerance=tolerance) bldg_shades.append(b_shades) b_min, b_max = bldg.min, bldg.max center = Point2D((b_min.x + b_max.x) / 2, (b_min.y + b_max.y) / 2) bldg_pts.append((b_min, center, b_max)) if context_shades is not None: for con in context_shades: con_shades.append(con.to_honeybee()) c_min, c_max = con.min, con.max center = Point2D((c_min.x + c_max.x) / 2, (c_min.y + c_max.y) / 2) con_pts.append((c_min, center, c_max)) return bldg_shades, bldg_pts, con_shades, con_pts @staticmethod def _add_context_to_honeybee(model, bldg_shades, bldg_pts, con_shades, con_pts, shade_distance, num_bldg, i): """Add context shades to a Honeybee Model based on shade distance.""" if shade_distance is None: # add all other bldg shades to the model for j in xrange(i + 1, num_bldg): # buildings before this one for shd in bldg_shades[j]: model.add_shade(shd) for k in xrange(i): # buildings after this one for shd in bldg_shades[k]: model.add_shade(shd) for c_shade in con_shades: # context shades for shd in c_shade: if isinstance(shd, Shade): model.add_shade(shd) else: model.add_shade_mesh(shd) elif shade_distance > 0: # add only shade within the distance for j in xrange(i + 1, num_bldg): # buildings before this one if Building._bound_rect_in_dist(bldg_pts[i], bldg_pts[j], shade_distance): for shd in bldg_shades[j]: model.add_shade(shd) for k in xrange(i): # buildings after this one if Building._bound_rect_in_dist(bldg_pts[i], bldg_pts[k], shade_distance): for shd in bldg_shades[k]: model.add_shade(shd) for s in xrange(len(con_shades)): # context shades if Building._bound_rect_in_dist(bldg_pts[i], con_pts[s], shade_distance): for shd in con_shades[s]: if isinstance(shd, Shade): model.add_shade(shd) else: model.add_shade_mesh(shd) @staticmethod def _generate_room_2ds(face3d_array, flr_to_ceiling, perim_offset, bldg_id, flr_count, tolerance): """Generate Room2D objects given geometry and information about their parent. Args: face3d_array: An array of Face3D objects to be turned into a Story's Room2Ds. flr_to_ceiling: The floor-to-ceiling height to use for all the Room2Ds. perim_offset: A perimeter offset to be used to subdivide Face3Ds bldg_id: Text for the identifier to which the rooms belong. flr_count: Integer for the which story the building belongs to. tolerance: Tolerance to be used in the creation of the Room2Ds. """ # if there is a non-zero perimeter offset, separate core vs. perimeter zones if perim_offset != 0: assert perim_offset > 0, 'perimeter_offset cannot be less than than 0.' new_face3d_array = [] for floor_face in face3d_array: try: floor_face = floor_face.remove_colinear_vertices(tolerance) perimeter, core = perimeter_core_subfaces( floor_face, perim_offset, tolerance) new_face3d_array.extend(perimeter) new_face3d_array.extend(core) except Exception as e: # the generation of the polyskel failed print('Core/perimeter generation failed:\n{}'.format(e)) new_face3d_array.append(floor_face) # just use existing floor face3d_array = new_face3d_array # replace with offset core/perimeter # create the Room2D objects room_2ds = [] for i, room_geo in enumerate(face3d_array): room = Room2D('{}_Floor{}_Room{}'.format(bldg_id, flr_count, i + 1), room_geo, flr_to_ceiling, tolerance=tolerance) room_2ds.append(room) # solve for interior adjacency if there core/perimeter zoning was requested if perim_offset != 0: room_2ds = Room2D.intersect_adjacency( room_2ds, tolerance, preserve_wall_props=False) Room2D.solve_adjacency(room_2ds, tolerance) return room_2ds @staticmethod def _is_story_equivalent(face1, face2, tolerance): """Check whether area, XY centerpoint and XY first point match between Face3D. Args: face1: First Face3D to check. face2: Second Face3D to check. tolerance: The maximum difference between x, y, and z values at which point vertices are considered to be the same. Returns: True if face1 is geometrically equivalent to face 2 else False. """ # check wether the center points match within tolerance. cent1 = face1.center cent2 = face2.center if abs(cent1.x - cent2.x) > tolerance or abs(cent1.y - cent2.y) > tolerance: return False # check wether the point at start matches within tolerance start1 = face1[0] start2 = face2[0] if abs(start1.x - start2.x) > tolerance or abs(start1.y - start2.y) > tolerance: return False # check whether areas match within tolerance area_tol = tolerance ** 2 if abs(face1.area - face2.area) > area_tol: return False return True @staticmethod def _bound_rect_in_dist(bound_pts1, bound_pts2, distance): """Check if the bounding rectangles of two footprints overlap within a distance. Checking the overlap of the bounding rectangles is extremely quick given this method's use of the Separating Axis Theorem. Args: bound_pts1: An array of Point2Ds (min, center, max) for the first footprint. bound_pts2: An array of Point2Ds (min, center, max) for the second footprint. distance: Acceptable distance between the two bounding rectangles. """ # Bounding rectangle check using the Separating Axis Theorem polygon1_width = bound_pts1[2].x - bound_pts1[0].x polygon2_width = bound_pts2[2].x - bound_pts2[0].x dist_btwn_x = abs(bound_pts1[1].x - bound_pts2[1].x) x_gap_btwn_rect = dist_btwn_x - (0.5 * polygon1_width) - (0.5 * polygon2_width) polygon1_height = bound_pts1[2].y - bound_pts1[0].y polygon2_height = bound_pts2[2].y - bound_pts2[0].y dist_btwn_y = abs(bound_pts1[1].y - bound_pts2[1].y) y_gap_btwn_rect = dist_btwn_y - (0.5 * polygon1_height) - (0.5 * polygon2_height) if x_gap_btwn_rect > distance or y_gap_btwn_rect > distance: return False # no overlap return True # overlap exists @staticmethod def _separated_ground_floor(base_story): """Get a separated ground floor from a base_story.""" bottom = base_story.duplicate() # generate a new bottom floor bottom.multiplier = 1 bottom.add_prefix('Ground') return bottom @staticmethod def _separated_top_floor(base_story): """Get a separated top floor from a base_story.""" top = base_story.duplicate() # generate a new top floor move_vec = Vector3D(0, 0, top.floor_to_floor_height * (top.multiplier - 1)) top.move(move_vec) top.multiplier = 1 top.add_prefix('Top') return top def __copy__(self): new_b = Building( self.identifier, tuple(story.duplicate() for story in self._unique_stories), tuple(room.duplicate() for room in self._room_3ds)) new_b._display_name = self._display_name new_b._user_data = None if self.user_data is None else self.user_data.copy() new_b._properties._duplicate_extension_attr(self._properties) return new_b def __len__(self): return len(self._unique_stories) def __getitem__(self, key): return self._unique_stories[key] def __iter__(self): return iter(self._unique_stories) def __repr__(self): return 'Building: %s' % self.display_name