Source code for honeybee_energy.internalmass

# coding=utf-8
"""Room internal mass, including construction and surface area."""
from __future__ import division

from honeybee._lockable import lockable
from honeybee.typing import valid_ep_string, float_positive
from honeybee.units import conversion_factor_to_meters

from .construction.opaque import OpaqueConstruction
from .material.opaque import EnergyMaterialVegetation
from .reader import parse_idf_string
from .writer import generate_idf_string


[docs] @lockable class InternalMass(object): """Room internal mass, including construction and surface area. Note that internal masses assigned this way cannot "see" solar radiation that may potentially hit them and, as such, caution should be taken when using this component with internal mass objects that are not always in shade. Masses are factored into the the thermal calculations of the Room by undergoing heat transfer with the indoor air. Args: identifier: Text string for a unique InternalMass ID. Must be < 100 characters and not contain any EnergyPlus special characters. This will be used to identify the object across a model and in the exported IDF. construction: An OpaqueConstruction object that represents the material that the internal thermal mass is composed of. area: A number representing the surface area of the internal mass that is exposed to the Room air. This value should always be in square meters regardless of what units system the parent model is a part of. Properties: * identifier * display_name * construction * area * user_data """ __slots__ = ('_identifier', '_display_name', '_construction', '_area', '_locked', '_user_data') def __init__(self, identifier, construction, area): self._locked = False # unlocked by default self.identifier = identifier self._display_name = None self.construction = construction self.area = area self._user_data = None @property def identifier(self): """Get or set the text string for object identifier.""" return self._identifier @identifier.setter def identifier(self, identifier): self._identifier = valid_ep_string(identifier) @property def display_name(self): """Get or set a string for the object name without any character restrictions. If not set, this will be equal to the identifier. """ if self._display_name is None: return self._identifier return self._display_name @display_name.setter def display_name(self, value): if value is not None: try: value = str(value) except UnicodeEncodeError: # Python 2 machine lacking the character set pass # keep it as unicode self._display_name = value @property def construction(self): """Get or set a Construction for the material the internal mass is composed of. """ return self._construction @construction.setter def construction(self, value): assert isinstance(value, OpaqueConstruction), \ 'Expected Opaque Construction for InternalMass. Got {}'.format(type(value)) assert not isinstance(value.materials[0], EnergyMaterialVegetation), \ 'InternalMass constructions cannot contain vegetation materials' value.lock() # lock editing in case construction has multiple references self._construction = value @property def area(self): """Get or set a number for the surface area of the mass exposed to the Room air. """ return self._area @area.setter def area(self, value): self._area = float_positive(value, 'internal mass area') @property def user_data(self): """Get or set an optional dictionary for additional meta data for this object. This will be None until it has been set. All keys and values of this dictionary should be of a standard Python type to ensure correct serialization of the object to/from JSON (eg. str, float, int, list, dict) """ if self._user_data is not None: return self._user_data @user_data.setter def user_data(self, value): if value is not None: assert isinstance(value, dict), 'Expected dictionary for honeybee_energy' \ 'object user_data. Got {}.'.format(type(value)) self._user_data = value
[docs] @classmethod def from_geometry(cls, identifier, construction, geometry, units='Meters'): """Create an InternalMass object from a list of geometries. Args: identifier: Text string for a unique InternalMass ID. Must be < 100 characters and not contain any EnergyPlus special characters. This will be used to identify the object across a model and in the exported IDF. construction: An OpaqueConstruction object that represents the material that the internal thermal mass is composed of. geometry: An array of Face3D representing the exposed surface of the internal mass. Note that these Face3D are assumed to be one-sided so, if they are meant to represent a 2-sided object, the Face3D should be duplicated and offset. units: Text for the units system of the geometry. Choose from the following: * Meters * Millimeters * Feet * Inches * Centimeters """ area = sum(geo.area for geo in geometry) * conversion_factor_to_meters(units) return cls(identifier, construction, area)
[docs] @classmethod def from_idf(cls, idf_string, construction_dict): """Create an InternalMass object from an EnergyPlus IDF text string. Args: idf_string: A text string fully describing an EnergyPlus InternalMass definition. construction_dict: A dictionary with construction identifiers as keys and honeybee construction objects as values. This will be used to assign the construction to the InternalMass object. Returns: An InternalMass object loaded from the idf_string. """ ep_strs = parse_idf_string(idf_string, 'InternalMass,') obj_id = ep_strs[0].split('..')[0] return cls(obj_id, construction_dict[ep_strs[1]], ep_strs[4])
[docs] @classmethod def from_dict(cls, data): """Create an InternalMass object from a dictionary. Note that the dictionary must be a non-abridged version for this classmethod to work. Args: data: An InternalMass dictionary in following the format below. .. code-block:: python { "type": 'InternalMass', "identifier": 'Kitchen_Table_Wood_050', "display_name": 'Kitchen Table', "construction": {}, # OpaqueConstruction definition "area": 5 # surface area of internal mass in square meters } """ assert data['type'] == 'InternalMass', \ 'Expected InternalMass dictionary. Got {}.'.format(data['type']) constr = OpaqueConstruction.from_dict(data['construction']) new_obj = cls(data['identifier'], constr, data['area']) if 'display_name' in data and data['display_name'] is not None: new_obj.display_name = data['display_name'] if 'user_data' in data and data['user_data'] is not None: new_obj.user_data = data['user_data'] return new_obj
[docs] @classmethod def from_dict_abridged(cls, data, construction_dict): """Create a InternalMass from an abridged dictionary. Args: data: An InternalMassAbridged dictionary. construction_dict: A dictionary with construction identifiers as keys and honeybee construction objects as values. This will be used to assign the construction to the InternalMass object. .. code-block:: python { "type": 'InternalMassAbridged', "identifier": 'Kitchen_Table_Wood_050', "display_name": 'Kitchen Table', "construction": 'Hardwood_050' # OpaqueConstruction identifier "area": 5 # surface area of internal mass in square meters } """ assert data['type'] == 'InternalMassAbridged', \ 'Expected InternalMassAbridged dictionary. Got {}.'.format(data['type']) new_obj = cls( data['identifier'], construction_dict[data['construction']], data['area']) if 'display_name' in data and data['display_name'] is not None: new_obj.display_name = data['display_name'] if 'user_data' in data and data['user_data'] is not None: new_obj.user_data = data['user_data'] return new_obj
[docs] def to_idf(self, room_identifier, is_zone=True): """IDF string representation of InternalMass object. Note that this method only outputs a single string for the InternalMass object and, to write everything needed to describe the object into an IDF, this object's construction must also be written. Args: room_identifier: Text for the Room identifier that the InternalMass object is assigned to. is_zone: Boolean for whether the room_identifier refers to a Zone object or a Space object. (Default: True). .. code-block:: shell InternalMass, Zn002:IntM001, !- Surface Name INTERIOR, !- Construction Name DORM ROOMS AND COMMON AREAS, !- Zone or ZoneList Name , !- Space or SpaceList Name 408.7734; !- Total area exposed to Zone {m2} """ int_m_id = '{}..{}'.format(self.identifier, room_identifier) values = [int_m_id, self.construction.identifier] if is_zone: values.extend([room_identifier, '', self.area]) else: values.extend(['', room_identifier, self.area]) comments = ('name', 'construction name', 'zone name', 'space name', 'surface area') return generate_idf_string('InternalMass', values, comments)
[docs] def to_dict(self, abridged=False): """InternalMass dictionary representation. Args: abridged: Boolean to note whether the full dictionary describing the object should be returned (False) or just an abridged version (True), which only specifies the identifiers of the construction. (Default: False). """ base = {'type': 'InternalMass'} if not abridged \ else {'type': 'InternalMassAbridged'} base['identifier'] = self.identifier base['construction'] = self.construction.to_dict() if not \ abridged else self.construction.identifier base['area'] = self.area if self._display_name is not None: base['display_name'] = self.display_name if self._user_data is not None: base['user_data'] = self.user_data return base
[docs] def duplicate(self): """Get a copy of this object.""" return self.__copy__()
def __copy__(self): new_obj = InternalMass(self.identifier, self.construction, self.area) new_obj._display_name = self._display_name new_obj._user_data = None if self._user_data is None else self._user_data.copy() return new_obj def __key(self): """A tuple based on the object properties, useful for hashing.""" return (self.identifier, hash(self.construction), self.area) def __hash__(self): return hash(self.__key()) def __eq__(self, other): return isinstance(other, InternalMass) and self.__key() == other.__key() def __ne__(self, other): return not self.__eq__(other)
[docs] def ToString(self): """Overwrite .NET ToString.""" return self.__repr__()
def __repr__(self): return 'InternalMass: {}'.format(self.display_name)